forked from External/mage
479 lines
12 KiB
Java
479 lines
12 KiB
Java
/*
|
|
* Copyright 2010 BetaSteward_at_googlemail.com. All rights reserved.
|
|
*
|
|
* Redistribution and use in source and binary forms, with or without modification, are
|
|
* permitted provided that the following conditions are met:
|
|
*
|
|
* 1. Redistributions of source code must retain the above copyright notice, this list of
|
|
* conditions and the following disclaimer.
|
|
*
|
|
* 2. Redistributions in binary form must reproduce the above copyright notice, this list
|
|
* of conditions and the following disclaimer in the documentation and/or other materials
|
|
* provided with the distribution.
|
|
*
|
|
* THIS SOFTWARE IS PROVIDED BY BetaSteward_at_googlemail.com ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
|
* WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
|
* FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL BetaSteward_at_googlemail.com OR
|
|
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
|
|
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
|
|
* ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
|
|
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
|
* ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
*
|
|
* The views and conclusions contained in the software and documentation are those of the
|
|
* authors and should not be interpreted as representing official policies, either expressed
|
|
* or implied, of BetaSteward_at_googlemail.com.
|
|
*/
|
|
|
|
package mage.abilities;
|
|
|
|
import mage.Constants.AbilityType;
|
|
import mage.Constants.EffectType;
|
|
import mage.Constants.Outcome;
|
|
import mage.Constants.Zone;
|
|
import mage.MageObject;
|
|
import mage.abilities.costs.*;
|
|
import mage.abilities.costs.mana.KickerManaCost;
|
|
import mage.abilities.costs.mana.ManaCost;
|
|
import mage.abilities.costs.mana.ManaCosts;
|
|
import mage.abilities.costs.mana.ManaCostsImpl;
|
|
import mage.abilities.effects.*;
|
|
import mage.abilities.mana.ManaAbility;
|
|
import mage.cards.Card;
|
|
import mage.choices.Choice;
|
|
import mage.choices.Choices;
|
|
import mage.game.Game;
|
|
import mage.target.Target;
|
|
import mage.target.Targets;
|
|
import org.apache.log4j.Logger;
|
|
|
|
import java.util.ArrayList;
|
|
import java.util.List;
|
|
import java.util.UUID;
|
|
|
|
/**
|
|
*
|
|
* @author BetaSteward_at_googlemail.com
|
|
*/
|
|
public abstract class AbilityImpl<T extends AbilityImpl<T>> implements Ability {
|
|
|
|
private final static transient Logger logger = Logger.getLogger(AbilityImpl.class);
|
|
|
|
protected UUID id;
|
|
protected UUID originalId;
|
|
protected AbilityType abilityType;
|
|
protected UUID controllerId;
|
|
protected UUID sourceId;
|
|
protected ManaCosts<ManaCost> manaCosts;
|
|
protected ManaCosts<ManaCost> manaCostsToPay;
|
|
protected Costs<Cost> costs;
|
|
protected ArrayList<AlternativeCost> alternativeCosts = new ArrayList<AlternativeCost>();
|
|
protected Costs<Cost> optionalCosts;
|
|
protected Modes modes;
|
|
protected Zone zone;
|
|
protected String name;
|
|
protected boolean usesStack = true;
|
|
|
|
@Override
|
|
public abstract T copy();
|
|
|
|
public AbilityImpl(AbilityType abilityType, Zone zone) {
|
|
this.id = UUID.randomUUID();
|
|
this.originalId = id;
|
|
this.abilityType = abilityType;
|
|
this.zone = zone;
|
|
this.manaCosts = new ManaCostsImpl<ManaCost>();
|
|
this.manaCostsToPay = new ManaCostsImpl<ManaCost>();
|
|
this.costs = new CostsImpl<Cost>();
|
|
this.optionalCosts = new CostsImpl<Cost>();
|
|
this.modes = new Modes();
|
|
}
|
|
|
|
public AbilityImpl(final AbilityImpl<T> ability) {
|
|
this.id = ability.id;
|
|
this.originalId = ability.originalId;
|
|
this.abilityType = ability.abilityType;
|
|
this.controllerId = ability.controllerId;
|
|
this.sourceId = ability.sourceId;
|
|
this.zone = ability.zone;
|
|
this.name = ability.name;
|
|
this.usesStack = ability.usesStack;
|
|
this.manaCosts = ability.manaCosts;
|
|
this.manaCostsToPay = ability.manaCostsToPay.copy();
|
|
this.costs = ability.costs.copy();
|
|
this.optionalCosts = ability.optionalCosts.copy();
|
|
for (AlternativeCost cost: ability.alternativeCosts) {
|
|
this.alternativeCosts.add((AlternativeCost)cost.copy());
|
|
}
|
|
this.modes = ability.modes.copy();
|
|
}
|
|
|
|
@Override
|
|
public UUID getId() {
|
|
return id;
|
|
}
|
|
|
|
@Override
|
|
public void newId() {
|
|
this.id = UUID.randomUUID();
|
|
getEffects().newId();
|
|
}
|
|
|
|
@Override
|
|
public void newOriginalId() {
|
|
this.id = UUID.randomUUID();
|
|
this.originalId = id;
|
|
getEffects().newId();
|
|
}
|
|
|
|
@Override
|
|
public AbilityType getAbilityType() {
|
|
return this.abilityType;
|
|
}
|
|
|
|
@Override
|
|
public boolean resolve(Game game) {
|
|
boolean result = true;
|
|
//20100716 - 117.12
|
|
if (checkIfClause(game)) {
|
|
for (Effect effect: getEffects()) {
|
|
if (effect instanceof OneShotEffect) {
|
|
if (!(effect instanceof PostResolveEffect))
|
|
result &= effect.apply(game, this);
|
|
}
|
|
else {
|
|
game.addEffect((ContinuousEffect) effect, this);
|
|
}
|
|
}
|
|
}
|
|
return result;
|
|
}
|
|
|
|
@Override
|
|
public boolean activate(Game game, boolean noMana) {
|
|
// 20110204 - 700.2
|
|
if (!modes.choose(game, this))
|
|
return false;
|
|
//20100716 - 601.2b
|
|
if (getChoices().size() > 0 && getChoices().choose(game, this) == false) {
|
|
logger.debug("activate failed - choice");
|
|
return false;
|
|
}
|
|
//20100716 - 601.2b
|
|
if (getTargets().size() > 0 && getTargets().chooseTargets(getEffects().get(0).getOutcome(), this.controllerId, this, game) == false) {
|
|
logger.debug("activate failed - target");
|
|
return false;
|
|
}
|
|
|
|
for (Cost cost : optionalCosts) {
|
|
if (cost instanceof KickerManaCost) {
|
|
cost.clearPaid();
|
|
if (game.getPlayer(this.controllerId).chooseUse(Outcome.Benefit, "Pay " + cost.getText() + "?", game)) {
|
|
manaCostsToPay.add((ManaCost) cost);
|
|
}
|
|
} else if (cost instanceof ManaCost) {
|
|
cost.clearPaid();
|
|
if (game.getPlayer(this.controllerId).chooseUse(Outcome.Benefit, "Pay optional cost " + cost.getText() + "?", game)) {
|
|
manaCostsToPay.add((ManaCost) cost);
|
|
}
|
|
}
|
|
}
|
|
//20100716 - 601.2e
|
|
Card card = game.getCard(sourceId);
|
|
if (card != null) {
|
|
card.adjustCosts(this, game);
|
|
for (Ability ability : card.getAbilities()) {
|
|
if (ability instanceof AdjustingSourceCosts) {
|
|
((AdjustingSourceCosts)ability).adjustCosts(this, game);
|
|
}
|
|
}
|
|
}
|
|
|
|
// this is a hack to prevent mana abilities with mana costs from causing endless loops - pay other costs first
|
|
if (this instanceof ManaAbility && !costs.pay(this, game, sourceId, controllerId, noMana)) {
|
|
logger.debug("activate mana ability failed - non mana costs");
|
|
return false;
|
|
}
|
|
|
|
if (!useAlternativeCost(game)) {
|
|
//20101001 - 601.2e
|
|
game.getContinuousEffects().costModification(this, game);
|
|
|
|
//20100716 - 601.2f
|
|
if (!manaCostsToPay.pay(this, game, sourceId, controllerId, noMana)) {
|
|
logger.debug("activate failed - mana");
|
|
return false;
|
|
}
|
|
}
|
|
//20100716 - 601.2g
|
|
if (!costs.pay(this, game, sourceId, controllerId, noMana)) {
|
|
logger.debug("activate failed - non mana costs");
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public void reset(Game game) {}
|
|
|
|
protected boolean useAlternativeCost(Game game) {
|
|
for (AlternativeCost cost: alternativeCosts) {
|
|
if (cost.isAvailable(game, this)) {
|
|
if (game.getPlayer(this.controllerId).chooseUse(Outcome.Neutral, "Use alternative cost " + cost.getName(), game))
|
|
return cost.pay(this, game, sourceId, controllerId, false);
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean checkIfClause(Game game) {
|
|
return true;
|
|
}
|
|
|
|
@Override
|
|
public UUID getControllerId() {
|
|
return controllerId;
|
|
}
|
|
|
|
@Override
|
|
public void setControllerId(UUID controllerId) {
|
|
this.controllerId = controllerId;
|
|
}
|
|
|
|
|
|
@Override
|
|
public UUID getSourceId() {
|
|
return sourceId;
|
|
}
|
|
|
|
@Override
|
|
public void setSourceId(UUID sourceId) {
|
|
this.sourceId = sourceId;
|
|
}
|
|
|
|
@Override
|
|
public Costs getCosts() {
|
|
return costs;
|
|
}
|
|
|
|
@Override
|
|
public ManaCosts<ManaCost> getManaCosts() {
|
|
return manaCosts;
|
|
}
|
|
|
|
/**
|
|
* Should be used by {@link mage.abilities.effects.CostModificationEffect cost modification effects}
|
|
* to manipulate what is actually paid before resolution.
|
|
*
|
|
* @return
|
|
*/
|
|
@Override
|
|
public ManaCosts<ManaCost> getManaCostsToPay ( ) {
|
|
return manaCostsToPay;
|
|
}
|
|
|
|
@Override
|
|
public List<AlternativeCost> getAlternativeCosts() {
|
|
return alternativeCosts;
|
|
}
|
|
|
|
@Override
|
|
public Costs getOptionalCosts() {
|
|
return optionalCosts;
|
|
}
|
|
|
|
@Override
|
|
public Effects getEffects() {
|
|
return modes.getMode().getEffects();
|
|
}
|
|
|
|
@Override
|
|
public Effects getEffects(Game game, EffectType effectType) {
|
|
Effects typedEffects = new Effects();
|
|
for (Effect effect: getEffects()) {
|
|
if (effect.getEffectType() == effectType) {
|
|
typedEffects.add(effect);
|
|
}
|
|
}
|
|
return typedEffects;
|
|
}
|
|
|
|
@Override
|
|
public Choices getChoices() {
|
|
return modes.getMode().getChoices();
|
|
}
|
|
|
|
@Override
|
|
public Zone getZone() {
|
|
return zone;
|
|
}
|
|
|
|
@Override
|
|
public boolean isUsesStack() {
|
|
return usesStack;
|
|
}
|
|
|
|
@Override
|
|
public String getRule() {
|
|
return getRule(false);
|
|
}
|
|
|
|
@Override
|
|
public String getRule(boolean all) {
|
|
StringBuilder sbRule = new StringBuilder();
|
|
|
|
if (all || this.abilityType != AbilityType.SPELL) {
|
|
if (manaCosts.size() > 0) {
|
|
sbRule.append(manaCosts.getText());
|
|
}
|
|
if (costs.size() > 0) {
|
|
if (sbRule.length() > 0) {
|
|
sbRule.append(",");
|
|
}
|
|
sbRule.append(costs.getText());
|
|
}
|
|
if (sbRule.length() > 0) {
|
|
sbRule.append(": ");
|
|
}
|
|
}
|
|
else if (this.alternativeCosts.size() > 0) {
|
|
for (AlternativeCost cost: alternativeCosts) {
|
|
sbRule.append(cost.getText()).append("\n");
|
|
}
|
|
}
|
|
else {
|
|
for (Cost cost: this.costs) {
|
|
if (!(cost instanceof ManaCost)) {
|
|
sbRule.append("As an additional cost to cast {this}, ").append(cost.getText()).append("\n");
|
|
}
|
|
}
|
|
}
|
|
|
|
sbRule.append(modes.getText());
|
|
|
|
return sbRule.toString();
|
|
}
|
|
|
|
@Override
|
|
public String getRule(String source) {
|
|
return formatRule(getRule(), source);
|
|
}
|
|
|
|
protected String formatRule(String rule, String source) {
|
|
String replace = rule.replace("{this}", source);
|
|
replace = replace.replace("{source}", source);
|
|
return replace;
|
|
}
|
|
|
|
@Override
|
|
public void addCost(Cost cost) {
|
|
if (cost != null) {
|
|
if (cost instanceof ManaCost) {
|
|
this.addManaCost((ManaCost)cost);
|
|
}
|
|
else {
|
|
this.costs.add(cost);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addManaCost(ManaCost cost) {
|
|
if (cost != null) {
|
|
this.manaCosts.add(cost);
|
|
this.manaCostsToPay.add(cost);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addAlternativeCost(AlternativeCost cost) {
|
|
if (cost != null) {
|
|
this.alternativeCosts.add(cost);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addOptionalCost(Cost cost) {
|
|
if (cost != null) {
|
|
this.optionalCosts.add(cost);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addEffect(Effect effect) {
|
|
if (effect != null) {
|
|
getEffects().add(effect);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addTarget(Target target) {
|
|
if (target != null) {
|
|
getTargets().add(target);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public void addChoice(Choice choice) {
|
|
if (choice != null) {
|
|
getChoices().add(choice);
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public Targets getTargets() {
|
|
return modes.getMode().getTargets();
|
|
}
|
|
|
|
@Override
|
|
public UUID getFirstTarget() {
|
|
return getTargets().getFirstTarget();
|
|
}
|
|
|
|
@Override
|
|
public boolean isModal() {
|
|
return this.modes.size() > 1;
|
|
}
|
|
|
|
@Override
|
|
public void addMode(Mode mode) {
|
|
this.modes.addMode(mode);
|
|
}
|
|
|
|
@Override
|
|
public Modes getModes() {
|
|
return modes;
|
|
}
|
|
|
|
@Override
|
|
public boolean canChooseTarget(Game game) {
|
|
for (Mode mode: modes.values()) {
|
|
if (mode.getTargets().canChoose(sourceId, controllerId, game))
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
@Override
|
|
public boolean isInUseableZone(Game game, boolean checkLKI) {
|
|
// try LKI first
|
|
|
|
if (checkLKI) {
|
|
MageObject lkiTest = game.getLastKnownInformation(getSourceId(), zone);
|
|
if (lkiTest != null) {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
// check against current state
|
|
Zone test = game.getState().getZone(sourceId);
|
|
return test != null && zone.match(test);
|
|
}
|
|
|
|
@Override
|
|
public String toString() {
|
|
return getRule();
|
|
}
|
|
}
|