added X cost options to AI

This commit is contained in:
BetaSteward 2011-02-24 22:01:04 -05:00
parent d1f0f5bf43
commit 1945538041
18 changed files with 150 additions and 31 deletions

View file

@ -669,7 +669,7 @@ public class ComputerPlayer<T extends ComputerPlayer<T>> extends PlayerImpl<T> i
} }
@Override @Override
public boolean playXMana(VariableManaCost cost, Game game) { public boolean playXMana(VariableManaCost cost, ManaCosts<ManaCost> costs, Game game) {
logger.debug("playXMana"); logger.debug("playXMana");
//put everything into X //put everything into X
for (Permanent perm: this.getAvailableManaProducers(game)) { for (Permanent perm: this.getAvailableManaProducers(game)) {

View file

@ -42,9 +42,14 @@ import java.util.concurrent.TimeoutException;
import mage.Constants.Outcome; import mage.Constants.Outcome;
import mage.Constants.PhaseStep; import mage.Constants.PhaseStep;
import mage.Constants.RangeOfInfluence; import mage.Constants.RangeOfInfluence;
import mage.Mana;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.ActivatedAbility; import mage.abilities.ActivatedAbility;
import mage.abilities.common.PassAbility; import mage.abilities.common.PassAbility;
import mage.abilities.costs.mana.GenericManaCost;
import mage.abilities.costs.mana.ManaCost;
import mage.abilities.costs.mana.ManaCosts;
import mage.abilities.costs.mana.VariableManaCost;
import mage.abilities.effects.Effect; import mage.abilities.effects.Effect;
import mage.abilities.effects.SearchEffect; import mage.abilities.effects.SearchEffect;
import mage.cards.Cards; import mage.cards.Cards;
@ -497,6 +502,20 @@ public class ComputerPlayer2 extends ComputerPlayer<ComputerPlayer2> implements
return true; return true;
} }
@Override
public boolean playXMana(VariableManaCost cost, ManaCosts<ManaCost> costs, Game game) {
//SimulatedPlayer.simulateVariableCosts method adds a generic mana cost for each option
for (ManaCost manaCost: costs) {
if (manaCost instanceof GenericManaCost) {
cost.setPayment(manaCost.getPayment());
logger.debug("using X = " + cost.getPayment().count());
break;
}
}
cost.setPaid();
return true;
}
public void playNext(Game game, UUID activePlayerId, SimulationNode node) { public void playNext(Game game, UUID activePlayerId, SimulationNode node) {
boolean skip = false; boolean skip = false;
while (true) { while (true) {
@ -611,12 +630,15 @@ public class ComputerPlayer2 extends ComputerPlayer<ComputerPlayer2> implements
@Override @Override
public void selectAttackers(Game game) { public void selectAttackers(Game game) {
logger.debug("selectAttackers"); if (logger.isDebugEnabled() && combat == null || combat.getGroups().isEmpty())
logger.debug("not attacking");
if (combat != null) { if (combat != null) {
UUID opponentId = game.getCombat().getDefenders().iterator().next(); UUID opponentId = game.getCombat().getDefenders().iterator().next();
for (UUID attackerId: combat.getAttackers()) { for (UUID attackerId: combat.getAttackers()) {
this.declareAttacker(attackerId, opponentId, game); this.declareAttacker(attackerId, opponentId, game);
} if (logger.isDebugEnabled())
logger.debug("attacking with:" + game.getPermanent(attackerId).getName());
}
} }
} }
@ -629,6 +651,8 @@ public class ComputerPlayer2 extends ComputerPlayer<ComputerPlayer2> implements
if (i < combat.getGroups().size()) { if (i < combat.getGroups().size()) {
for (UUID blockerId: combat.getGroups().get(i).getBlockers()) { for (UUID blockerId: combat.getGroups().get(i).getBlockers()) {
this.declareBlocker(blockerId, groups.get(i).getAttackers().get(0), game); this.declareBlocker(blockerId, groups.get(i).getAttackers().get(0), game);
if (logger.isDebugEnabled())
logger.debug("blocking with:" + game.getPermanent(blockerId).getName());
} }
} }
} }

View file

@ -1,6 +1,29 @@
/* /*
* To change this template, choose Tools | Templates * Copyright 2010 BetaSteward_at_googlemail.com. All rights reserved.
* and open the template in the editor. *
* 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.player.ai; package mage.player.ai;
@ -13,6 +36,8 @@ import mage.abilities.keyword.DoubleStrikeAbility;
import mage.abilities.keyword.FirstStrikeAbility; import mage.abilities.keyword.FirstStrikeAbility;
import mage.abilities.keyword.TrampleAbility; import mage.abilities.keyword.TrampleAbility;
import mage.abilities.mana.ManaAbility; import mage.abilities.mana.ManaAbility;
import mage.counters.BoostCounter;
import mage.counters.Counter;
import mage.game.Game; import mage.game.Game;
import mage.game.permanent.Permanent; import mage.game.permanent.Permanent;
import mage.players.Player; import mage.players.Player;
@ -39,9 +64,9 @@ public class GameStateEvaluator {
Player opponent = game.getPlayer(game.getOpponents(playerId).iterator().next()); Player opponent = game.getPlayer(game.getOpponents(playerId).iterator().next());
if (game.isGameOver()) { if (game.isGameOver()) {
if (player.hasLost() || opponent.hasWon()) if (player.hasLost() || opponent.hasWon())
return Integer.MIN_VALUE; return Integer.MIN_VALUE + 1;
if (opponent.hasLost() || player.hasWon()) if (opponent.hasLost() || player.hasWon())
return Integer.MAX_VALUE; return Integer.MAX_VALUE - 1;
} }
int lifeScore = (player.getLife() - opponent.getLife()) * LIFE_FACTOR; int lifeScore = (player.getLife() - opponent.getLife()) * LIFE_FACTOR;
int permanentScore = 0; int permanentScore = 0;
@ -73,6 +98,11 @@ public class GameStateEvaluator {
if (!(ability instanceof ManaAbility) && ability.canActivate(ability.getControllerId(), game)) if (!(ability instanceof ManaAbility) && ability.canActivate(ability.getControllerId(), game))
value += ability.getEffects().size(); value += ability.getEffects().size();
} }
for (Counter counter: permanent.getCounters().values()) {
if (!(counter instanceof BoostCounter)) {
value += counter.getCount();
}
}
value += permanent.getAbilities().getStaticAbilities(Zone.BATTLEFIELD).size(); value += permanent.getAbilities().getStaticAbilities(Zone.BATTLEFIELD).size();
value += permanent.getAbilities().getTriggeredAbilities(Zone.BATTLEFIELD).size(); value += permanent.getAbilities().getTriggeredAbilities(Zone.BATTLEFIELD).size();
value += permanent.getManaCost().convertedManaCost(); value += permanent.getManaCost().convertedManaCost();

View file

@ -38,6 +38,10 @@ import java.util.concurrent.ConcurrentLinkedQueue;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.TriggeredAbility; import mage.abilities.TriggeredAbility;
import mage.abilities.common.PassAbility; import mage.abilities.common.PassAbility;
import mage.abilities.costs.mana.GenericManaCost;
import mage.abilities.costs.mana.ManaCost;
import mage.abilities.costs.mana.ManaCosts;
import mage.abilities.costs.mana.VariableManaCost;
import mage.abilities.mana.ManaOptions; import mage.abilities.mana.ManaOptions;
import mage.choices.Choice; import mage.choices.Choice;
import mage.filter.FilterAbility; import mage.filter.FilterAbility;
@ -99,13 +103,23 @@ public class SimulatedPlayer extends ComputerPlayer<SimulatedPlayer> {
for (Ability ability: playables) { for (Ability ability: playables) {
List<Ability> options = game.getPlayer(playerId).getPlayableOptions(ability, game); List<Ability> options = game.getPlayer(playerId).getPlayableOptions(ability, game);
if (options.size() == 0) { if (options.size() == 0) {
allActions.add(ability); if (ability.getManaCosts().getVariableCosts().size() > 0) {
simulateVariableCosts(ability, game);
}
else {
allActions.add(ability);
}
// simulateAction(game, previousActions, ability); // simulateAction(game, previousActions, ability);
} }
else { else {
// ExecutorService simulationExecutor = Executors.newFixedThreadPool(4); // ExecutorService simulationExecutor = Executors.newFixedThreadPool(4);
for (Ability option: options) { for (Ability option: options) {
allActions.add(option); if (ability.getManaCosts().getVariableCosts().size() > 0) {
simulateVariableCosts(option, game);
}
else {
allActions.add(option);
}
// SimulationWorker worker = new SimulationWorker(game, this, previousActions, option); // SimulationWorker worker = new SimulationWorker(game, this, previousActions, option);
// simulationExecutor.submit(worker); // simulationExecutor.submit(worker);
} }
@ -126,6 +140,30 @@ public class SimulatedPlayer extends ComputerPlayer<SimulatedPlayer> {
// } // }
// } // }
//add a generic mana cost for each amount possible
protected void simulateVariableCosts(Ability ability, Game game) {
int numAvailable = getAvailableManaProducers(game).size();
for (int i = 0; i < numAvailable; i++) {
Ability newAbility = ability.copy();
newAbility.addManaCost(new GenericManaCost(i));
allActions.add(newAbility);
}
}
@Override
public boolean playXMana(VariableManaCost cost, ManaCosts<ManaCost> costs, Game game) {
//simulateVariableCosts method adds a generic mana cost for each option
for (ManaCost manaCost: costs) {
if (manaCost instanceof GenericManaCost) {
cost.setPayment(manaCost.getPayment());
logger.debug("simulating -- X = " + cost.getPayment().count());
break;
}
}
cost.setPaid();
return true;
}
public List<Combat> addAttackers(Game game) { public List<Combat> addAttackers(Game game) {
Map<Integer, Combat> engagements = new HashMap<Integer, Combat>(); Map<Integer, Combat> engagements = new HashMap<Integer, Combat>();
//useful only for two player games - will only attack first opponent //useful only for two player games - will only attack first opponent

View file

@ -51,6 +51,7 @@ import mage.abilities.ActivatedAbility;
import mage.abilities.SpecialAction; import mage.abilities.SpecialAction;
import mage.abilities.costs.common.SacrificeSourceCost; import mage.abilities.costs.common.SacrificeSourceCost;
import mage.abilities.costs.mana.ManaCost; import mage.abilities.costs.mana.ManaCost;
import mage.abilities.costs.mana.ManaCosts;
import mage.abilities.costs.mana.VariableManaCost; import mage.abilities.costs.mana.VariableManaCost;
import mage.cards.decks.Deck; import mage.cards.decks.Deck;
import mage.choices.ChoiceImpl; import mage.choices.ChoiceImpl;
@ -355,7 +356,7 @@ public class HumanPlayer extends PlayerImpl<HumanPlayer> {
} }
@Override @Override
public boolean playXMana(VariableManaCost cost, Game game) { public boolean playXMana(VariableManaCost cost, ManaCosts<ManaCost> costs, Game game) {
game.firePlayXManaEvent(playerId, "Pay {X}: {X}=" + cost.getAmount()); game.firePlayXManaEvent(playerId, "Pay {X}: {X}=" + cost.getAmount());
waitForResponse(); waitForResponse();
if (response.getBoolean() != null) { if (response.getBoolean() != null) {

View file

@ -39,6 +39,7 @@ public interface ManaCost extends Cost {
public Mana getMana(); public Mana getMana();
public Mana getPayment(); public Mana getPayment();
public void assignPayment(ManaPool pool); public void assignPayment(ManaPool pool);
public void setPayment(Mana mana);
@Override @Override
public String getText(); public String getText();
public ManaCost getUnpaid(); public ManaCost getUnpaid();

View file

@ -209,6 +209,11 @@ public abstract class ManaCostImpl<T extends ManaCostImpl<T>> extends CostImpl<T
this.paid = true; this.paid = true;
} }
@Override
public void setPayment(Mana mana) {
this.payment.add(mana);
}
protected void addColoredOption(ColoredManaSymbol symbol) { protected void addColoredOption(ColoredManaSymbol symbol) {
switch (symbol) { switch (symbol) {
case B: case B:

View file

@ -116,7 +116,7 @@ public class ManaCostsImpl<T extends ManaCost> extends ArrayList<T> implements M
for (ManaCost cost: this.getUnpaidVariableCosts()) { for (ManaCost cost: this.getUnpaidVariableCosts()) {
VariableManaCost vCost = (VariableManaCost) cost; VariableManaCost vCost = (VariableManaCost) cost;
while (!vCost.isPaid()) { while (!vCost.isPaid()) {
if (player.playXMana(vCost, game)) if (player.playXMana(vCost, (ManaCosts<ManaCost>) this, game))
vCost.assignPayment(player.getManaPool()); vCost.assignPayment(player.getManaPool());
else else
return false; return false;
@ -155,6 +155,9 @@ public class ManaCostsImpl<T extends ManaCost> extends ArrayList<T> implements M
return variableCosts; return variableCosts;
} }
@Override
public void setPayment(Mana mana) { }
@Override @Override
public void assignPayment(ManaPool pool) { public void assignPayment(ManaPool pool) {
//attempt to pay colored costs first //attempt to pay colored costs first

View file

@ -38,7 +38,6 @@ import mage.players.ManaPool;
*/ */
public class VariableManaCost extends ManaCostImpl<VariableManaCost> implements VariableCost { public class VariableManaCost extends ManaCostImpl<VariableManaCost> implements VariableCost {
protected Mana manaPaid = new Mana();
protected int multiplier; protected int multiplier;
public VariableManaCost() { public VariableManaCost() {
@ -53,7 +52,6 @@ public class VariableManaCost extends ManaCostImpl<VariableManaCost> implements
public VariableManaCost(VariableManaCost cost) { public VariableManaCost(VariableManaCost cost) {
super(cost); super(cost);
this.manaPaid = cost.manaPaid.copy();
this.multiplier = cost.multiplier; this.multiplier = cost.multiplier;
} }
@ -64,7 +62,7 @@ public class VariableManaCost extends ManaCostImpl<VariableManaCost> implements
@Override @Override
public void assignPayment(ManaPool pool) { public void assignPayment(ManaPool pool) {
manaPaid.add(pool.getMana()); payment.add(pool.getMana());
pool.emptyPool(); pool.emptyPool();
} }
@ -87,19 +85,9 @@ public class VariableManaCost extends ManaCostImpl<VariableManaCost> implements
return this; return this;
} }
@Override
public void clearPaid() {
paid = false;
manaPaid.clear();
}
public Mana getPaid() {
return manaPaid;
}
@Override @Override
public int getAmount() { public int getAmount() {
return manaPaid.count() / multiplier; return payment.count() / multiplier;
} }
@Override @Override

View file

@ -59,6 +59,7 @@ public interface Card extends MageObject {
public boolean moveToZone(Zone zone, UUID sourceId, Game game, boolean flag); public boolean moveToZone(Zone zone, UUID sourceId, Game game, boolean flag);
public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game); public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game);
public boolean cast(Game game, Zone fromZone, SpellAbility ability, UUID controllerId);
public boolean putOntoBattlefield(Game game, Zone fromZone, UUID sourceId, UUID controllerId); public boolean putOntoBattlefield(Game game, Zone fromZone, UUID sourceId, UUID controllerId);
public void checkTriggers(Zone zone, GameEvent event, Game game); public void checkTriggers(Zone zone, GameEvent event, Game game);

View file

@ -246,6 +246,28 @@ public abstract class CardImpl<T extends CardImpl<T>> extends MageObjectImpl<T>
return false; return false;
} }
@Override
public boolean cast(Game game, Zone fromZone, SpellAbility ability, UUID controllerId) {
ZoneChangeEvent event = new ZoneChangeEvent(this.objectId, ability.getId(), controllerId, fromZone, Zone.STACK);
if (!game.replaceEvent(event)) {
if (event.getFromZone() != null) {
switch (event.getFromZone()) {
case GRAVEYARD:
game.getPlayer(ownerId).removeFromGraveyard(this, game);
break;
default:
//logger.warning("moveToZone, not fully implemented: from="+event.getFromZone() + ", to="+event.getToZone());
}
game.rememberLKI(objectId, event.getFromZone(), this);
}
game.getStack().push(new Spell(this, ability.copy(), controllerId));
game.setZone(objectId, event.getToZone());
game.fireEvent(event);
return game.getZone(objectId) == Zone.STACK;
}
return false;
}
@Override @Override
public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game) { public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game) {
Zone fromZone = game.getZone(objectId); Zone fromZone = game.getZone(objectId);

View file

@ -297,17 +297,17 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
@Override @Override
public boolean moveToZone(Zone zone, UUID sourceId, Game game, boolean flag) { public boolean moveToZone(Zone zone, UUID sourceId, Game game, boolean flag) {
throw new UnsupportedOperationException("Not supported yet."); throw new UnsupportedOperationException("Unsupported operation");
} }
@Override @Override
public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game) { public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game) {
throw new UnsupportedOperationException("Not supported yet."); throw new UnsupportedOperationException("Unsupported operation");
} }
@Override @Override
public boolean putOntoBattlefield(Game game, Zone fromZone, UUID sourceId, UUID controllerId) { public boolean putOntoBattlefield(Game game, Zone fromZone, UUID sourceId, UUID controllerId) {
throw new UnsupportedOperationException("Not supported yet."); throw new UnsupportedOperationException("Unsupported operation");
} }
@Override @Override
@ -322,6 +322,11 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
@Override @Override
public List<Mana> getMana() { public List<Mana> getMana() {
throw new UnsupportedOperationException("Not supported yet."); return card.getMana();
}
@Override
public boolean cast(Game game, Zone fromZone, SpellAbility ability, UUID controllerId) {
throw new UnsupportedOperationException("Unsupported operation");
} }
} }

View file

@ -43,6 +43,7 @@ import mage.abilities.SpellAbility;
import mage.abilities.TriggeredAbilities; import mage.abilities.TriggeredAbilities;
import mage.abilities.TriggeredAbility; import mage.abilities.TriggeredAbility;
import mage.abilities.costs.mana.ManaCost; import mage.abilities.costs.mana.ManaCost;
import mage.abilities.costs.mana.ManaCosts;
import mage.abilities.costs.mana.VariableManaCost; import mage.abilities.costs.mana.VariableManaCost;
import mage.abilities.effects.ReplacementEffect; import mage.abilities.effects.ReplacementEffect;
import mage.abilities.mana.ManaOptions; import mage.abilities.mana.ManaOptions;
@ -149,7 +150,7 @@ public interface Player extends MageItem, Copyable<Player> {
public abstract boolean chooseUse(Outcome outcome, String message, Game game); public abstract boolean chooseUse(Outcome outcome, String message, Game game);
public abstract boolean choose(Outcome outcome, Choice choice, Game game); public abstract boolean choose(Outcome outcome, Choice choice, Game game);
public abstract boolean playMana(ManaCost unpaid, Game game); public abstract boolean playMana(ManaCost unpaid, Game game);
public abstract boolean playXMana(VariableManaCost cost, Game game); public abstract boolean playXMana(VariableManaCost cost, ManaCosts<ManaCost> costs, Game game);
public abstract int chooseEffect(List<ReplacementEffect> rEffects, Game game); public abstract int chooseEffect(List<ReplacementEffect> rEffects, Game game);
public abstract TriggeredAbility chooseTriggeredAbility(TriggeredAbilities abilities, Game game); public abstract TriggeredAbility chooseTriggeredAbility(TriggeredAbilities abilities, Game game);
public abstract void selectAttackers(Game game); public abstract void selectAttackers(Game game);

View file

@ -375,7 +375,7 @@ public abstract class PlayerImpl<T extends PlayerImpl<T>> implements Player, Ser
if (!game.replaceEvent(GameEvent.getEvent(GameEvent.EventType.CAST_SPELL, ability.getId(), playerId))) { if (!game.replaceEvent(GameEvent.getEvent(GameEvent.EventType.CAST_SPELL, ability.getId(), playerId))) {
game.bookmarkState(); game.bookmarkState();
removeFromHand(card, game); removeFromHand(card, game);
card.moveToZone(Zone.STACK, ability.getId(), game, false); card.cast(game, Zone.HAND, ability, playerId);
Ability spellAbility = game.getStack().getSpell(ability.getId()).getSpellAbility(); Ability spellAbility = game.getStack().getSpell(ability.getId()).getSpellAbility();
if (spellAbility.activate(game, noMana)) { if (spellAbility.activate(game, noMana)) {
for (KickerAbility kicker: card.getAbilities().getKickerAbilities()) { for (KickerAbility kicker: card.getAbilities().getKickerAbilities()) {