diff --git a/Mage.Sets/src/mage/cards/t/TheBindingOfTheTitans.java b/Mage.Sets/src/mage/cards/t/TheBindingOfTheTitans.java new file mode 100644 index 0000000000..a6c8366987 --- /dev/null +++ b/Mage.Sets/src/mage/cards/t/TheBindingOfTheTitans.java @@ -0,0 +1,119 @@ +package mage.cards.t; + +import mage.abilities.Ability; +import mage.abilities.common.SagaAbility; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.PutTopCardOfLibraryIntoGraveEachPlayerEffect; +import mage.abilities.effects.common.ReturnFromGraveyardToHandTargetEffect; +import mage.cards.*; +import mage.constants.*; +import mage.filter.FilterCard; +import mage.filter.StaticFilters; +import mage.filter.predicate.Predicates; +import mage.filter.predicate.mageobject.CardTypePredicate; +import mage.game.Game; +import mage.players.Player; +import mage.target.Target; +import mage.target.common.TargetCardInGraveyard; +import mage.target.common.TargetCardInYourGraveyard; + +import java.util.Collection; +import java.util.UUID; +import java.util.stream.Collectors; + +/** + * @author TheElk801 + */ +public final class TheBindingOfTheTitans extends CardImpl { + + private static final FilterCard filter = new FilterCard("creature or land card from your graveyard"); + + static { + filter.add(Predicates.or( + new CardTypePredicate(CardType.CREATURE), + new CardTypePredicate(CardType.LAND) + )); + } + + public TheBindingOfTheTitans(UUID ownerId, CardSetInfo setInfo) { + super(ownerId, setInfo, new CardType[]{CardType.ENCHANTMENT}, "{1}{G}"); + + this.subtype.add(SubType.SAGA); + + // (As this Saga enters and after your draw step, add a lore counter. Sacrifice after III.) + SagaAbility sagaAbility = new SagaAbility(this, SagaChapter.CHAPTER_III); + + // I — Each player puts the top three cards of their library into their graveyard. + sagaAbility.addChapterEffect( + this, SagaChapter.CHAPTER_I, new PutTopCardOfLibraryIntoGraveEachPlayerEffect( + 3, TargetController.ANY + ) + ); + + // II — Exile up to two target cards from graveyards. For each creature card exiled this way, you gain 1 life. + sagaAbility.addChapterEffect( + this, SagaChapter.CHAPTER_II, SagaChapter.CHAPTER_II, new TheBindingOfTheTitansEffect(), + new TargetCardInGraveyard(0, 2, StaticFilters.FILTER_CARD) + ); + + // III — Return target creature or land card from your graveyard to your hand. + sagaAbility.addChapterEffect( + this, SagaChapter.CHAPTER_III, SagaChapter.CHAPTER_III, + new ReturnFromGraveyardToHandTargetEffect(), new TargetCardInYourGraveyard(filter) + ); + this.addAbility(sagaAbility); + } + + private TheBindingOfTheTitans(final TheBindingOfTheTitans card) { + super(card); + } + + @Override + public TheBindingOfTheTitans copy() { + return new TheBindingOfTheTitans(this); + } +} + +class TheBindingOfTheTitansEffect extends OneShotEffect { + + TheBindingOfTheTitansEffect() { + super(Outcome.Benefit); + staticText = "Exile up to two target cards from graveyards. " + + "For each creature card exiled this way, you gain 1 life."; + } + + private TheBindingOfTheTitansEffect(final TheBindingOfTheTitansEffect effect) { + super(effect); + } + + @Override + public TheBindingOfTheTitansEffect copy() { + return new TheBindingOfTheTitansEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player player = game.getPlayer(source.getControllerId()); + if (player == null) { + return false; + } + Cards cards = new CardsImpl( + source.getTargets() + .stream() + .map(Target::getTargets) + .flatMap(Collection::stream) + .collect(Collectors.toList()) + ); + player.moveCards(cards, Zone.EXILED, source, game); + int lifeToGain = cards + .getCards(game) + .stream() + .filter(Card::isCreature) + .map(Card::getId) + .map(game.getState()::getZone) + .map(Zone.EXILED::equals) + .mapToInt(b -> b ? 1 : 0) + .sum(); + return player.gainLife(lifeToGain, game, source) > 0; + } +} \ No newline at end of file diff --git a/Mage.Sets/src/mage/cards/t/TymaretCallsTheDead.java b/Mage.Sets/src/mage/cards/t/TymaretCallsTheDead.java new file mode 100644 index 0000000000..4c1d8e00cd --- /dev/null +++ b/Mage.Sets/src/mage/cards/t/TymaretCallsTheDead.java @@ -0,0 +1,144 @@ +package mage.cards.t; + +import mage.abilities.Ability; +import mage.abilities.common.SagaAbility; +import mage.abilities.effects.Effect; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.CreateTokenEffect; +import mage.abilities.effects.common.PutTopCardOfLibraryIntoGraveControllerEffect; +import mage.cards.CardImpl; +import mage.cards.CardSetInfo; +import mage.constants.CardType; +import mage.constants.SagaChapter; +import mage.constants.SubType; +import mage.constants.Zone; +import mage.filter.FilterCard; +import mage.filter.FilterPermanent; +import mage.filter.predicate.Predicates; +import mage.filter.predicate.mageobject.CardTypePredicate; +import mage.game.Game; +import mage.game.permanent.token.ZombieToken; +import mage.players.Player; +import mage.target.TargetCard; +import mage.target.common.TargetCardInYourGraveyard; + +import java.util.UUID; + +import static mage.constants.Outcome.Benefit; + +/** + * @author TheElk801 + */ +public final class TymaretCallsTheDead extends CardImpl { + + public TymaretCallsTheDead(UUID ownerId, CardSetInfo setInfo) { + super(ownerId, setInfo, new CardType[]{CardType.ENCHANTMENT}, "{2}{B}"); + + this.subtype.add(SubType.SAGA); + + // (As this Saga enters and after your draw step, add a lore counter. Sacrifice after III.) + SagaAbility sagaAbility = new SagaAbility(this, SagaChapter.CHAPTER_III); + + // I, II — Put the top three cards of your library into your graveyard. Then you may exile a creature or enchantment card from your graveyard. If you do, create a 2/2 black Zombie creature token. + sagaAbility.addChapterEffect( + this, SagaChapter.CHAPTER_I, SagaChapter.CHAPTER_II, new TymaretCallsTheDeadFirstEffect() + ); + + // III — You gain X life and scry X, where X is the number of Zombies you control. + sagaAbility.addChapterEffect(this, SagaChapter.CHAPTER_III, new TymaretCallsTheDeadLastEffect()); + + this.addAbility(sagaAbility); + } + + private TymaretCallsTheDead(final TymaretCallsTheDead card) { + super(card); + } + + @Override + public TymaretCallsTheDead copy() { + return new TymaretCallsTheDead(this); + } +} + +class TymaretCallsTheDeadFirstEffect extends OneShotEffect { + + private static final Effect millEffect = new PutTopCardOfLibraryIntoGraveControllerEffect(3); + private static final Effect tokenEffect = new CreateTokenEffect(new ZombieToken()); + private static final FilterCard filter = new FilterCard("creature or enchantment card from your graveyard"); + + static { + filter.add(Predicates.or( + new CardTypePredicate(CardType.CREATURE), + new CardTypePredicate(CardType.ENCHANTMENT) + )); + } + + TymaretCallsTheDeadFirstEffect() { + super(Benefit); + staticText = "put the top three cards of your library into your graveyard. " + + "Then you may exile a creature or enchantment card from your graveyard. " + + "If you do, create a 2/2 black Zombie creature token"; + } + + private TymaretCallsTheDeadFirstEffect(final TymaretCallsTheDeadFirstEffect effect) { + super(effect); + } + + @Override + public TymaretCallsTheDeadFirstEffect copy() { + return new TymaretCallsTheDeadFirstEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player player = game.getPlayer(source.getControllerId()); + if (player == null) { + return false; + } + millEffect.apply(game, source); + if (player.getGraveyard().count(filter, game) == 0 + || !player.chooseUse(outcome, "Exile a creature or enchantment card from your graveyard?", source, game)) { + return true; + } + TargetCard target = new TargetCardInYourGraveyard(filter); + target.setNotTarget(true); + if (!player.choose(outcome, player.getGraveyard(), target, game)) { + return false; + } + return player.moveCards(game.getCard(target.getFirstTarget()), Zone.EXILED, source, game) + && tokenEffect.apply(game, source); + } +} + +class TymaretCallsTheDeadLastEffect extends OneShotEffect { + + private static final FilterPermanent filter = new FilterPermanent(SubType.ZOMBIE, ""); + + TymaretCallsTheDeadLastEffect() { + super(Benefit); + staticText = "You gain X life and scry X, where X is the number of Zombies you control."; + } + + private TymaretCallsTheDeadLastEffect(final TymaretCallsTheDeadLastEffect effect) { + super(effect); + } + + @Override + public TymaretCallsTheDeadLastEffect copy() { + return new TymaretCallsTheDeadLastEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player player = game.getPlayer(source.getControllerId()); + if (player == null) { + return false; + } + int zombieCount = game.getBattlefield().countAll(filter, source.getControllerId(), game); + if (zombieCount <= 0) { + return true; + } + player.gainLife(zombieCount, game, source); + return player.scry(zombieCount, source, game); + } +} \ No newline at end of file diff --git a/Mage.Sets/src/mage/sets/TherosBeyondDeath.java b/Mage.Sets/src/mage/sets/TherosBeyondDeath.java index aa1d8e6b50..ee2b50445f 100644 --- a/Mage.Sets/src/mage/sets/TherosBeyondDeath.java +++ b/Mage.Sets/src/mage/sets/TherosBeyondDeath.java @@ -62,7 +62,9 @@ public final class TherosBeyondDeath extends ExpansionSet { cards.add(new SetCardInfo("Swamp", 252, Rarity.LAND, mage.cards.basiclands.Swamp.class, FULL_ART_BFZ_VARIOUS)); cards.add(new SetCardInfo("Terror of Mount Velus", 295, Rarity.RARE, mage.cards.t.TerrorOfMountVelus.class)); cards.add(new SetCardInfo("The Akroan War", 124, Rarity.RARE, mage.cards.t.TheAkroanWar.class)); + cards.add(new SetCardInfo("The Binding of the Titans", 166, Rarity.UNCOMMON, mage.cards.t.TheBindingOfTheTitans.class)); cards.add(new SetCardInfo("Treeshaker Chimera", 297, Rarity.RARE, mage.cards.t.TreeshakerChimera.class)); + cards.add(new SetCardInfo("Tymaret Calls the Dead", 118, Rarity.RARE, mage.cards.t.TymaretCallsTheDead.class)); cards.add(new SetCardInfo("Underworld Rage-Hound", 163, Rarity.COMMON, mage.cards.u.UnderworldRageHound.class)); cards.add(new SetCardInfo("Underworld Sentinel", 293, Rarity.RARE, mage.cards.u.UnderworldSentinel.class)); cards.add(new SetCardInfo("Victory's Envoy", 289, Rarity.RARE, mage.cards.v.VictorysEnvoy.class)); diff --git a/Mage/src/main/java/mage/abilities/effects/common/search/SearchLibraryPutInPlayEffect.java b/Mage/src/main/java/mage/abilities/effects/common/search/SearchLibraryPutInPlayEffect.java index 22033b3f87..99b26c22ac 100644 --- a/Mage/src/main/java/mage/abilities/effects/common/search/SearchLibraryPutInPlayEffect.java +++ b/Mage/src/main/java/mage/abilities/effects/common/search/SearchLibraryPutInPlayEffect.java @@ -86,7 +86,7 @@ public class SearchLibraryPutInPlayEffect extends SearchEffect { } sb.append(target.getTargetName()).append(" and put them onto the battlefield"); } else { - sb.append(target.getTargetName().startsWith("a ") || target.getTargetName().startsWith("an ") ? "" : sb.append("a ")) + sb.append(target.getTargetName().startsWith("a ") || target.getTargetName().startsWith("an ") ? "" : "a ") .append(target.getTargetName()) .append(" and put it onto the battlefield"); }