mirror of
https://github.com/correl/mage.git
synced 2025-01-13 19:11:33 +00:00
Merge branch 'master' of https://github.com/magefree/mage into copy_constructor_watchers
This commit is contained in:
commit
998c3be2a5
4 changed files with 266 additions and 1 deletions
119
Mage.Sets/src/mage/cards/t/TheBindingOfTheTitans.java
Normal file
119
Mage.Sets/src/mage/cards/t/TheBindingOfTheTitans.java
Normal file
|
@ -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;
|
||||||
|
}
|
||||||
|
}
|
144
Mage.Sets/src/mage/cards/t/TymaretCallsTheDead.java
Normal file
144
Mage.Sets/src/mage/cards/t/TymaretCallsTheDead.java
Normal file
|
@ -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);
|
||||||
|
}
|
||||||
|
}
|
|
@ -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("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("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 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("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 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("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));
|
cards.add(new SetCardInfo("Victory's Envoy", 289, Rarity.RARE, mage.cards.v.VictorysEnvoy.class));
|
||||||
|
|
|
@ -86,7 +86,7 @@ public class SearchLibraryPutInPlayEffect extends SearchEffect {
|
||||||
}
|
}
|
||||||
sb.append(target.getTargetName()).append(" and put them onto the battlefield");
|
sb.append(target.getTargetName()).append(" and put them onto the battlefield");
|
||||||
} else {
|
} 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(target.getTargetName())
|
||||||
.append(" and put it onto the battlefield");
|
.append(" and put it onto the battlefield");
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue