fixed implementation of Tetravus to allow for multiple counters/tokens in a single turn

The tokens were also incorrectly implemented
This commit is contained in:
Evan Kranzler 2018-06-13 20:58:31 -04:00
parent 6bb8b68952
commit 0a635ac341
2 changed files with 157 additions and 41 deletions

View file

@ -1,35 +1,40 @@
package mage.cards.t; package mage.cards.t;
import java.util.HashSet;
import java.util.Set;
import java.util.UUID; import java.util.UUID;
import mage.MageInt; import mage.MageInt;
import mage.MageObject; import mage.MageObjectReference;
import mage.abilities.StaticAbility; import mage.abilities.Ability;
import mage.abilities.common.BeginningOfUpkeepTriggeredAbility; import mage.abilities.common.BeginningOfUpkeepTriggeredAbility;
import mage.abilities.common.EntersBattlefieldAbility; import mage.abilities.common.EntersBattlefieldAbility;
import mage.abilities.condition.common.SourceHasCounterCondition; import mage.abilities.costs.Cost;
import mage.abilities.costs.common.ExileTargetCost; import mage.abilities.costs.common.ExileTargetCost;
import mage.abilities.costs.common.RemoveCountersSourceCost; import mage.abilities.costs.common.RemoveCountersSourceCost;
import mage.abilities.decorator.ConditionalTriggeredAbility; import mage.abilities.effects.OneShotEffect;
import mage.abilities.effects.common.CreateTokenEffect; import mage.abilities.effects.common.CreateTokenEffect;
import mage.abilities.effects.common.DoIfCostPaid;
import mage.abilities.effects.common.counter.AddCountersSourceEffect; import mage.abilities.effects.common.counter.AddCountersSourceEffect;
import mage.abilities.keyword.FlyingAbility; import mage.abilities.keyword.FlyingAbility;
import mage.cards.CardImpl; import mage.cards.CardImpl;
import mage.cards.CardSetInfo; import mage.cards.CardSetInfo;
import mage.constants.CardType; import mage.constants.CardType;
import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.TargetController; import mage.constants.TargetController;
import mage.constants.Zone;
import mage.counters.CounterType; import mage.counters.CounterType;
import mage.filter.common.FilterControlledPermanent; import mage.filter.common.FilterControlledPermanent;
import mage.filter.predicate.Predicate;
import mage.filter.predicate.permanent.TokenPredicate;
import mage.game.Game; import mage.game.Game;
import mage.game.permanent.Permanent;
import mage.game.permanent.token.TetraviteToken; import mage.game.permanent.token.TetraviteToken;
import mage.players.Player;
import mage.target.common.TargetControlledPermanent; import mage.target.common.TargetControlledPermanent;
import mage.util.CardUtil;
/** /**
* *
* @author MarcoMarin * @author TheElk801
*/ */
public final class Tetravus extends CardImpl { public final class Tetravus extends CardImpl {
@ -41,15 +46,24 @@ public final class Tetravus extends CardImpl {
// Flying // Flying
this.addAbility(FlyingAbility.getInstance()); this.addAbility(FlyingAbility.getInstance());
// Tetravus enters the battlefield with three +1/+1 counters on it. // Tetravus enters the battlefield with three +1/+1 counters on it.
this.addAbility(new EntersBattlefieldAbility(new AddCountersSourceEffect(CounterType.P1P1.createInstance(3)), "{this} enters the battlefield with three +1/+1 counters on it")); this.addAbility(new EntersBattlefieldAbility(
new AddCountersSourceEffect(CounterType.P1P1.createInstance(3)),
"with three +1/+1 counters on it"
));
// At the beginning of your upkeep, you may remove any number of +1/+1 counters from Tetravus. If you do, create that many 1/1 colorless Tetravite artifact creature tokens. They each have flying and "This creature can't be enchanted." // At the beginning of your upkeep, you may remove any number of +1/+1 counters from Tetravus. If you do, create that many 1/1 colorless Tetravite artifact creature tokens. They each have flying and "This creature can't be enchanted."
this.addAbility(new ConditionalTriggeredAbility(new BeginningOfUpkeepTriggeredAbility(new DoIfCostPaid(new CreateTokenEffect(new TetraviteToken()), this.addAbility(new BeginningOfUpkeepTriggeredAbility(
new RemoveCountersSourceCost(CounterType.P1P1.createInstance(1))), TargetController.YOU, true), new TetravusCreateTokensEffect(),
new SourceHasCounterCondition(CounterType.P1P1, 1), "At the beginning of your upkeep, you may remove any number of +1/+1 counters from Tetravus. If you do, create that many 1/1 colorless Tetravite artifact creature tokens. They each have flying and \"This creature can't be enchanted.\"")); TargetController.YOU, true
));
// At the beginning of your upkeep, you may exile any number of tokens created with Tetravus. If you do, put that many +1/+1 counters on Tetravus. // At the beginning of your upkeep, you may exile any number of tokens created with Tetravus. If you do, put that many +1/+1 counters on Tetravus.
this.addAbility(new BeginningOfUpkeepTriggeredAbility(new DoIfCostPaid(new AddCountersSourceEffect(CounterType.P1P1.createInstance(1)), this.addAbility(new BeginningOfUpkeepTriggeredAbility(
new ExileTargetCost(new TargetControlledPermanent(new FilterControlledPermanent("Tetravite")))), TargetController.YOU, true)); new TetravusAddCountersEffect(),
TargetController.YOU, true
));
} }
@ -63,27 +77,120 @@ public final class Tetravus extends CardImpl {
} }
} }
class CantBeEnchantedAbility extends StaticAbility { class TetravusCreateTokensEffect extends OneShotEffect {
public CantBeEnchantedAbility() { public TetravusCreateTokensEffect() {
super(Zone.BATTLEFIELD, null); super(Outcome.Benefit);
this.staticText = "remove any number of +1/+1 counters from {this}. "
+ "If you do, create that many 1/1 colorless Tetravite artifact creature tokens. "
+ "They each have flying and \"This creature can't be enchanted.\"";
} }
public CantBeEnchantedAbility(final CantBeEnchantedAbility ability) { public TetravusCreateTokensEffect(final TetravusCreateTokensEffect effect) {
super(ability); super(effect);
} }
@Override @Override
public CantBeEnchantedAbility copy() { public TetravusCreateTokensEffect copy() {
return new CantBeEnchantedAbility(this); return new TetravusCreateTokensEffect(this);
} }
public boolean canTarget(MageObject source, Game game) { @Override
if (source.isEnchantment() public boolean apply(Game game, Ability source) {
&& source.hasSubtype(SubType.AURA, game)) { Player player = game.getPlayer(source.getControllerId());
Permanent permanent = source.getSourcePermanentIfItStillExists(game);
if (player == null || permanent == null) {
return false; return false;
} }
return true; int countersToRemove = permanent.getCounters(game).getCount(CounterType.P1P1);
if (countersToRemove == 0) {
return false;
}
countersToRemove = player.getAmount(0, countersToRemove, "Choose an amount of counters to remove", game);
Cost cost = new RemoveCountersSourceCost(CounterType.P1P1.createInstance(countersToRemove));
if (cost.pay(source, game, source.getSourceId(), source.getControllerId(), true)) {
CreateTokenEffect effect = new CreateTokenEffect(new TetraviteToken(), countersToRemove);
effect.apply(game, source);
Object object = game.getState().getValue(CardUtil.getObjectZoneString("_tokensCreated", permanent, game));
Set<UUID> tokensCreated;
if (object != null) {
tokensCreated = (Set<UUID>) object;
} else {
tokensCreated = new HashSet<>();
}
for (UUID tokenId : effect.getLastAddedTokenIds()) {
if (tokenId != null) {
tokensCreated.add(tokenId);
}
}
game.getState().setValue(CardUtil.getCardZoneString("_tokensCreated", source.getSourceId(), game), tokensCreated);
}
return false;
}
}
class TetravusAddCountersEffect extends OneShotEffect {
public TetravusAddCountersEffect() {
super(Outcome.Benefit);
this.staticText = "exile any number of tokens created with {this}. "
+ "If you do, put that many +1/+1 counters on {this}";
} }
public TetravusAddCountersEffect(final TetravusAddCountersEffect effect) {
super(effect);
}
@Override
public TetravusAddCountersEffect copy() {
return new TetravusAddCountersEffect(this);
}
@Override
public boolean apply(Game game, Ability source) {
Player player = game.getPlayer(source.getControllerId());
Permanent permanent = source.getSourcePermanentIfItStillExists(game);
if (player == null || permanent == null) {
return false;
}
FilterControlledPermanent filter = new FilterControlledPermanent("tokens created with " + permanent.getName());
filter.add(new TetravusPredicate(new MageObjectReference(permanent, game)));
filter.add(new TokenPredicate());
ExileTargetCost cost = new ExileTargetCost(new TargetControlledPermanent(0, Integer.MAX_VALUE, filter, true));
if (cost.pay(source, game, source.getSourceId(), player.getId(), true)) {
return new AddCountersSourceEffect(CounterType.P1P1.createInstance(cost.getPermanents().size())).apply(game, source);
}
return false;
}
}
class TetravusPredicate implements Predicate<Permanent> {
private final MageObjectReference mor;
public TetravusPredicate(MageObjectReference mor) {
this.mor = mor;
}
@Override
public boolean apply(Permanent input, Game game) {
if (mor == null) {
return false;
}
Permanent permanent = mor.getPermanent(game);
if (permanent == null) {
return false;
}
Object object = game.getState().getValue(CardUtil.getObjectZoneString("_tokensCreated", permanent, game));
if (object == null) {
return false;
}
Set<UUID> tokensCreated = (Set<UUID>) object;
return tokensCreated.contains(input.getId());
}
@Override
public String toString() {
return "";
}
} }

View file

@ -1,14 +1,17 @@
package mage.game.permanent.token; package mage.game.permanent.token;
import mage.constants.CardType; import mage.constants.CardType;
import mage.MageInt; import mage.MageInt;
import mage.MageObject; import mage.abilities.Ability;
import mage.abilities.StaticAbility; import mage.abilities.common.SimpleStaticAbility;
import mage.abilities.effects.ContinuousRuleModifyingEffectImpl;
import mage.abilities.keyword.FlyingAbility; import mage.abilities.keyword.FlyingAbility;
import mage.constants.Duration;
import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone; import mage.constants.Zone;
import mage.game.Game; import mage.game.Game;
import mage.game.events.GameEvent;
/** /**
* *
@ -17,7 +20,7 @@ import mage.game.Game;
public final class TetraviteToken extends TokenImpl { public final class TetraviteToken extends TokenImpl {
public TetraviteToken() { public TetraviteToken() {
super("Tetravite", "1/1 colorless Tetravite artifact creature token"); super("Tetravite", "1/1 colorless Tetravite artifact creature token with flying and \"This creature cant be enchanted.\"");
cardType.add(CardType.CREATURE); cardType.add(CardType.CREATURE);
cardType.add(CardType.ARTIFACT); cardType.add(CardType.ARTIFACT);
subtype.add(SubType.TETRAVITE); subtype.add(SubType.TETRAVITE);
@ -25,7 +28,7 @@ public final class TetraviteToken extends TokenImpl {
toughness = new MageInt(1); toughness = new MageInt(1);
this.addAbility(FlyingAbility.getInstance()); this.addAbility(FlyingAbility.getInstance());
this.addAbility(new CantBeEnchantedAbility()); this.addAbility(new SimpleStaticAbility(Zone.BATTLEFIELD, new TetraviteTokenEffect()));
} }
public TetraviteToken(final TetraviteToken token) { public TetraviteToken(final TetraviteToken token) {
@ -37,24 +40,30 @@ public final class TetraviteToken extends TokenImpl {
} }
} }
class CantBeEnchantedAbility extends StaticAbility { class TetraviteTokenEffect extends ContinuousRuleModifyingEffectImpl {
public CantBeEnchantedAbility() { public TetraviteTokenEffect() {
super(Zone.BATTLEFIELD, null); super(Duration.WhileOnBattlefield, Outcome.Detriment);
staticText = "this creature can't be enchanted";
} }
public CantBeEnchantedAbility(final CantBeEnchantedAbility ability) { public TetraviteTokenEffect(final TetraviteTokenEffect effect) {
super(ability); super(effect);
} }
@Override @Override
public CantBeEnchantedAbility copy() { public TetraviteTokenEffect copy() {
return new CantBeEnchantedAbility(this); return new TetraviteTokenEffect(this);
} }
public boolean canTarget(MageObject source, Game game) { @Override
return !(source.isEnchantment() public boolean checksEventType(GameEvent event, Game game) {
&& source.hasSubtype(SubType.AURA, game)); return event.getType() == GameEvent.EventType.ATTACH
|| event.getType() == GameEvent.EventType.STAY_ATTACHED;
} }
@Override
public boolean applies(GameEvent event, Ability source, Game game) {
return event.getTargetId().equals(source.getSourceId());
}
} }