mirror of
https://github.com/correl/mage.git
synced 2025-04-04 01:06:04 -09:00
* Available mana calculation - Fixed a problem with repeatable mana converting abilities (e.g. Farrelite Priest) that were only considered once (#6698).
This commit is contained in:
parent
cbb289337d
commit
653a2dd7b2
5 changed files with 109 additions and 40 deletions
Mage.Sets/src/mage/cards/e
Mage.Tests/src/test/java/org/mage/test/cards/mana
Mage/src/main/java/mage
|
@ -1,4 +1,3 @@
|
||||||
|
|
||||||
package mage.cards.e;
|
package mage.cards.e;
|
||||||
|
|
||||||
import java.util.UUID;
|
import java.util.UUID;
|
||||||
|
@ -18,11 +17,11 @@ import mage.constants.Zone;
|
||||||
public final class EyeOfRamos extends CardImpl {
|
public final class EyeOfRamos extends CardImpl {
|
||||||
|
|
||||||
public EyeOfRamos(UUID ownerId, CardSetInfo setInfo) {
|
public EyeOfRamos(UUID ownerId, CardSetInfo setInfo) {
|
||||||
super(ownerId,setInfo,new CardType[]{CardType.ARTIFACT},"{3}");
|
super(ownerId, setInfo, new CardType[]{CardType.ARTIFACT}, "{3}");
|
||||||
|
|
||||||
// {tap}: Add {U}.
|
// {T}: Add {U}.
|
||||||
this.addAbility(new BlueManaAbility());
|
this.addAbility(new BlueManaAbility());
|
||||||
|
|
||||||
// Sacrifice Eye of Ramos: Add {U}.
|
// Sacrifice Eye of Ramos: Add {U}.
|
||||||
this.addAbility(new SimpleManaAbility(Zone.BATTLEFIELD, Mana.BlueMana(1), new SacrificeSourceCost()));
|
this.addAbility(new SimpleManaAbility(Zone.BATTLEFIELD, Mana.BlueMana(1), new SacrificeSourceCost()));
|
||||||
}
|
}
|
||||||
|
|
|
@ -240,4 +240,42 @@ public class TappedForManaRelatedTest extends CardTestPlayerBase {
|
||||||
Assert.assertEquals("mana variations don't fit", 1, manaOptions.size());
|
Assert.assertEquals("mana variations don't fit", 1, manaOptions.size());
|
||||||
assertManaOptions("{Any}", manaOptions);
|
assertManaOptions("{Any}", manaOptions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void TestEyeOfRamos() {
|
||||||
|
setStrictChooseMode(true);
|
||||||
|
// {T}: Add {U}.
|
||||||
|
// Sacrifice Eye of Ramos: Add {U}.
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, "Eye of Ramos", 2);
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.BEGIN_COMBAT);
|
||||||
|
execute();
|
||||||
|
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
ManaOptions manaOptions = playerA.getAvailableManaTest(currentGame);
|
||||||
|
Assert.assertEquals("mana variations don't fit", 1, manaOptions.size());
|
||||||
|
assertManaOptions("{U}{U}{U}{U}", manaOptions);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void TestFarrelitePriest() {
|
||||||
|
setStrictChooseMode(true);
|
||||||
|
// {1}: Add {W}. If this ability has been activated four or more times this turn, sacrifice Farrelite Priest at the beginning of the next end step.
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, "Farrelite Priest", 1);
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, "Swamp", 4);
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.BEGIN_COMBAT);
|
||||||
|
execute();
|
||||||
|
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
ManaOptions manaOptions = playerA.getAvailableManaTest(currentGame);
|
||||||
|
Assert.assertEquals("mana variations don't fit", 5, manaOptions.size());
|
||||||
|
assertManaOptions("{W}{W}{W}{W}", manaOptions);
|
||||||
|
assertManaOptions("{W}{W}{W}{B}", manaOptions);
|
||||||
|
assertManaOptions("{W}{W}{B}{B}", manaOptions);
|
||||||
|
assertManaOptions("{W}{B}{B}{B}", manaOptions);
|
||||||
|
assertManaOptions("{B}{B}{B}{B}", manaOptions);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
package mage.abilities;
|
package mage.abilities;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
import mage.MageObject;
|
import mage.MageObject;
|
||||||
import mage.abilities.costs.*;
|
import mage.abilities.costs.*;
|
||||||
import mage.abilities.costs.common.PayLifeCost;
|
import mage.abilities.costs.common.PayLifeCost;
|
||||||
|
@ -8,6 +12,7 @@ import mage.abilities.effects.ContinuousEffect;
|
||||||
import mage.abilities.effects.Effect;
|
import mage.abilities.effects.Effect;
|
||||||
import mage.abilities.effects.Effects;
|
import mage.abilities.effects.Effects;
|
||||||
import mage.abilities.effects.OneShotEffect;
|
import mage.abilities.effects.OneShotEffect;
|
||||||
|
import mage.abilities.effects.common.ManaEffect;
|
||||||
import mage.abilities.hint.Hint;
|
import mage.abilities.hint.Hint;
|
||||||
import mage.abilities.mana.ActivatedManaAbilityImpl;
|
import mage.abilities.mana.ActivatedManaAbilityImpl;
|
||||||
import mage.cards.Card;
|
import mage.cards.Card;
|
||||||
|
@ -30,12 +35,6 @@ import mage.util.ThreadLocalStringBuilder;
|
||||||
import mage.watchers.Watcher;
|
import mage.watchers.Watcher;
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.UUID;
|
|
||||||
import mage.abilities.effects.common.ManaEffect;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author BetaSteward_at_googlemail.com
|
* @author BetaSteward_at_googlemail.com
|
||||||
*/
|
*/
|
||||||
|
@ -54,7 +53,7 @@ public abstract class AbilityImpl implements Ability {
|
||||||
protected ManaCosts<ManaCost> manaCostsToPay;
|
protected ManaCosts<ManaCost> manaCostsToPay;
|
||||||
protected Costs<Cost> costs;
|
protected Costs<Cost> costs;
|
||||||
protected Costs<Cost> optionalCosts;
|
protected Costs<Cost> optionalCosts;
|
||||||
protected Modes modes; // access to it by GetModes only (it's can be override by some abilities)
|
protected Modes modes; // access to it by GetModes only (it can be overridden by some abilities)
|
||||||
protected Zone zone;
|
protected Zone zone;
|
||||||
protected String name;
|
protected String name;
|
||||||
protected AbilityWord abilityWord;
|
protected AbilityWord abilityWord;
|
||||||
|
@ -65,7 +64,7 @@ public abstract class AbilityImpl implements Ability {
|
||||||
protected boolean activated = false;
|
protected boolean activated = false;
|
||||||
protected boolean worksFaceDown = false;
|
protected boolean worksFaceDown = false;
|
||||||
protected int sourceObjectZoneChangeCounter;
|
protected int sourceObjectZoneChangeCounter;
|
||||||
protected List<Watcher> watchers = new ArrayList<>(); // access to it by GetWatchers only (it's can be override by some abilities)
|
protected List<Watcher> watchers = new ArrayList<>(); // access to it by GetWatchers only (it can be overridden by some abilities)
|
||||||
protected List<Ability> subAbilities = null;
|
protected List<Ability> subAbilities = null;
|
||||||
protected boolean canFizzle = true;
|
protected boolean canFizzle = true;
|
||||||
protected TargetAdjuster targetAdjuster = null;
|
protected TargetAdjuster targetAdjuster = null;
|
||||||
|
@ -1243,10 +1242,13 @@ public abstract class AbilityImpl implements Ability {
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Dynamic cost modification for ability.
|
* Dynamic cost modification for ability.<br>
|
||||||
* Example: if it need stack related info (like real targets) then must check two states (game.inCheckPlayableState):
|
* Example: if it need stack related info (like real targets) then must
|
||||||
* 1. In playable state it must check all possible use cases (e.g. allow to reduce on any available target and modes)
|
* check two states (game.inCheckPlayableState): <br>
|
||||||
* 2. In real cast state it must check current use case (e.g. real selected targets and modes)
|
* 1. In playable state it must check all possible use cases (e.g. allow to
|
||||||
|
* reduce on any available target and modes) <br>
|
||||||
|
* 2. In real cast state it must check current use case (e.g. real selected
|
||||||
|
* targets and modes)
|
||||||
*
|
*
|
||||||
* @param costAdjuster
|
* @param costAdjuster
|
||||||
*/
|
*/
|
||||||
|
|
|
@ -348,26 +348,34 @@ public class ManaOptions extends ArrayList<Mana> {
|
||||||
private boolean subtractCostAddMana(Mana cost, Mana manaToAdd, boolean onlyManaCosts, Mana currentMana) {
|
private boolean subtractCostAddMana(Mana cost, Mana manaToAdd, boolean onlyManaCosts, Mana currentMana) {
|
||||||
boolean oldManaWasReplaced = false; // true if the newly created mana includes all mana possibilities of the old
|
boolean oldManaWasReplaced = false; // true if the newly created mana includes all mana possibilities of the old
|
||||||
boolean repeatable = false;
|
boolean repeatable = false;
|
||||||
if (manaToAdd.getAny() == 1 && manaToAdd.count() == 1 && onlyManaCosts) {
|
if ((manaToAdd.countColored() > 0 || manaToAdd.getAny() > 0) && manaToAdd.count() > 0 && onlyManaCosts) {
|
||||||
// deactivated because it does cause loops TODO: Find reason
|
// deactivated because it does cause loops TODO: Find reason
|
||||||
repeatable = true; // only replace to any with mana costs only will be repeated if able
|
repeatable = true; // only replace to any with mana costs only will be repeated if able
|
||||||
}
|
}
|
||||||
Mana prevMana = currentMana.copy();
|
Mana prevMana = currentMana.copy();
|
||||||
if (currentMana.includesMana(cost)) { // it can be paid
|
if (currentMana.includesMana(cost)) { // cost can be paid
|
||||||
// generic mana costs can be paid with different colored mana, can lead to different color combinations
|
// generic mana costs can be paid with different colored mana, can lead to different color combinations
|
||||||
if (cost.getGeneric() > 0 && cost.getGeneric() > (currentMana.getGeneric() + currentMana.getColorless())) {
|
if (cost.getGeneric() > 0 && cost.getGeneric() > (currentMana.getGeneric() + currentMana.getColorless())) {
|
||||||
Mana coloredCost = cost.copy();
|
|
||||||
coloredCost.setGeneric(0);
|
|
||||||
currentMana.subtract(coloredCost);
|
|
||||||
for (Mana payCombination : getPossiblePayCombinations(cost.getGeneric(), currentMana)) {
|
for (Mana payCombination : getPossiblePayCombinations(cost.getGeneric(), currentMana)) {
|
||||||
Mana newMana = currentMana.copy();
|
Mana currentManaCopy = currentMana.copy();
|
||||||
newMana.subtract(payCombination);
|
while (currentManaCopy.includesMana(payCombination)) { // loop for multiple usage if possible
|
||||||
newMana.add(manaToAdd);
|
boolean newCombinations = false;
|
||||||
Mana moreValuable = Mana.getMoreValuableMana(prevMana, newMana);
|
|
||||||
if (!prevMana.equals(moreValuable)) {
|
Mana newMana = currentManaCopy.copy();
|
||||||
this.add(newMana);
|
newMana.subtract(payCombination);
|
||||||
if (moreValuable != null) {
|
newMana.add(manaToAdd);
|
||||||
oldManaWasReplaced = true; // the new mana includes all possibilities of the old one
|
// Mana moreValuable = Mana.getMoreValuableMana(currentMana, newMana);
|
||||||
|
if (!isExistingManaCombination(newMana)) {
|
||||||
|
this.add(newMana); // add the new combination
|
||||||
|
newCombinations = true; // repeat the while as long there are new combinations and usage is repeatable
|
||||||
|
currentManaCopy = newMana.copy();
|
||||||
|
Mana moreValuable = Mana.getMoreValuableMana(currentMana, newMana);
|
||||||
|
if (!oldManaWasReplaced && newMana.equals(moreValuable)) {
|
||||||
|
oldManaWasReplaced = true; // the new mana includes all possibilities of the old one, so no need to add it after return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!newCombinations || !repeatable) {
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -447,6 +455,16 @@ public class ManaOptions extends ArrayList<Mana> {
|
||||||
return payCombinations;
|
return payCombinations;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private boolean isExistingManaCombination(Mana newMana) {
|
||||||
|
for (Mana mana : this) {
|
||||||
|
Mana moreValuable = Mana.getMoreValuableMana(mana, newMana);
|
||||||
|
if (mana.equals(moreValuable)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
private void addManaCombination(Mana mana, Mana existingMana, List<Mana> payCombinations, List<String> payCombinationsStrings) {
|
private void addManaCombination(Mana mana, Mana existingMana, List<Mana> payCombinations, List<String> payCombinationsStrings) {
|
||||||
Mana newMana = existingMana.copy();
|
Mana newMana = existingMana.copy();
|
||||||
newMana.add(mana);
|
newMana.add(mana);
|
||||||
|
@ -478,4 +496,20 @@ public class ManaOptions extends ArrayList<Mana> {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks if the given mana (cost) is already included in one available mana
|
||||||
|
* option
|
||||||
|
*
|
||||||
|
* @param mana
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
public boolean enough(Mana mana) {
|
||||||
|
for (Mana avail : this) {
|
||||||
|
if (mana.enough(avail)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -3157,10 +3157,8 @@ public abstract class PlayerImpl implements Player, Serializable {
|
||||||
game.getContinuousEffects().costModification(copyAbility, game);
|
game.getContinuousEffects().costModification(copyAbility, game);
|
||||||
|
|
||||||
for (Mana mana : copyAbility.getManaCostsToPay().getOptions()) {
|
for (Mana mana : copyAbility.getManaCostsToPay().getOptions()) {
|
||||||
for (Mana avail : availableMana) {
|
if (availableMana.enough(mana)) {
|
||||||
if (mana.enough(avail)) {
|
return true;
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3196,10 +3194,8 @@ public abstract class PlayerImpl implements Player, Serializable {
|
||||||
game.getContinuousEffects().costModification(copyAbility, game);
|
game.getContinuousEffects().costModification(copyAbility, game);
|
||||||
|
|
||||||
for (Mana mana : copyAbility.getManaCostsToPay().getOptions()) {
|
for (Mana mana : copyAbility.getManaCostsToPay().getOptions()) {
|
||||||
for (Mana avail : availableMana) {
|
if (availableMana.enough(mana)) {
|
||||||
if (mana.enough(avail)) {
|
return true;
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -3247,7 +3243,7 @@ public abstract class PlayerImpl implements Player, Serializable {
|
||||||
|
|
||||||
protected ActivatedAbility findActivatedAbilityFromAlternativeSourceCost(MageObject object, ManaOptions availableMana, Ability ability, Game game) {
|
protected ActivatedAbility findActivatedAbilityFromAlternativeSourceCost(MageObject object, ManaOptions availableMana, Ability ability, Game game) {
|
||||||
// return play ability that can activate AlternativeSourceCosts
|
// return play ability that can activate AlternativeSourceCosts
|
||||||
if (ability instanceof AlternativeSourceCosts && !(object instanceof Permanent)) {
|
if (ability instanceof AlternativeSourceCosts && object != null && !(object instanceof Permanent)) {
|
||||||
ActivatedAbility playAbility = null;
|
ActivatedAbility playAbility = null;
|
||||||
if (object.isLand()) {
|
if (object.isLand()) {
|
||||||
playAbility = (PlayLandAbility) CardUtil.getAbilities(object, game).stream().filter(a -> a instanceof PlayLandAbility).findFirst().orElse(null);
|
playAbility = (PlayLandAbility) CardUtil.getAbilities(object, game).stream().filter(a -> a instanceof PlayLandAbility).findFirst().orElse(null);
|
||||||
|
@ -3263,8 +3259,8 @@ public abstract class PlayerImpl implements Player, Serializable {
|
||||||
// Even mana cost can't be checked here without lookahead
|
// Even mana cost can't be checked here without lookahead
|
||||||
// So make it available all the time
|
// So make it available all the time
|
||||||
boolean canUse;
|
boolean canUse;
|
||||||
if (ability instanceof MorphAbility) {
|
if (ability instanceof MorphAbility && object instanceof Card && game.canPlaySorcery(getId())) {
|
||||||
canUse = game.canPlaySorcery(playerId) && ((MorphAbility) ability).isAvailable(playAbility, game);
|
canUse = canPlayCardByAlternateCost((Card) object, availableMana, ability, game);
|
||||||
} else {
|
} else {
|
||||||
canUse = canPlay(playAbility, availableMana, object, game); // canPlay already checks alternative source costs and all conditions
|
canUse = canPlay(playAbility, availableMana, object, game); // canPlay already checks alternative source costs and all conditions
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue