combined "enchanted player's upkeep" triggers into one class

This commit is contained in:
Evan Kranzler 2021-02-23 09:44:26 -05:00
parent 2ffa719278
commit 5736b00f0f
8 changed files with 160 additions and 419 deletions

View file

@ -1,8 +1,7 @@
package mage.cards.c; package mage.cards.c;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.effects.OneShotEffect; import mage.abilities.effects.OneShotEffect;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
import mage.abilities.keyword.EnchantAbility; import mage.abilities.keyword.EnchantAbility;
@ -11,22 +10,18 @@ import mage.cards.CardSetInfo;
import mage.constants.CardType; import mage.constants.CardType;
import mage.constants.Outcome; import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone; import mage.filter.FilterPermanent;
import mage.filter.common.FilterControlledPermanent; import mage.filter.common.FilterControlledPermanent;
import mage.filter.predicate.Predicates; import mage.filter.predicate.Predicates;
import mage.game.Game; import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.events.GameEvent.EventType;
import mage.game.permanent.Permanent; import mage.game.permanent.Permanent;
import mage.players.Player; import mage.players.Player;
import mage.target.TargetPermanent; import mage.target.TargetPermanent;
import mage.target.TargetPlayer; import mage.target.TargetPlayer;
import mage.target.targetpointer.FixedTarget;
import java.util.UUID; import java.util.UUID;
/** /**
*
* @author jeffwadsworth * @author jeffwadsworth
*/ */
public final class CruelReality extends CardImpl { public final class CruelReality extends CardImpl {
@ -43,8 +38,7 @@ public final class CruelReality extends CardImpl {
this.addAbility(new EnchantAbility(auraTarget.getTargetName())); this.addAbility(new EnchantAbility(auraTarget.getTargetName()));
//At the beginning of enchanted player's upkeep, that player sacrifices a creature or planeswalker. If the player can't, they lose 5 life. //At the beginning of enchanted player's upkeep, that player sacrifices a creature or planeswalker. If the player can't, they lose 5 life.
this.addAbility(new CruelRealityTriggeredAbiilty()); this.addAbility(new BeginningOfUpkeepAttachedTriggeredAbility(new CruelRealityEffect()));
} }
private CruelReality(final CruelReality card) { private CruelReality(final CruelReality card) {
@ -57,55 +51,23 @@ public final class CruelReality extends CardImpl {
} }
} }
class CruelRealityTriggeredAbiilty extends TriggeredAbilityImpl {
public CruelRealityTriggeredAbiilty() {
super(Zone.BATTLEFIELD, new CruelRealityEffect());
}
public CruelRealityTriggeredAbiilty(final CruelRealityTriggeredAbiilty ability) {
super(ability);
}
@Override
public CruelRealityTriggeredAbiilty copy() {
return new CruelRealityTriggeredAbiilty(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = game.getPermanent(this.sourceId);
if (enchantment != null
&& enchantment.getAttachedTo() != null) {
Player cursedPlayer = game.getPlayer(enchantment.getAttachedTo());
if (cursedPlayer != null
&& game.isActivePlayer(cursedPlayer.getId())) {
this.getEffects().get(0).setTargetPointer(new FixedTarget(cursedPlayer.getId()));
return true;
}
}
return false;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, " + super.getRule();
}
}
class CruelRealityEffect extends OneShotEffect { class CruelRealityEffect extends OneShotEffect {
public CruelRealityEffect() { private static final FilterPermanent filter = new FilterControlledPermanent("creature or planeswalker");
static {
filter.add(Predicates.or(
CardType.CREATURE.getPredicate(),
CardType.PLANESWALKER.getPredicate()
));
}
CruelRealityEffect() {
super(Outcome.LoseLife); super(Outcome.LoseLife);
staticText = "that player sacrifices a creature or planeswalker. If the player can't, they lose 5 life"; staticText = "that player sacrifices a creature or planeswalker. If the player can't, they lose 5 life";
} }
public CruelRealityEffect(final CruelRealityEffect effect) { private CruelRealityEffect(final CruelRealityEffect effect) {
super(effect); super(effect);
} }
@ -118,24 +80,21 @@ class CruelRealityEffect extends OneShotEffect {
public boolean apply(Game game, Ability source) { public boolean apply(Game game, Ability source) {
Player cursedPlayer = game.getPlayer(targetPointer.getFirst(game, source)); Player cursedPlayer = game.getPlayer(targetPointer.getFirst(game, source));
Player controller = game.getPlayer(source.getControllerId()); Player controller = game.getPlayer(source.getControllerId());
if (cursedPlayer != null if (cursedPlayer == null || controller == null) {
&& controller != null) { return false;
FilterControlledPermanent filter = new FilterControlledPermanent("creature or planeswalker");
filter.add(Predicates.or(
CardType.CREATURE.getPredicate(),
CardType.PLANESWALKER.getPredicate()));
TargetPermanent target = new TargetPermanent(filter);
if (cursedPlayer.choose(Outcome.Sacrifice, target, source.getId(), game)) {
Permanent objectToBeSacrificed = game.getPermanent(target.getFirstTarget());
if (objectToBeSacrificed != null) {
if (objectToBeSacrificed.sacrifice(source, game)) {
return true;
}
}
}
cursedPlayer.loseLife(5, game, source, false);
return true;
} }
return false; TargetPermanent target = new TargetPermanent(filter);
target.setNotTarget(true);
if (target.canChoose(source.getSourceId(), cursedPlayer.getId(), game)
&& cursedPlayer.choose(Outcome.Sacrifice, target, source.getId(), game)) {
Permanent objectToBeSacrificed = game.getPermanent(target.getFirstTarget());
if (objectToBeSacrificed != null) {
if (objectToBeSacrificed.sacrifice(source, game)) {
return true;
}
}
}
cursedPlayer.loseLife(5, game, source, false);
return true;
} }
} }

View file

@ -1,7 +1,7 @@
package mage.cards.c; package mage.cards.c;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
import mage.abilities.effects.common.ExileFromZoneTargetEffect; import mage.abilities.effects.common.ExileFromZoneTargetEffect;
import mage.abilities.keyword.EnchantAbility; import mage.abilities.keyword.EnchantAbility;
@ -12,11 +12,7 @@ import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone; import mage.constants.Zone;
import mage.filter.StaticFilters; import mage.filter.StaticFilters;
import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.permanent.Permanent;
import mage.target.TargetPlayer; import mage.target.TargetPlayer;
import mage.target.targetpointer.FixedTarget;
import java.util.UUID; import java.util.UUID;
@ -37,7 +33,9 @@ public final class CurseOfOblivion extends CardImpl {
this.addAbility(ability); this.addAbility(ability);
// At the beginning of enchanted player's upkeep, that player exiles two cards from their graveyard. // At the beginning of enchanted player's upkeep, that player exiles two cards from their graveyard.
this.addAbility(new CurseOfOblivionAbility()); this.addAbility(new BeginningOfUpkeepAttachedTriggeredAbility(new ExileFromZoneTargetEffect(
Zone.GRAVEYARD, StaticFilters.FILTER_CARD_CARDS, 2, false
).setText("that player exiles two cards from their graveyard")));
} }
private CurseOfOblivion(final CurseOfOblivion card) { private CurseOfOblivion(final CurseOfOblivion card) {
@ -49,39 +47,3 @@ public final class CurseOfOblivion extends CardImpl {
return new CurseOfOblivion(this); return new CurseOfOblivion(this);
} }
} }
class CurseOfOblivionAbility extends TriggeredAbilityImpl {
CurseOfOblivionAbility() {
super(Zone.BATTLEFIELD, new ExileFromZoneTargetEffect(Zone.GRAVEYARD, StaticFilters.FILTER_CARD_CARDS, 2, false));
}
private CurseOfOblivionAbility(final CurseOfOblivionAbility ability) {
super(ability);
}
@Override
public CurseOfOblivionAbility copy() {
return new CurseOfOblivionAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = getSourcePermanentOrLKI(game);
if (enchantment == null || !game.isActivePlayer(enchantment.getAttachedTo())) {
return false;
}
this.getEffects().setTargetPointer(new FixedTarget(enchantment.getAttachedTo()));
return true;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, that player exiles two cards from their graveyard.";
}
}

View file

@ -1,7 +1,7 @@
package mage.cards.c; package mage.cards.c;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
import mage.abilities.effects.common.PutLibraryIntoGraveTargetEffect; import mage.abilities.effects.common.PutLibraryIntoGraveTargetEffect;
import mage.abilities.keyword.EnchantAbility; import mage.abilities.keyword.EnchantAbility;
@ -10,19 +10,11 @@ import mage.cards.CardSetInfo;
import mage.constants.CardType; import mage.constants.CardType;
import mage.constants.Outcome; import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone;
import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.events.GameEvent.EventType;
import mage.game.permanent.Permanent;
import mage.players.Player;
import mage.target.TargetPlayer; import mage.target.TargetPlayer;
import mage.target.targetpointer.FixedTarget;
import java.util.UUID; import java.util.UUID;
/** /**
*
* @author Alvin * @author Alvin
*/ */
public final class CurseOfTheBloodyTome extends CardImpl { public final class CurseOfTheBloodyTome extends CardImpl {
@ -39,8 +31,9 @@ public final class CurseOfTheBloodyTome extends CardImpl {
this.addAbility(ability); this.addAbility(ability);
// At the beginning of enchanted player's upkeep, that player puts the top two cards of their library into their graveyard. // At the beginning of enchanted player's upkeep, that player puts the top two cards of their library into their graveyard.
this.addAbility(new CurseOfTheBloodyTomeAbility()); this.addAbility(new BeginningOfUpkeepAttachedTriggeredAbility(
new PutLibraryIntoGraveTargetEffect(2).setText("that player mills two cards")
));
} }
private CurseOfTheBloodyTome(final CurseOfTheBloodyTome card) { private CurseOfTheBloodyTome(final CurseOfTheBloodyTome card) {
@ -52,42 +45,3 @@ public final class CurseOfTheBloodyTome extends CardImpl {
return new CurseOfTheBloodyTome(this); return new CurseOfTheBloodyTome(this);
} }
} }
class CurseOfTheBloodyTomeAbility extends TriggeredAbilityImpl {
public CurseOfTheBloodyTomeAbility() {
super(Zone.BATTLEFIELD, new PutLibraryIntoGraveTargetEffect(2));
}
public CurseOfTheBloodyTomeAbility(final CurseOfTheBloodyTomeAbility ability) {
super(ability);
}
@Override
public CurseOfTheBloodyTomeAbility copy() {
return new CurseOfTheBloodyTomeAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = game.getPermanent(this.sourceId);
if (enchantment != null && enchantment.getAttachedTo() != null) {
Player player = game.getPlayer(enchantment.getAttachedTo());
if (player != null && game.isActivePlayer(player.getId())) {
this.getEffects().get(0).setTargetPointer(new FixedTarget(player.getId()));
return true;
}
}
return false;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, that player mills two cards.";
}
}

View file

@ -1,7 +1,7 @@
package mage.cards.c; package mage.cards.c;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.effects.OneShotEffect; import mage.abilities.effects.OneShotEffect;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
import mage.abilities.keyword.EnchantAbility; import mage.abilities.keyword.EnchantAbility;
@ -10,17 +10,15 @@ import mage.cards.CardSetInfo;
import mage.constants.CardType; import mage.constants.CardType;
import mage.constants.Outcome; import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone; import mage.filter.FilterPermanent;
import mage.filter.StaticFilters;
import mage.filter.common.FilterPlaneswalkerPermanent; import mage.filter.common.FilterPlaneswalkerPermanent;
import mage.filter.predicate.permanent.ControllerIdPredicate; import mage.filter.predicate.permanent.ControllerIdPredicate;
import mage.game.Game; import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.events.GameEvent.EventType;
import mage.game.permanent.Permanent; import mage.game.permanent.Permanent;
import mage.players.Player; import mage.players.Player;
import mage.target.TargetPermanent; import mage.target.TargetPermanent;
import mage.target.TargetPlayer; import mage.target.TargetPlayer;
import mage.target.targetpointer.FixedTarget;
import java.util.UUID; import java.util.UUID;
@ -41,7 +39,7 @@ public final class CurseOfThePiercedHeart extends CardImpl {
this.addAbility(ability); this.addAbility(ability);
// At the beginning of enchanted player's upkeep, Curse of the Pierced Heart deals 1 damage to that player. // At the beginning of enchanted player's upkeep, Curse of the Pierced Heart deals 1 damage to that player.
this.addAbility(new CurseOfThePiercedHeartAbility()); this.addAbility(new BeginningOfUpkeepAttachedTriggeredAbility(new CurseOfThePiercedHeartEffect()));
} }
private CurseOfThePiercedHeart(final CurseOfThePiercedHeart card) { private CurseOfThePiercedHeart(final CurseOfThePiercedHeart card) {
@ -54,55 +52,14 @@ public final class CurseOfThePiercedHeart extends CardImpl {
} }
} }
class CurseOfThePiercedHeartAbility extends TriggeredAbilityImpl {
public CurseOfThePiercedHeartAbility() {
super(Zone.BATTLEFIELD, new CurseOfThePiercedHeartEffect());
}
public CurseOfThePiercedHeartAbility(final CurseOfThePiercedHeartAbility ability) {
super(ability);
}
@Override
public CurseOfThePiercedHeartAbility copy() {
return new CurseOfThePiercedHeartAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = game.getPermanent(this.sourceId);
if (enchantment != null && enchantment.getAttachedTo() != null) {
Player player = game.getPlayer(enchantment.getAttachedTo());
if (player != null && game.isActivePlayer(player.getId())) {
this.getEffects().get(0).setTargetPointer(new FixedTarget(player.getId()));
return true;
}
}
return false;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, "
+ "{this} deals 1 damage to that player or a planeswalker that player controls.";
}
}
class CurseOfThePiercedHeartEffect extends OneShotEffect { class CurseOfThePiercedHeartEffect extends OneShotEffect {
public CurseOfThePiercedHeartEffect() { CurseOfThePiercedHeartEffect() {
super(Outcome.Damage); super(Outcome.Damage);
this.staticText = "{this} deals 1 damage to that player or a planeswalker that player controls"; this.staticText = "{this} deals 1 damage to that player or a planeswalker that player controls";
} }
public CurseOfThePiercedHeartEffect(final CurseOfThePiercedHeartEffect effect) { private CurseOfThePiercedHeartEffect(final CurseOfThePiercedHeartEffect effect) {
super(effect); super(effect);
} }
@ -114,29 +71,23 @@ class CurseOfThePiercedHeartEffect extends OneShotEffect {
@Override @Override
public boolean apply(Game game, Ability source) { public boolean apply(Game game, Ability source) {
Player controller = game.getPlayer(source.getControllerId()); Player controller = game.getPlayer(source.getControllerId());
Permanent enchantment = game.getPermanent(source.getSourceId()); Player opponent = game.getPlayer(targetPointer.getFirst(game, source));
if (controller == null || enchantment == null) { if (controller == null || opponent == null) {
return false; return false;
} }
UUID opponentId = enchantment.getAttachedTo(); if (game.getBattlefield().count(StaticFilters.FILTER_CONTROLLED_PERMANENT_PLANESWALKER, source.getSourceId(), opponent.getId(), game) < 1
Player opponent = game.getPlayer(opponentId); || !controller.chooseUse(Outcome.Damage, "Redirect to a planeswalker controlled by " + opponent.getLogName() + "?", source, game)) {
if (opponent == null) { return opponent.damage(1, source.getSourceId(), source, game) > 0;
return false;
} }
if (!game.getBattlefield().getAllActivePermanents(new FilterPlaneswalkerPermanent(), opponentId, game).isEmpty()) { FilterPermanent filter = new FilterPlaneswalkerPermanent("a planeswalker controlled by " + opponent.getLogName());
if (controller.chooseUse(Outcome.Damage, "Redirect to a planeswalker controlled by " + opponent.getLogName() + "?", source, game)) { filter.add(new ControllerIdPredicate(opponent.getId()));
FilterPlaneswalkerPermanent filter = new FilterPlaneswalkerPermanent("a planeswalker controlled by " + opponent.getLogName()); TargetPermanent target = new TargetPermanent(filter);
filter.add(new ControllerIdPredicate(opponentId)); target.setNotTarget(true);
TargetPermanent target = new TargetPermanent(1, 1, filter, false); controller.choose(outcome, target, source.getSourceId(), game);
if (target.choose(Outcome.Damage, controller.getId(), source.getSourceId(), game)) { Permanent permanent = game.getPermanent(target.getFirstTarget());
Permanent permanent = game.getPermanent(target.getFirstTarget()); if (permanent != null) {
if (permanent != null) { return permanent.damage(1, source.getSourceId(), source, game, false, true) > 0;
return permanent.damage(1, source.getSourceId(), source, game, false, true) > 0;
}
}
}
} }
opponent.damage(1, source.getSourceId(), source, game); return opponent.damage(1, source.getSourceId(), source, game) > 0;
return true;
} }
} }

View file

@ -1,7 +1,7 @@
package mage.cards.c; package mage.cards.c;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.dynamicvalue.DynamicValue; import mage.abilities.dynamicvalue.DynamicValue;
import mage.abilities.effects.Effect; import mage.abilities.effects.Effect;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
@ -12,19 +12,14 @@ import mage.cards.CardSetInfo;
import mage.constants.CardType; import mage.constants.CardType;
import mage.constants.Outcome; import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone;
import mage.game.Game; import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.events.GameEvent.EventType;
import mage.game.permanent.Permanent;
import mage.players.Player; import mage.players.Player;
import mage.target.TargetPlayer; import mage.target.TargetPlayer;
import mage.target.targetpointer.FixedTarget;
import java.util.Objects;
import java.util.UUID; import java.util.UUID;
/** /**
*
* @author BetaSteward * @author BetaSteward
*/ */
public final class CurseOfThirst extends CardImpl { public final class CurseOfThirst extends CardImpl {
@ -40,8 +35,10 @@ public final class CurseOfThirst extends CardImpl {
this.addAbility(new EnchantAbility(auraTarget.getTargetName())); this.addAbility(new EnchantAbility(auraTarget.getTargetName()));
// At the beginning of enchanted player's upkeep, Curse of Thirst deals damage to that player equal to the number of Curses attached to them. // At the beginning of enchanted player's upkeep, Curse of Thirst deals damage to that player equal to the number of Curses attached to them.
this.addAbility(new CurseOfThirstAbility()); this.addAbility(new BeginningOfUpkeepAttachedTriggeredAbility(
new DamageTargetEffect(CursesAttachedCount.instance)
.setText("{this} deals damage to that player equal to the number of Curses attached to them")
));
} }
private CurseOfThirst(final CurseOfThirst card) { private CurseOfThirst(final CurseOfThirst card) {
@ -54,73 +51,28 @@ public final class CurseOfThirst extends CardImpl {
} }
} }
class CurseOfThirstAbility extends TriggeredAbilityImpl { enum CursesAttachedCount implements DynamicValue {
instance;
public CurseOfThirstAbility() {
super(Zone.BATTLEFIELD, new DamageTargetEffect(new CursesAttachedCount()));
}
public CurseOfThirstAbility(final CurseOfThirstAbility ability) {
super(ability);
}
@Override
public CurseOfThirstAbility copy() {
return new CurseOfThirstAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = game.getPermanent(this.sourceId);
if (enchantment != null && enchantment.getAttachedTo() != null) {
Player player = game.getPlayer(enchantment.getAttachedTo());
if (player != null && game.isActivePlayer(player.getId())) {
this.getEffects().get(0).setTargetPointer(new FixedTarget(player.getId()));
return true;
}
}
return false;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, Curse of Thirst "
+ "deals damage to that player equal to the number of Curses attached to them.";
}
}
class CursesAttachedCount implements DynamicValue {
public CursesAttachedCount() {
}
@Override @Override
public int calculate(Game game, Ability sourceAbility, Effect effect) { public int calculate(Game game, Ability sourceAbility, Effect effect) {
int count = 0; Player player = game.getPlayer(effect.getTargetPointer().getFirst(game, sourceAbility));
Permanent enchantment = game.getPermanent(sourceAbility.getSourceId()); if (player == null) {
if (enchantment != null && enchantment.getAttachedTo() != null) { return 0;
Player player = game.getPlayer(enchantment.getAttachedTo());
if (player != null) {
for (UUID attachmentId : player.getAttachments()) {
Permanent attachment = game.getPermanent(attachmentId);
if (attachment != null && attachment.hasSubtype(SubType.CURSE, game)) {
count++;
}
}
}
} }
return count; return player
.getAttachments()
.stream()
.map(game::getPermanent)
.filter(Objects::nonNull)
.map(p -> p.hasSubtype(SubType.CURSE, game))
.mapToInt(x -> x ? 1 : 0)
.sum();
} }
@Override @Override
public DynamicValue copy() { public CursesAttachedCount copy() {
return new CursesAttachedCount(); return instance;
} }
@Override @Override

View file

@ -2,7 +2,7 @@ package mage.cards.i;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.SpellAbility; import mage.abilities.SpellAbility;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.common.SimpleStaticAbility; import mage.abilities.common.SimpleStaticAbility;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
import mage.abilities.effects.common.GainLifeEffect; import mage.abilities.effects.common.GainLifeEffect;
@ -13,12 +13,9 @@ import mage.cards.CardImpl;
import mage.cards.CardSetInfo; import mage.cards.CardSetInfo;
import mage.constants.*; import mage.constants.*;
import mage.game.Game; import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.permanent.Permanent; import mage.game.permanent.Permanent;
import mage.game.stack.Spell; import mage.game.stack.Spell;
import mage.players.Player;
import mage.target.TargetPlayer; import mage.target.TargetPlayer;
import mage.target.targetpointer.FixedTarget;
import mage.util.CardUtil; import mage.util.CardUtil;
import java.util.Objects; import java.util.Objects;
@ -48,9 +45,11 @@ public final class InfectiousCurse extends CardImpl {
this.addAbility(new SimpleStaticAbility(Zone.BATTLEFIELD, new InfectiousCurseCostReductionEffect())); this.addAbility(new SimpleStaticAbility(Zone.BATTLEFIELD, new InfectiousCurseCostReductionEffect()));
// At the beginning of enchanted player's upkeep, that player loses 1 life and you gain 1 life. // At the beginning of enchanted player's upkeep, that player loses 1 life and you gain 1 life.
InfectiousCurseAbility curseAbility = new InfectiousCurseAbility(); Ability ability = new BeginningOfUpkeepAttachedTriggeredAbility(
curseAbility.addEffect(new GainLifeEffect(1)); new LoseLifeTargetEffect(1).setText("that player loses 1 life")
this.addAbility(curseAbility); );
ability.addEffect(new GainLifeEffect(1).concatBy("and"));
this.addAbility(ability);
} }
private InfectiousCurse(final InfectiousCurse card) { private InfectiousCurse(final InfectiousCurse card) {
@ -63,54 +62,14 @@ public final class InfectiousCurse extends CardImpl {
} }
} }
class InfectiousCurseAbility extends TriggeredAbilityImpl {
public InfectiousCurseAbility() {
super(Zone.BATTLEFIELD, new LoseLifeTargetEffect(1));
}
public InfectiousCurseAbility(final InfectiousCurseAbility ability) {
super(ability);
}
@Override
public InfectiousCurseAbility copy() {
return new InfectiousCurseAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = game.getPermanent(this.sourceId);
if (enchantment != null && enchantment.getAttachedTo() != null) {
Player player = game.getPlayer(enchantment.getAttachedTo());
if (player != null && game.isActivePlayer(player.getId())) {
this.getEffects().get(0).setTargetPointer(new FixedTarget(player.getId()));
return true;
}
}
return false;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, that player loses 1 life and you gain 1 life.";
}
}
class InfectiousCurseCostReductionEffect extends CostModificationEffectImpl { class InfectiousCurseCostReductionEffect extends CostModificationEffectImpl {
public InfectiousCurseCostReductionEffect() { InfectiousCurseCostReductionEffect() {
super(Duration.WhileOnBattlefield, Outcome.Benefit, CostModificationType.REDUCE_COST); super(Duration.WhileOnBattlefield, Outcome.Benefit, CostModificationType.REDUCE_COST);
this.staticText = "Spells you cast that target enchanted player cost {1} less to cast"; this.staticText = "Spells you cast that target enchanted player cost {1} less to cast";
} }
protected InfectiousCurseCostReductionEffect(InfectiousCurseCostReductionEffect effect) { private InfectiousCurseCostReductionEffect(InfectiousCurseCostReductionEffect effect) {
super(effect); super(effect);
} }

View file

@ -1,8 +1,7 @@
package mage.cards.t; package mage.cards.t;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbilityImpl; import mage.abilities.common.BeginningOfUpkeepAttachedTriggeredAbility;
import mage.abilities.effects.OneShotEffect; import mage.abilities.effects.OneShotEffect;
import mage.abilities.effects.common.AttachEffect; import mage.abilities.effects.common.AttachEffect;
import mage.abilities.keyword.EnchantAbility; import mage.abilities.keyword.EnchantAbility;
@ -11,11 +10,8 @@ import mage.cards.CardSetInfo;
import mage.constants.CardType; import mage.constants.CardType;
import mage.constants.Outcome; import mage.constants.Outcome;
import mage.constants.SubType; import mage.constants.SubType;
import mage.constants.Zone;
import mage.filter.StaticFilters; import mage.filter.StaticFilters;
import mage.game.Game; import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.events.GameEvent.EventType;
import mage.game.permanent.Permanent; import mage.game.permanent.Permanent;
import mage.players.Player; import mage.players.Player;
import mage.target.Target; import mage.target.Target;
@ -25,7 +21,6 @@ import mage.target.TargetPlayer;
import java.util.UUID; import java.util.UUID;
/** /**
*
* @author LevelX2 * @author LevelX2
*/ */
public final class TormentOfScarabs extends CardImpl { public final class TormentOfScarabs extends CardImpl {
@ -43,7 +38,7 @@ public final class TormentOfScarabs extends CardImpl {
this.addAbility(ability); this.addAbility(ability);
// At the beginning of enchanted player's upkeep, that player loses 3 life unless they sacrifice a nonland permanent or discards a card. // At the beginning of enchanted player's upkeep, that player loses 3 life unless they sacrifice a nonland permanent or discards a card.
this.addAbility(new TormentOfScarabsAbility()); this.addAbility(new BeginningOfUpkeepAttachedTriggeredAbility(new TormentOfScarabsEffect()));
} }
private TormentOfScarabs(final TormentOfScarabs card) { private TormentOfScarabs(final TormentOfScarabs card) {
@ -56,52 +51,14 @@ public final class TormentOfScarabs extends CardImpl {
} }
} }
class TormentOfScarabsAbility extends TriggeredAbilityImpl {
public TormentOfScarabsAbility() {
super(Zone.BATTLEFIELD, new TormentOfScarabsEffect());
}
public TormentOfScarabsAbility(final TormentOfScarabsAbility ability) {
super(ability);
}
@Override
public TormentOfScarabsAbility copy() {
return new TormentOfScarabsAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = game.getPermanent(this.getSourceId());
if (enchantment != null && enchantment.getAttachedTo() != null) {
if (game.isActivePlayer(enchantment.getAttachedTo())) {
return true;
}
}
return false;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, that player loses 3 life unless they sacrifice a nonland permanent or discards a card.";
}
}
class TormentOfScarabsEffect extends OneShotEffect { class TormentOfScarabsEffect extends OneShotEffect {
public TormentOfScarabsEffect() { TormentOfScarabsEffect() {
super(Outcome.LoseLife); super(Outcome.LoseLife);
this.staticText = "that player loses 3 life unless they sacrifice a nonland permanent or discards a card"; this.staticText = "that player loses 3 life unless they sacrifice a nonland permanent or discard a card";
} }
public TormentOfScarabsEffect(final TormentOfScarabsEffect effect) { private TormentOfScarabsEffect(final TormentOfScarabsEffect effect) {
super(effect); super(effect);
} }
@ -112,33 +69,28 @@ class TormentOfScarabsEffect extends OneShotEffect {
@Override @Override
public boolean apply(Game game, Ability source) { public boolean apply(Game game, Ability source) {
Permanent enchantment = game.getPermanent(source.getSourceId()); Player enchantedPlayer = game.getPlayer(targetPointer.getFirst(game, source));
if (enchantment == null || enchantment.getAttachedTo() == null) { if (enchantedPlayer == null) {
return false; return false;
} }
Player enchantedPlayer = game.getPlayer(enchantment.getAttachedTo()); int permanents = game.getBattlefield().countAll(StaticFilters.FILTER_PERMANENT_NON_LAND, enchantedPlayer.getId(), game);
if (enchantedPlayer != null) { if (permanents > 0 && enchantedPlayer.chooseUse(outcome, "Sacrifice a nonland permanent?",
int permanents = game.getBattlefield().countAll(StaticFilters.FILTER_PERMANENT_NON_LAND, enchantedPlayer.getId(), game); "Otherwise you have to discard a card or lose 3 life.", "Sacrifice", "Discard or life loss", source, game)) {
if (permanents > 0 && enchantedPlayer.chooseUse(outcome, "Sacrifice a nonland permanent?", Target target = new TargetPermanent(StaticFilters.FILTER_CONTROLLED_PERMANENT_NON_LAND);
"Otherwise you have to discard a card or lose 3 life.", "Sacrifice", "Discard or life loss", source, game)) { if (enchantedPlayer.choose(outcome, target, source.getSourceId(), game)) {
Target target = new TargetPermanent(StaticFilters.FILTER_CONTROLLED_PERMANENT_NON_LAND); Permanent permanent = game.getPermanent(target.getFirstTarget());
if (enchantedPlayer.choose(outcome, target, source.getSourceId(), game)) { if (permanent != null) {
Permanent permanent = game.getPermanent(target.getFirstTarget()); permanent.sacrifice(source, game);
if (permanent != null) { return true;
permanent.sacrifice(source, game);
return true;
}
} }
} }
if (!enchantedPlayer.getHand().isEmpty() && enchantedPlayer.chooseUse(outcome, "Discard a card?", }
"Otherwise you lose 3 life.", "Discard", "Lose 3 life", source, game)) { if (!enchantedPlayer.getHand().isEmpty() && enchantedPlayer.chooseUse(outcome, "Discard a card?",
enchantedPlayer.discardOne(false, false, source, game); "Otherwise you lose 3 life.", "Discard", "Lose 3 life", source, game)) {
return true; enchantedPlayer.discardOne(false, false, source, game);
}
enchantedPlayer.loseLife(3, game, source, false);
return true; return true;
} }
enchantedPlayer.loseLife(3, game, source, false);
return false; return true;
} }
} }

View file

@ -0,0 +1,52 @@
package mage.abilities.common;
import mage.abilities.TriggeredAbilityImpl;
import mage.abilities.effects.Effect;
import mage.constants.Zone;
import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.permanent.Permanent;
import mage.target.targetpointer.FixedTarget;
/**
* @author TheElk801
*/
public class BeginningOfUpkeepAttachedTriggeredAbility extends TriggeredAbilityImpl {
public BeginningOfUpkeepAttachedTriggeredAbility(Effect effect) {
this(effect, false);
}
public BeginningOfUpkeepAttachedTriggeredAbility(Effect effect, boolean optional) {
super(Zone.BATTLEFIELD, effect, optional);
}
private BeginningOfUpkeepAttachedTriggeredAbility(final BeginningOfUpkeepAttachedTriggeredAbility ability) {
super(ability);
}
@Override
public BeginningOfUpkeepAttachedTriggeredAbility copy() {
return new BeginningOfUpkeepAttachedTriggeredAbility(this);
}
@Override
public boolean checkEventType(GameEvent event, Game game) {
return event.getType() == GameEvent.EventType.UPKEEP_STEP_PRE;
}
@Override
public boolean checkTrigger(GameEvent event, Game game) {
Permanent enchantment = getSourcePermanentOrLKI(game);
if (enchantment == null || !game.isActivePlayer(enchantment.getAttachedTo())) {
return false;
}
this.getEffects().setTargetPointer(new FixedTarget(enchantment.getAttachedTo()));
return true;
}
@Override
public String getRule() {
return "At the beginning of enchanted player's upkeep, " + super.getRule();
}
}