forked from External/mage
* add new framework for batch triggers apply for tapped, untapped, sacrificed, milled simplify Ob Nixilis, Captive Kingpin * add a verify check * fix mistakes * add simple tests * another test * zone change - enters battlefield * zone change: not battlefield * zone change - leaves battlefield * fix Kaya Spirit's Justice * rename OneOrMoreCombatDamagePlayerTriggeredAbility * refactor OneOrMoreDamagePlayerTriggeredAbility * new YoureDealtDamageTriggeredAbility * new OpponentDealtNoncombatDamageTriggeredAbility * rework Risona, Asari Commander * simplify War Elemental * Add damage batch by source rework some delayed triggered abilities * fix Mindblade Render * rework Initiative and a few others * [temp] initiative test * refactor: common style for DealsDamageSourceTriggeredAbility * refactor cards to use common DealsDamageSourceTriggeredAbility * update damage players batch triggers * fix mistake in initiative * new DealtDamageAnyTriggeredAbility * new DealtCombatDamageToSourceTriggeredAbility * update dealt damage to permanent batch triggered abilities * refactor Hot Soup and param in DealtDamageAttachedTriggeredAbility * a few more permanent batch triggered abilities * fix mistake * update some more damage batch triggers * add test for Phyrexian Negator * update Felix Five-Boots and enable test update Wayta, Trainer Prodigy to align * update damage batch by source triggers * undo mistaken change * fix verify * cleanup unused methods * Revert "[temp] initiative test" This reverts commit 11ed19295fb4f54f5e0870acd4d3d515b54761f1. * Revert "add a verify check" This reverts commit e7de47a6562f13c127fdc4c29a7735a08f8da9ea. * fixes from checking text discrepancies * fix Shriekwood Devourer * merge fix --------- Co-authored-by: Susucre <34709007+Susucre@users.noreply.github.com>
128 lines
5 KiB
Java
128 lines
5 KiB
Java
package mage.abilities.common;
|
|
|
|
import mage.abilities.BatchTriggeredAbility;
|
|
import mage.abilities.TriggeredAbilityImpl;
|
|
import mage.abilities.effects.Effect;
|
|
import mage.constants.SetTargetPointer;
|
|
import mage.constants.TargetController;
|
|
import mage.constants.Zone;
|
|
import mage.filter.FilterPermanent;
|
|
import mage.game.Game;
|
|
import mage.game.events.GameEvent;
|
|
import mage.game.events.SacrificedPermanentBatchEvent;
|
|
import mage.game.events.SacrificedPermanentEvent;
|
|
import mage.game.permanent.Permanent;
|
|
import mage.target.targetpointer.FixedTargets;
|
|
|
|
import java.util.List;
|
|
import java.util.Objects;
|
|
import java.util.stream.Collectors;
|
|
|
|
/**
|
|
* @author TheElk801, xenohedron
|
|
*/
|
|
public class SacrificeOneOrMorePermanentsTriggeredAbility extends TriggeredAbilityImpl implements BatchTriggeredAbility<SacrificedPermanentEvent> {
|
|
|
|
private final FilterPermanent filter;
|
|
private final SetTargetPointer setTargetPointer;
|
|
|
|
private final TargetController sacrificingPlayer;
|
|
|
|
/**
|
|
* Whenever you sacrifice one or more "[filter]", "[effect]".
|
|
* zone = battlefield, setTargetPointer = NONE, optional = false
|
|
*/
|
|
public SacrificeOneOrMorePermanentsTriggeredAbility(Effect effect, FilterPermanent filter) {
|
|
this(Zone.BATTLEFIELD, effect, filter, TargetController.YOU, SetTargetPointer.NONE, false);
|
|
}
|
|
|
|
public SacrificeOneOrMorePermanentsTriggeredAbility(Zone zone, Effect effect, FilterPermanent filter,
|
|
TargetController sacrificingPlayer,
|
|
SetTargetPointer setTargetPointer, boolean optional) {
|
|
super(zone, effect, optional);
|
|
if (Zone.BATTLEFIELD.match(zone)) {
|
|
setLeavesTheBattlefieldTrigger(true);
|
|
}
|
|
this.filter = filter;
|
|
this.setTargetPointer = setTargetPointer;
|
|
this.sacrificingPlayer = sacrificingPlayer;
|
|
setTriggerPhrase(generateTriggerPhrase());
|
|
}
|
|
|
|
protected SacrificeOneOrMorePermanentsTriggeredAbility(final SacrificeOneOrMorePermanentsTriggeredAbility ability) {
|
|
super(ability);
|
|
this.filter = ability.filter;
|
|
this.setTargetPointer = ability.setTargetPointer;
|
|
this.sacrificingPlayer = ability.sacrificingPlayer;
|
|
}
|
|
|
|
@Override
|
|
public SacrificeOneOrMorePermanentsTriggeredAbility copy() {
|
|
return new SacrificeOneOrMorePermanentsTriggeredAbility(this);
|
|
}
|
|
|
|
@Override
|
|
public boolean checkEventType(GameEvent event, Game game) {
|
|
return event.getType() == GameEvent.EventType.SACRIFICED_PERMANENT_BATCH;
|
|
}
|
|
|
|
@Override
|
|
public boolean checkEvent(SacrificedPermanentEvent event, Game game) {
|
|
Permanent permanent = game.getPermanentOrLKIBattlefield(event.getTargetId());
|
|
if (permanent == null || !filter.match(permanent, getControllerId(), this, game)) {
|
|
return false;
|
|
}
|
|
switch (sacrificingPlayer) {
|
|
case YOU:
|
|
return isControlledBy(event.getPlayerId());
|
|
case OPPONENT:
|
|
return game.getOpponents(getControllerId()).contains(event.getPlayerId());
|
|
case ANY:
|
|
return true;
|
|
default:
|
|
throw new IllegalArgumentException("Unsupported TargetController in SacrificePermanentTriggeredAbility: " + sacrificingPlayer); }
|
|
}
|
|
|
|
@Override
|
|
public boolean checkTrigger(GameEvent event, Game game) {
|
|
List<Permanent> matchingPermanents = getFilteredEvents((SacrificedPermanentBatchEvent) event, game)
|
|
.stream()
|
|
.map(GameEvent::getTargetId)
|
|
.map(game::getPermanentOrLKIBattlefield)
|
|
.filter(Objects::nonNull)
|
|
.collect(Collectors.toList());
|
|
if (matchingPermanents.isEmpty()) {
|
|
return false;
|
|
}
|
|
this.getEffects().setValue("sacrificedPermanents", matchingPermanents);
|
|
switch (setTargetPointer) {
|
|
case PERMANENT:
|
|
this.getEffects().setTargetPointer(new FixedTargets(matchingPermanents, game));
|
|
break;
|
|
case NONE:
|
|
break;
|
|
default:
|
|
throw new IllegalArgumentException("Unsupported SetTargetPointer in SacrificePermanentTriggeredAbility: " + setTargetPointer);
|
|
}
|
|
return true;
|
|
}
|
|
|
|
private String generateTriggerPhrase() {
|
|
String targetControllerText;
|
|
switch (sacrificingPlayer) {
|
|
case YOU:
|
|
targetControllerText = "you sacrifice one or more ";
|
|
break;
|
|
case OPPONENT:
|
|
targetControllerText = "an opponent sacrifices one or more ";
|
|
break;
|
|
case ANY:
|
|
targetControllerText = "one or more players sacrifice one or more ";
|
|
break;
|
|
default:
|
|
throw new IllegalArgumentException("Unsupported TargetController in SacrificePermanentTriggeredAbility: " + sacrificingPlayer);
|
|
}
|
|
return getWhen() + targetControllerText + filter.getMessage() + ", ";
|
|
}
|
|
|
|
}
|