diff --git a/Mage/src/mage/abilities/AbilityImpl.java b/Mage/src/mage/abilities/AbilityImpl.java index 4608c5828d4..be15e14b18c 100644 --- a/Mage/src/mage/abilities/AbilityImpl.java +++ b/Mage/src/mage/abilities/AbilityImpl.java @@ -52,6 +52,7 @@ import org.apache.log4j.Logger; import java.util.ArrayList; import java.util.List; import java.util.UUID; +import mage.abilities.keyword.EntwineAbility; /** @@ -191,7 +192,8 @@ public abstract class AbilityImpl> implements Ability { break; } } - if (ability instanceof OptionalAdditionalSourceCosts) { + if (ability instanceof OptionalAdditionalSourceCosts + && !(ability instanceof EntwineAbility)) { ((OptionalAdditionalSourceCosts)ability).addOptionalAdditionalCosts(this, game); } } diff --git a/Mage/src/mage/abilities/Modes.java b/Mage/src/mage/abilities/Modes.java index 0637e4a4fcf..a27df19db6d 100644 --- a/Mage/src/mage/abilities/Modes.java +++ b/Mage/src/mage/abilities/Modes.java @@ -29,10 +29,12 @@ package mage.abilities; import java.util.LinkedHashMap; import java.util.LinkedHashSet; -import java.util.List; import java.util.Map; import java.util.Set; import java.util.UUID; +import mage.abilities.costs.OptionalAdditionalModeSourceCosts; +import mage.abilities.costs.OptionalAdditionalSourceCosts; +import mage.cards.Card; import mage.game.Game; import mage.players.Player; @@ -109,7 +111,25 @@ public class Modes extends LinkedHashMap { public boolean choose(Game game, Ability source) { if (this.size() > 1) { this.selectedModes.clear(); - + // check if mode modifying abilities exist + Card card = game.getCard(source.getSourceId()); + if (card != null) { + for (Ability modeModifyingAbility : card.getAbilities()) { + if (modeModifyingAbility instanceof OptionalAdditionalModeSourceCosts) { + ((OptionalAdditionalModeSourceCosts)modeModifyingAbility).addOptionalAdditionalModeCosts(source, game); + } + } + } + // check if all modes can be activated automatically + if (this.size() == this.getMinModes()) { + for (Mode mode: this.values()) { + if (mode.getTargets().canChoose(source.getSourceId(), source.getControllerId(), game)) { + this.selectedModes.add(mode.getId()); + } + } + return selectedModes.size() > 0; + } + // player chooses modes manually Player player = game.getPlayer(source.getControllerId()); while (this.selectedModes.size() < this.getMaxModes()) { Mode choice = player.chooseMode(this, source, game); diff --git a/Mage/src/mage/abilities/costs/OptionalAdditionalModeSourceCosts.java b/Mage/src/mage/abilities/costs/OptionalAdditionalModeSourceCosts.java new file mode 100644 index 00000000000..104542486d7 --- /dev/null +++ b/Mage/src/mage/abilities/costs/OptionalAdditionalModeSourceCosts.java @@ -0,0 +1,42 @@ +/* + * 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.costs; + +import mage.abilities.Ability; +import mage.game.Game; + +/** + * + * @author LevelX2 + */ + + +public interface OptionalAdditionalModeSourceCosts { + void addOptionalAdditionalModeCosts(Ability ability, Game game); + String getCastMessageSuffix(); +} diff --git a/Mage/src/mage/abilities/keyword/EntwineAbility.java b/Mage/src/mage/abilities/keyword/EntwineAbility.java new file mode 100644 index 00000000000..8e755cebded --- /dev/null +++ b/Mage/src/mage/abilities/keyword/EntwineAbility.java @@ -0,0 +1,160 @@ +/* + * 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.keyword; + +import java.util.Iterator; + +import mage.constants.Zone; +import mage.abilities.Ability; +import mage.abilities.SpellAbility; +import mage.abilities.StaticAbility; +import mage.abilities.costs.Cost; +import mage.abilities.costs.Costs; +import mage.abilities.costs.OptionalAdditionalCost; +import mage.abilities.costs.OptionalAdditionalCostImpl; +import mage.abilities.costs.OptionalAdditionalModeSourceCosts; +import mage.abilities.costs.mana.ManaCostsImpl; +import mage.constants.Outcome; +import mage.game.Game; +import mage.players.Player; + +/** + * 702.40. Entwine + * + * 702.40a Entwine is a static ability of modal spells (see rule 700.2) that functions + * while the spell is on the stack. "Entwine [cost]" means "You may choose all modes + * of this spell instead of just one. If you do, you pay an additional [cost]." Using + * the entwine ability follows the rules for choosing modes and paying additional costs + * in rules 601.2b and 601.2e-g. + * + * 702.40b If the entwine cost was paid, follow the text of each of the modes in the order + * written on the card when the spell resolves. + * + * @author LevelX2 + */ + +public class EntwineAbility extends StaticAbility implements OptionalAdditionalModeSourceCosts { + + private static final String keywordText = "Entwine"; + private static final String reminderText = " (Choose both if you pay the entwine cost.)"; + protected OptionalAdditionalCost additionalCost; + + public EntwineAbility(String manaString) { + super(Zone.STACK, null); + this.additionalCost = new OptionalAdditionalCostImpl(keywordText, reminderText, new ManaCostsImpl(manaString)); + } + + public EntwineAbility(Cost cost) { + super(Zone.STACK, null); + this.additionalCost = new OptionalAdditionalCostImpl(keywordText, "-", reminderText, cost); + setRuleAtTheTop(true); + } + + public EntwineAbility(final EntwineAbility ability) { + super(ability); + additionalCost = ability.additionalCost; + } + + @Override + public EntwineAbility copy() { + return new EntwineAbility(this); + } + + @Override + public void addCost(Cost cost) { + if (additionalCost != null) { + ((Costs) additionalCost).add(cost); + } + } + + public boolean isActivated() { + if (additionalCost != null) { + return additionalCost.isActivated(); + } + return false; + } + + public void resetCosts() { + if (additionalCost != null) { + additionalCost.reset(); + } + } + + @Override + public void addOptionalAdditionalModeCosts(Ability ability, Game game) { + if (ability instanceof SpellAbility) { + Player player = game.getPlayer(controllerId); + if (player != null) { + this.resetCosts(); + if (additionalCost != null) { + if (player.chooseUse(Outcome.Benefit,new StringBuilder("Pay ").append(additionalCost.getText(false)).append(" ?").toString(), game)) { + additionalCost.activate(); + for (Iterator it = ((Costs) additionalCost).iterator(); it.hasNext();) { + Cost cost = (Cost) it.next(); + if (cost instanceof ManaCostsImpl) { + ability.getManaCostsToPay().add((ManaCostsImpl) cost.copy()); + } else { + ability.getCosts().add(cost.copy()); + } + } + ability.getModes().setMinModes(2); + ability.getModes().setMaxModes(2); + } + } + } + } + } + + + @Override + public String getRule() { + StringBuilder sb = new StringBuilder(); + if (additionalCost != null) { + sb.append(additionalCost.getText(false)); + sb.append(" ").append(additionalCost.getReminderText()); + } + return sb.toString(); + } + + @Override + public String getCastMessageSuffix() { + if (additionalCost != null) { + return additionalCost.getCastSuffixMessage(0); + } else { + return ""; + } + } + + public String getReminderText() { + if (additionalCost != null) { + return additionalCost.getReminderText(); + } else { + return ""; + } + } +} diff --git a/Utils/keywords.txt b/Utils/keywords.txt index 7703191a843..4b4a2e77d18 100644 --- a/Utils/keywords.txt +++ b/Utils/keywords.txt @@ -3,6 +3,7 @@ Bloodthirst|number| Bushido|number| Dredge|number| Evoke|card, manaString| +Entwine|manaString| Soulshift|number| Basic landcycling|cost| Forestcycling|cost|