mirror of
https://github.com/magefree/mage.git
synced 2025-12-20 02:30:08 -08:00
[TLA] Implement Lost Days
This commit is contained in:
parent
7f85e6ef3f
commit
886dd1f0b2
5 changed files with 74 additions and 89 deletions
|
|
@ -3,16 +3,12 @@ package mage.cards.d;
|
||||||
import mage.abilities.Ability;
|
import mage.abilities.Ability;
|
||||||
import mage.abilities.common.SimpleStaticAbility;
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
import mage.abilities.dynamicvalue.common.CardsDrawnThisTurnDynamicValue;
|
import mage.abilities.dynamicvalue.common.CardsDrawnThisTurnDynamicValue;
|
||||||
import mage.abilities.effects.OneShotEffect;
|
import mage.abilities.effects.common.PutOnTopOrBottomLibraryTargetEffect;
|
||||||
import mage.abilities.effects.common.cost.SpellCostReductionSourceEffect;
|
import mage.abilities.effects.common.cost.SpellCostReductionSourceEffect;
|
||||||
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.Zone;
|
import mage.constants.Zone;
|
||||||
import mage.game.Game;
|
|
||||||
import mage.game.permanent.Permanent;
|
|
||||||
import mage.players.Player;
|
|
||||||
import mage.target.common.TargetNonlandPermanent;
|
import mage.target.common.TargetNonlandPermanent;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
@ -33,7 +29,7 @@ public final class DeemInferior extends CardImpl {
|
||||||
this.addAbility(ability.addHint(CardsDrawnThisTurnDynamicValue.getHint()));
|
this.addAbility(ability.addHint(CardsDrawnThisTurnDynamicValue.getHint()));
|
||||||
|
|
||||||
// The owner of target nonland permanent puts it into their library second from the top or on the bottom.
|
// The owner of target nonland permanent puts it into their library second from the top or on the bottom.
|
||||||
this.getSpellAbility().addEffect(new DeemInferiorEffect());
|
this.getSpellAbility().addEffect(new PutOnTopOrBottomLibraryTargetEffect(2, true));
|
||||||
this.getSpellAbility().addTarget(new TargetNonlandPermanent());
|
this.getSpellAbility().addTarget(new TargetNonlandPermanent());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -46,40 +42,3 @@ public final class DeemInferior extends CardImpl {
|
||||||
return new DeemInferior(this);
|
return new DeemInferior(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Same as Temporal Cleansing.
|
|
||||||
class DeemInferiorEffect extends OneShotEffect {
|
|
||||||
|
|
||||||
DeemInferiorEffect() {
|
|
||||||
super(Outcome.Benefit);
|
|
||||||
staticText = "the owner of target nonland permanent puts it into their library second from the top or on the bottom";
|
|
||||||
}
|
|
||||||
|
|
||||||
private DeemInferiorEffect(final DeemInferiorEffect effect) {
|
|
||||||
super(effect);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public DeemInferiorEffect copy() {
|
|
||||||
return new DeemInferiorEffect(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean apply(Game game, Ability source) {
|
|
||||||
Permanent permanent = game.getPermanent(getTargetPointer().getFirst(game, source));
|
|
||||||
if (permanent == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
Player player = game.getPlayer(permanent.getOwnerId());
|
|
||||||
if (player == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (player.chooseUse(
|
|
||||||
outcome, "Put " + permanent.getIdName() + " second from the top or on the bottom?",
|
|
||||||
null, "Second from top", "Bottom", source, game
|
|
||||||
)) {
|
|
||||||
return player.putCardOnTopXOfLibrary(permanent, game, source, 2, true);
|
|
||||||
}
|
|
||||||
return player.putCardsOnBottomOfLibrary(permanent, game, source);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
39
Mage.Sets/src/mage/cards/l/LostDays.java
Normal file
39
Mage.Sets/src/mage/cards/l/LostDays.java
Normal file
|
|
@ -0,0 +1,39 @@
|
||||||
|
package mage.cards.l;
|
||||||
|
|
||||||
|
import mage.abilities.effects.common.CreateTokenEffect;
|
||||||
|
import mage.abilities.effects.common.PutOnTopOrBottomLibraryTargetEffect;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.filter.StaticFilters;
|
||||||
|
import mage.game.permanent.token.ClueArtifactToken;
|
||||||
|
import mage.target.TargetPermanent;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class LostDays extends CardImpl {
|
||||||
|
|
||||||
|
public LostDays(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.INSTANT}, "{4}{U}");
|
||||||
|
|
||||||
|
this.subtype.add(SubType.LESSON);
|
||||||
|
|
||||||
|
// The owner of target creature or enchantment puts it into their library second from the top or on the bottom. You create a Clue token.
|
||||||
|
this.getSpellAbility().addEffect(new PutOnTopOrBottomLibraryTargetEffect(2, true));
|
||||||
|
this.getSpellAbility().addEffect(new CreateTokenEffect(new ClueArtifactToken()).concatBy("You"));
|
||||||
|
this.getSpellAbility().addTarget(new TargetPermanent(StaticFilters.FILTER_PERMANENT_CREATURE_OR_ENCHANTMENT));
|
||||||
|
}
|
||||||
|
|
||||||
|
private LostDays(final LostDays card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public LostDays copy() {
|
||||||
|
return new LostDays(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -1,15 +1,10 @@
|
||||||
package mage.cards.t;
|
package mage.cards.t;
|
||||||
|
|
||||||
import mage.abilities.Ability;
|
import mage.abilities.effects.common.PutOnTopOrBottomLibraryTargetEffect;
|
||||||
import mage.abilities.effects.OneShotEffect;
|
|
||||||
import mage.abilities.keyword.ConvokeAbility;
|
import mage.abilities.keyword.ConvokeAbility;
|
||||||
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.game.Game;
|
|
||||||
import mage.game.permanent.Permanent;
|
|
||||||
import mage.players.Player;
|
|
||||||
import mage.target.common.TargetNonlandPermanent;
|
import mage.target.common.TargetNonlandPermanent;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
|
@ -26,7 +21,7 @@ public final class TemporalCleansing extends CardImpl {
|
||||||
this.addAbility(new ConvokeAbility());
|
this.addAbility(new ConvokeAbility());
|
||||||
|
|
||||||
// The owner of target nonland permanent puts it into their library second from the top or on the bottom.
|
// The owner of target nonland permanent puts it into their library second from the top or on the bottom.
|
||||||
this.getSpellAbility().addEffect(new TemporalCleansingEffect());
|
this.getSpellAbility().addEffect(new PutOnTopOrBottomLibraryTargetEffect(2, true));
|
||||||
this.getSpellAbility().addTarget(new TargetNonlandPermanent());
|
this.getSpellAbility().addTarget(new TargetNonlandPermanent());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -39,39 +34,3 @@ public final class TemporalCleansing extends CardImpl {
|
||||||
return new TemporalCleansing(this);
|
return new TemporalCleansing(this);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class TemporalCleansingEffect extends OneShotEffect {
|
|
||||||
|
|
||||||
TemporalCleansingEffect() {
|
|
||||||
super(Outcome.Benefit);
|
|
||||||
staticText = "the owner of target nonland permanent puts it into their library second from the top or on the bottom";
|
|
||||||
}
|
|
||||||
|
|
||||||
private TemporalCleansingEffect(final TemporalCleansingEffect effect) {
|
|
||||||
super(effect);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public TemporalCleansingEffect copy() {
|
|
||||||
return new TemporalCleansingEffect(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean apply(Game game, Ability source) {
|
|
||||||
Permanent permanent = game.getPermanent(getTargetPointer().getFirst(game, source));
|
|
||||||
if (permanent == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
Player player = game.getPlayer(permanent.getOwnerId());
|
|
||||||
if (player == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
if (player.chooseUse(
|
|
||||||
outcome, "Put " + permanent.getIdName() + " second from the top or on the bottom?",
|
|
||||||
null, "Second from top", "Bottom", source, game
|
|
||||||
)) {
|
|
||||||
return player.putCardOnTopXOfLibrary(permanent, game, source, 2, true);
|
|
||||||
}
|
|
||||||
return player.putCardsOnBottomOfLibrary(permanent, game, source);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
||||||
|
|
@ -230,6 +230,7 @@ public final class AvatarTheLastAirbender extends ExpansionSet {
|
||||||
cards.add(new SetCardInfo("Lightning Strike", 146, Rarity.COMMON, mage.cards.l.LightningStrike.class));
|
cards.add(new SetCardInfo("Lightning Strike", 146, Rarity.COMMON, mage.cards.l.LightningStrike.class));
|
||||||
cards.add(new SetCardInfo("Lo and Li, Twin Tutors", 108, Rarity.UNCOMMON, mage.cards.l.LoAndLiTwinTutors.class));
|
cards.add(new SetCardInfo("Lo and Li, Twin Tutors", 108, Rarity.UNCOMMON, mage.cards.l.LoAndLiTwinTutors.class));
|
||||||
cards.add(new SetCardInfo("Long Feng, Grand Secretariat", 233, Rarity.UNCOMMON, mage.cards.l.LongFengGrandSecretariat.class));
|
cards.add(new SetCardInfo("Long Feng, Grand Secretariat", 233, Rarity.UNCOMMON, mage.cards.l.LongFengGrandSecretariat.class));
|
||||||
|
cards.add(new SetCardInfo("Lost Days", 62, Rarity.COMMON, mage.cards.l.LostDays.class));
|
||||||
cards.add(new SetCardInfo("Mai, Jaded Edge", 147, Rarity.UNCOMMON, mage.cards.m.MaiJadedEdge.class));
|
cards.add(new SetCardInfo("Mai, Jaded Edge", 147, Rarity.UNCOMMON, mage.cards.m.MaiJadedEdge.class));
|
||||||
cards.add(new SetCardInfo("Mai, Scornful Striker", 109, Rarity.RARE, mage.cards.m.MaiScornfulStriker.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Mai, Scornful Striker", 109, Rarity.RARE, mage.cards.m.MaiScornfulStriker.class, NON_FULL_USE_VARIOUS));
|
||||||
cards.add(new SetCardInfo("Mai, Scornful Striker", 374, Rarity.RARE, mage.cards.m.MaiScornfulStriker.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Mai, Scornful Striker", 374, Rarity.RARE, mage.cards.m.MaiScornfulStriker.class, NON_FULL_USE_VARIOUS));
|
||||||
|
|
|
||||||
|
|
@ -6,21 +6,29 @@ import mage.abilities.effects.OneShotEffect;
|
||||||
import mage.constants.Outcome;
|
import mage.constants.Outcome;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
import mage.players.Player;
|
import mage.players.Player;
|
||||||
|
import mage.util.CardUtil;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author TheElk801
|
* @author TheElk801
|
||||||
*/
|
*/
|
||||||
public class PutOnTopOrBottomLibraryTargetEffect extends OneShotEffect {
|
public class PutOnTopOrBottomLibraryTargetEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
private final int position;
|
||||||
private final boolean textOwnerOf;
|
private final boolean textOwnerOf;
|
||||||
|
|
||||||
public PutOnTopOrBottomLibraryTargetEffect(boolean textOwnerOf) {
|
public PutOnTopOrBottomLibraryTargetEffect(boolean textOwnerOf) {
|
||||||
|
this(1, textOwnerOf);
|
||||||
|
}
|
||||||
|
|
||||||
|
public PutOnTopOrBottomLibraryTargetEffect(int position, boolean textOwnerOf) {
|
||||||
super(Outcome.ReturnToHand);
|
super(Outcome.ReturnToHand);
|
||||||
|
this.position = position;
|
||||||
this.textOwnerOf = textOwnerOf;
|
this.textOwnerOf = textOwnerOf;
|
||||||
}
|
}
|
||||||
|
|
||||||
private PutOnTopOrBottomLibraryTargetEffect(final PutOnTopOrBottomLibraryTargetEffect effect) {
|
private PutOnTopOrBottomLibraryTargetEffect(final PutOnTopOrBottomLibraryTargetEffect effect) {
|
||||||
super(effect);
|
super(effect);
|
||||||
|
this.position = effect.position;
|
||||||
this.textOwnerOf = effect.textOwnerOf;
|
this.textOwnerOf = effect.textOwnerOf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -35,10 +43,14 @@ public class PutOnTopOrBottomLibraryTargetEffect extends OneShotEffect {
|
||||||
if (player == null) {
|
if (player == null) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
String message = position > 1 ? "into your library " + CardUtil.numberToOrdinalText(position) + " from the top or on the" : "on the top or";
|
||||||
boolean onTop = player.chooseUse(
|
boolean onTop = player.chooseUse(
|
||||||
Outcome.Detriment, "Put the targeted object on the top or bottom of your library?",
|
Outcome.Detriment, "Put the targeted object " + message + " bottom of your library?",
|
||||||
null, "Top", "Bottom", source, game
|
null, "Top", "Bottom", source, game
|
||||||
);
|
);
|
||||||
|
if (onTop && position > 1) {
|
||||||
|
return new PutIntoLibraryNFromTopTargetEffect(position).apply(game, source);
|
||||||
|
}
|
||||||
return new PutOnLibraryTargetEffect(onTop).apply(game, source);
|
return new PutOnLibraryTargetEffect(onTop).apply(game, source);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -47,8 +59,23 @@ public class PutOnTopOrBottomLibraryTargetEffect extends OneShotEffect {
|
||||||
if (staticText != null && !staticText.isEmpty()) {
|
if (staticText != null && !staticText.isEmpty()) {
|
||||||
return staticText;
|
return staticText;
|
||||||
}
|
}
|
||||||
|
StringBuilder sb = new StringBuilder();
|
||||||
String targetText = getTargetPointer().describeTargets(mode.getTargets(), "that permanent");
|
String targetText = getTargetPointer().describeTargets(mode.getTargets(), "that permanent");
|
||||||
return (textOwnerOf ? "the owner of " + targetText : targetText + "'s owner") +
|
if (textOwnerOf) {
|
||||||
" puts it on their choice of the top or bottom of their library";
|
sb.append("the owner of ");
|
||||||
|
sb.append(targetText);
|
||||||
|
} else {
|
||||||
|
sb.append(targetText);
|
||||||
|
sb.append("'s owner");
|
||||||
|
}
|
||||||
|
sb.append(" puts it");
|
||||||
|
if (position > 1) {
|
||||||
|
sb.append("into their library ");
|
||||||
|
sb.append(CardUtil.numberToOrdinalText(position));
|
||||||
|
sb.append(" from the top or on the bottom");
|
||||||
|
} else {
|
||||||
|
sb.append("on their choice of the top or bottom of their library");
|
||||||
|
}
|
||||||
|
return sb.toString();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue