diff --git a/Mage.Sets/src/mage/cards/g/GeralfTheFleshwright.java b/Mage.Sets/src/mage/cards/g/GeralfTheFleshwright.java new file mode 100644 index 00000000000..4184498b3a9 --- /dev/null +++ b/Mage.Sets/src/mage/cards/g/GeralfTheFleshwright.java @@ -0,0 +1,210 @@ +package mage.cards.g; + +import mage.MageInt; +import mage.MageObjectReference; +import mage.abilities.Ability; +import mage.abilities.TriggeredAbilityImpl; +import mage.abilities.common.EntersBattlefieldControlledTriggeredAbility; +import mage.abilities.dynamicvalue.DynamicValue; +import mage.abilities.effects.Effect; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.CreateTokenEffect; +import mage.abilities.effects.common.counter.AddCountersTargetEffect; +import mage.abilities.hint.Hint; +import mage.abilities.hint.ValueHint; +import mage.cards.CardImpl; +import mage.cards.CardSetInfo; +import mage.constants.*; +import mage.counters.CounterType; +import mage.filter.FilterPermanent; +import mage.game.Game; +import mage.game.events.GameEvent; +import mage.game.permanent.Permanent; +import mage.game.permanent.token.ZombieRogueToken; +import mage.players.Player; +import mage.watchers.Watcher; +import mage.watchers.common.SpellsCastWatcher; + +import java.util.*; + +/** + * @author Susucr + */ +public final class GeralfTheFleshwright extends CardImpl { + + private static final FilterPermanent filter = new FilterPermanent(SubType.ZOMBIE, "Zombie"); + private static final Hint hint = new ValueHint("Number of Zombies that entered this turn", GeralfTheFleshwrightValue.instance); + + public GeralfTheFleshwright(UUID ownerId, CardSetInfo setInfo) { + super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{2}{U}"); + + this.supertype.add(SuperType.LEGENDARY); + this.subtype.add(SubType.HUMAN); + this.subtype.add(SubType.WARLOCK); + this.power = new MageInt(2); + this.toughness = new MageInt(3); + + // Whenever you cast a spell during your turn other than your first spell that turn, create a 2/2 blue and black Zombie Rogue creature token. + this.addAbility(new GeralfTheFleshwrightTriggeredAbility()); + + // Whenever a Zombie enters the battlefield under your control, put a +1/+1 counter on it for each other Zombie that entered the battlefield under your control this turn. + this.addAbility(new EntersBattlefieldControlledTriggeredAbility( + Zone.BATTLEFIELD, + new GeralfTheFleshwrightEffect(), + filter, + false, + SetTargetPointer.PERMANENT + ).addHint(hint), new GeralfTheFleshwrightWatcher()); + } + + private GeralfTheFleshwright(final GeralfTheFleshwright card) { + super(card); + } + + @Override + public GeralfTheFleshwright copy() { + return new GeralfTheFleshwright(this); + } +} + + +class GeralfTheFleshwrightTriggeredAbility extends TriggeredAbilityImpl { + + public GeralfTheFleshwrightTriggeredAbility() { + super(Zone.BATTLEFIELD, new CreateTokenEffect(new ZombieRogueToken())); + setTriggerPhrase("Whenever you cast a spell during your turn other than your first spell each turn, "); + } + + private GeralfTheFleshwrightTriggeredAbility(final GeralfTheFleshwrightTriggeredAbility ability) { + super(ability); + } + + @Override + public GeralfTheFleshwrightTriggeredAbility copy() { + return new GeralfTheFleshwrightTriggeredAbility(this); + } + + @Override + public boolean checkEventType(GameEvent event, Game game) { + return event.getType() == GameEvent.EventType.SPELL_CAST; + } + + @Override + public boolean checkTrigger(GameEvent event, Game game) { + if (event.getPlayerId().equals(getControllerId()) && game.isActivePlayer(getControllerId())) { + SpellsCastWatcher watcher = game.getState().getWatcher(SpellsCastWatcher.class); + return watcher != null && watcher.getSpellsCastThisTurn(event.getPlayerId()).size() > 1; + } + return false; + } +} + + +class GeralfTheFleshwrightWatcher extends Watcher { + + // player -> MOR of zombies that entered this turn under that player control. + private final Map> enteredThisTurn = new HashMap<>(); + + GeralfTheFleshwrightWatcher() { + super(WatcherScope.GAME); + } + + @Override + public void watch(GameEvent event, Game game) { + if (event.getType() != GameEvent.EventType.ENTERS_THE_BATTLEFIELD) { + return; + } + + Permanent permanent = game.getPermanentOrLKIBattlefield(event.getTargetId()); + if (permanent == null || !permanent.getSubtype().contains(SubType.ZOMBIE)) { + return; + } + + Player controller = game.getPlayer(event.getPlayerId()); + if (controller == null) { + return; + } + enteredThisTurn.computeIfAbsent(controller.getId(), (k -> new HashSet<>())) + .add(new MageObjectReference(permanent, game)); + } + + @Override + public void reset() { + super.reset(); + enteredThisTurn.clear(); + } + + int getZombiesThatEnteredThisTurn(UUID playerId, MageObjectReference toExclude) { + if (toExclude == null) { + return enteredThisTurn + .getOrDefault(playerId, new HashSet<>()) + .size(); + } else { + return enteredThisTurn + .getOrDefault(playerId, new HashSet<>()) + .stream() + .filter(mor -> !toExclude.equals(mor)) + .mapToInt(x -> 1) + .sum(); + } + } +} + +enum GeralfTheFleshwrightValue implements DynamicValue { + instance; + + @Override + public int calculate(Game game, Ability sourceAbility, Effect effect) { + GeralfTheFleshwrightWatcher watcher = game.getState().getWatcher(GeralfTheFleshwrightWatcher.class); + Player controller = game.getPlayer(sourceAbility.getControllerId()); + if (watcher == null || controller == null) { + return 0; + } + return watcher.getZombiesThatEnteredThisTurn(controller.getId(), null); + } + + @Override + public GeralfTheFleshwrightValue copy() { + return instance; + } + + @Override + public String getMessage() { + return "Number of Zombies that entered this turn"; + } +} + +class GeralfTheFleshwrightEffect extends OneShotEffect { + + GeralfTheFleshwrightEffect() { + super(Outcome.BoostCreature); + } + + private GeralfTheFleshwrightEffect(final GeralfTheFleshwrightEffect effect) { + super(effect); + } + + @Override + public GeralfTheFleshwrightEffect copy() { + return new GeralfTheFleshwrightEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + GeralfTheFleshwrightWatcher watcher = game.getState().getWatcher(GeralfTheFleshwrightWatcher.class); + if (watcher == null) { + return false; + } + Permanent permanent = game.getPermanent(getTargetPointer().getFirst(game, source)); + if (permanent == null) { + return false; + } + int count = watcher.getZombiesThatEnteredThisTurn(source.getControllerId(), new MageObjectReference(permanent, game)); + if (count > 0) { + new AddCountersTargetEffect(CounterType.P1P1.createInstance(count)) + .setTargetPointer(getTargetPointer().copy()) + .apply(game, source); + } + return true; + } +} \ No newline at end of file diff --git a/Mage.Sets/src/mage/sets/OutlawsOfThunderJunction.java b/Mage.Sets/src/mage/sets/OutlawsOfThunderJunction.java index 4a9b876e39b..a2329ae003f 100644 --- a/Mage.Sets/src/mage/sets/OutlawsOfThunderJunction.java +++ b/Mage.Sets/src/mage/sets/OutlawsOfThunderJunction.java @@ -89,6 +89,7 @@ public final class OutlawsOfThunderJunction extends ExpansionSet { cards.add(new SetCardInfo("Freestrider Lookout", 163, Rarity.RARE, mage.cards.f.FreestriderLookout.class)); cards.add(new SetCardInfo("Frontier Seeker", 13, Rarity.UNCOMMON, mage.cards.f.FrontierSeeker.class)); cards.add(new SetCardInfo("Full Steam Ahead", 164, Rarity.UNCOMMON, mage.cards.f.FullSteamAhead.class)); + cards.add(new SetCardInfo("Geralf, the Fleshwright", 50, Rarity.MYTHIC, mage.cards.g.GeralfTheFleshwright.class)); cards.add(new SetCardInfo("Ghired, Mirror of the Wilds", 205, Rarity.MYTHIC, mage.cards.g.GhiredMirrorOfTheWilds.class)); cards.add(new SetCardInfo("Gila Courser", 124, Rarity.UNCOMMON, mage.cards.g.GilaCourser.class)); cards.add(new SetCardInfo("Gisa, the Hellraiser", 89, Rarity.MYTHIC, mage.cards.g.GisaTheHellraiser.class, NON_FULL_USE_VARIOUS));