Update to split cards functionality. Fuse and flashback works now. No correct ability filtering yet for split cards.

This commit is contained in:
LevelX2 2013-05-07 00:44:57 +02:00
parent 623264d9c0
commit b01cb551b7
11 changed files with 301 additions and 226 deletions

View file

@ -108,7 +108,7 @@ class ChainOfVaporEffect extends OneShotEffect<ChainOfVaporEffect> {
copy.setCopiedSpell(true);
game.getStack().push(copy);
copy.chooseNewTargets(game, source.getControllerId());
String activateMessage = copy.getSpellAbility().getActivatedMessage(game);
String activateMessage = copy.getActivatedMessage(game);
if (activateMessage.startsWith(" casts ")) {
activateMessage = activateMessage.substring(6);
}

View file

@ -154,6 +154,27 @@ public final class Constants {
}
}
public enum SpellAbilityType {
BASE("Basic SpellAbility"),
SPLIT("Split SpellAbility"),
SPLIT_FUSED("Split SpellAbility"),
SPLIT_LEFT("LeftSplit SpellAbility"),
SPLIT_RIGHT("RightSplit SpellAbility"),
MODE("Mode SpellAbility"),
SPLICE("Spliced SpellAbility");
private String text;
SpellAbilityType(String text) {
this.text = text;
}
@Override
public String toString() {
return text;
}
}
public enum EffectType {
ONESHOT("One Shot Effect"),

View file

@ -40,6 +40,7 @@ import mage.abilities.keyword.FlashAbility;
import mage.game.Game;
import java.util.UUID;
import mage.Constants.SpellAbilityType;
/**
*
@ -47,23 +48,39 @@ import java.util.UUID;
*/
public class SpellAbility extends ActivatedAbilityImpl<SpellAbility> {
private SpellAbilityType spellAbilityType;
public SpellAbility(ManaCost cost, String cardName) {
this(cost, cardName, Zone.HAND);
}
public SpellAbility(ManaCost cost, String cardName, Zone zone) {
super(AbilityType.SPELL, zone);
this.addManaCost(cost);
this.name = "Cast " + cardName;
this(cost, cardName, zone, SpellAbilityType.BASE);
}
public SpellAbility(Cost cost, String cardName, Effect effect, Zone zone) {
super(zone, effect, cost);
this.name = "Cast " + cardName;
public SpellAbility(ManaCost cost, String cardName, Zone zone, SpellAbilityType spellAbilityType) {
super(AbilityType.SPELL, zone);
this.spellAbilityType = spellAbilityType;
this.addManaCost(cost);
switch(spellAbilityType) {
case SPLIT_FUSED:
this.name = "Cast fused " + cardName;
break;
default:
this.name = "Cast " + cardName;
}
}
// public SpellAbility(Cost cost, String cardName, Effect effect, Zone zone) {
// super(zone, effect, cost);
// this.spellAbilityType = SpellAbilityType.BASE;
// this.name = "Cast " + cardName;
// }
public SpellAbility(SpellAbility ability) {
super(ability);
this.spellAbilityType = ability.spellAbilityType;
}
@Override
@ -124,4 +141,12 @@ public class SpellAbility extends ActivatedAbilityImpl<SpellAbility> {
return spell;
}
public SpellAbilityType getSpellAbilityType() {
return spellAbilityType;
}
public void setSpellAbilityType(SpellAbilityType spellAbilityType) {
this.spellAbilityType = spellAbilityType;
}
}

View file

@ -61,7 +61,7 @@ public class CopyTargetSpellEffect extends OneShotEffect<CopyTargetSpellEffect>
game.getStack().push(copy);
copy.chooseNewTargets(game, source.getControllerId());
Player player = game.getPlayer(source.getControllerId());
String activateMessage = copy.getSpellAbility().getActivatedMessage(game);
String activateMessage = copy.getActivatedMessage(game);
if (activateMessage.startsWith(" casts ")) {
activateMessage = activateMessage.substring(6);
}

View file

@ -28,14 +28,18 @@
package mage.abilities.keyword;
import mage.Constants;
import mage.Constants.SpellAbilityType;
import static mage.Constants.SpellAbilityType.SPLIT_LEFT;
import mage.abilities.Ability;
import mage.abilities.ActivatedAbilityImpl;
import mage.abilities.DelayedTriggeredAbility;
import mage.abilities.SpellAbility;
import mage.abilities.costs.Cost;
import mage.abilities.effects.OneShotEffect;
import mage.abilities.effects.common.CreateDelayedTriggeredAbilityEffect;
import mage.abilities.effects.common.ExileSourceEffect;
import mage.cards.Card;
import mage.cards.SplitCard;
import mage.game.Game;
import mage.game.events.GameEvent;
import mage.game.events.ZoneChangeEvent;
@ -48,16 +52,22 @@ import mage.target.Target;
*/
public class FlashbackAbility extends /*SpellAbility*/ ActivatedAbilityImpl<FlashbackAbility> {
private Constants.SpellAbilityType spellAbilityType;
private String abilityName;
public FlashbackAbility(Cost cost, Constants.TimingRule timingRule) {
//super(cost, "", new FlashbackEffect(), Constants.Zone.GRAVEYARD);
super(Constants.Zone.GRAVEYARD, new FlashbackEffect(), cost);
this.timing = timingRule;
this.usesStack = false;
this.spellAbilityType = SpellAbilityType.BASE;
this.addEffect(new CreateDelayedTriggeredAbilityEffect(new FlashbackTriggeredAbility()));
}
public FlashbackAbility(final FlashbackAbility ability) {
super(ability);
this.spellAbilityType = ability.spellAbilityType;
this.abilityName = ability.abilityName;
}
@Override
@ -65,11 +75,16 @@ public class FlashbackAbility extends /*SpellAbility*/ ActivatedAbilityImpl<Flas
return new FlashbackAbility(this);
}
@Override
public String getRule(boolean all) {
return this.getRule();
}
@Override
public String getRule() {
StringBuilder sbRule = new StringBuilder("Flashback");
if (costs.size() > 0) {
sbRule.append("--");
sbRule.append(" - ");
} else {
sbRule.append(" ");
}
@ -80,9 +95,26 @@ public class FlashbackAbility extends /*SpellAbility*/ ActivatedAbilityImpl<Flas
sbRule.append(costs.getText());
sbRule.append(".");
}
if (abilityName != null) {
sbRule.append(" ");
sbRule.append(abilityName);
}
sbRule.append(" <i>(You may cast this card from your graveyard for its flashback cost. Then exile it.)</i>");
return sbRule.toString();
}
public void setSpellAbilityType(SpellAbilityType spellAbilityType) {
this.spellAbilityType = spellAbilityType;
}
public SpellAbilityType getSpellAbilityType() {
return this.spellAbilityType;
}
public void setAbilityName(String abilityName) {
this.abilityName = abilityName;
}
}
class FlashbackEffect extends OneShotEffect<FlashbackEffect> {
@ -107,13 +139,25 @@ class FlashbackEffect extends OneShotEffect<FlashbackEffect> {
if (card != null) {
Player controller = game.getPlayer(source.getControllerId());
if (controller != null) {
card.getSpellAbility().clear();
SpellAbility spellAbility;
switch(((FlashbackAbility) source).getSpellAbilityType()) {
case SPLIT_LEFT:
spellAbility = ((SplitCard)card).getLeftHalfCard().getSpellAbility();
break;
case SPLIT_RIGHT:
spellAbility = ((SplitCard)card).getRightHalfCard().getSpellAbility();
break;
default:
spellAbility = card.getSpellAbility();
}
spellAbility.clear();
int amount = source.getManaCostsToPay().getX();
card.getSpellAbility().getManaCostsToPay().setX(amount);
for (Target target : card.getSpellAbility().getTargets()) {
spellAbility.getManaCostsToPay().setX(amount);
for (Target target : spellAbility.getTargets()) {
target.setRequired(true);
}
return controller.cast(card.getSpellAbility(), game, true);
return controller.cast(spellAbility, game, true);
}
}
return false;

View file

@ -1,66 +0,0 @@
/*
* 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 mage.Constants.Zone;
import mage.abilities.SpellAbility;
import mage.abilities.costs.mana.ManaCosts;
import mage.cards.Card;
/**
*
* @author LevelX2
*/
public class FuseAbility extends SpellAbility {
public FuseAbility(Card card, ManaCosts costs) {
super(costs, "fused " + card.getName(), Zone.HAND);
}
public FuseAbility(final FuseAbility ability) {
super(ability);
}
@Override
public FuseAbility copy() {
return new FuseAbility(this);
}
@Override
public String getRule(boolean all) {
return getRule();
}
@Override
public String getRule() {
return "Fuse <i>(You may cast one or both halves of this card from your hand.)</i>";
}
}

View file

@ -49,6 +49,7 @@ import org.apache.log4j.Logger;
import java.lang.reflect.Constructor;
import java.util.*;
import mage.Constants.SpellAbilityType;
public abstract class CardImpl<T extends CardImpl<T>> extends MageObjectImpl<T> implements Card {
@ -75,6 +76,10 @@ public abstract class CardImpl<T extends CardImpl<T>> extends MageObjectImpl<T>
protected boolean splitCard;
public CardImpl(UUID ownerId, int cardNumber, String name, Rarity rarity, CardType[] cardTypes, String costs) {
this(ownerId, cardNumber, name, rarity, cardTypes, costs, SpellAbilityType.BASE);
}
public CardImpl(UUID ownerId, int cardNumber, String name, Rarity rarity, CardType[] cardTypes, String costs, SpellAbilityType spellAbilityType) {
this(ownerId, name);
this.rarity = rarity;
this.cardNumber = cardNumber;
@ -84,7 +89,7 @@ public abstract class CardImpl<T extends CardImpl<T>> extends MageObjectImpl<T>
addAbility(new PlayLandAbility(name));
}
else {
addAbility(new SpellAbility(manaCost, name));
addAbility(new SpellAbility(manaCost, name, Zone.HAND, spellAbilityType));
}
this.usesVariousArt = Character.isDigit(this.getClass().getName().charAt(this.getClass().getName().length()-1));
this.counters = new Counters();

View file

@ -31,16 +31,13 @@ package mage.cards;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
import mage.Constants;
import mage.Constants.CardType;
import mage.Constants.Rarity;
import mage.Constants.Zone;
import mage.Constants.SpellAbilityType;
import mage.abilities.Abilities;
import mage.abilities.AbilitiesImpl;
import mage.abilities.Ability;
import mage.abilities.SpellAbility;
import mage.abilities.keyword.FuseAbility;
import mage.game.Game;
import mage.watchers.Watcher;
/**
@ -50,16 +47,11 @@ import mage.watchers.Watcher;
public abstract class SplitCard<T extends SplitCard<T>> extends CardImpl<T> {
public enum ActiveCardHalf {
NONE, LEFT, RIGHT, BOTH
}
private Card leftHalfCard;
private Card rightHalfCard;
private ActiveCardHalf activeCardHalf = ActiveCardHalf.NONE;
public SplitCard(UUID ownerId, int cardNumber, String nameLeft, String nameRight, Rarity rarity, CardType[] cardTypes, String costsLeft, String costsRight) {
super(ownerId, cardNumber, new StringBuilder(nameLeft).append(" - ").append(nameRight).toString(), rarity, cardTypes, costsLeft + costsRight);
public SplitCard(UUID ownerId, int cardNumber, String nameLeft, String nameRight, Rarity rarity, CardType[] cardTypes, String costsLeft, String costsRight, boolean fused) {
super(ownerId, cardNumber, new StringBuilder(nameLeft).append(" - ").append(nameRight).toString(), rarity, cardTypes, costsLeft + costsRight, (fused ?SpellAbilityType.SPLIT_FUSED:SpellAbilityType.SPLIT));
this.createLeftHalfCard(nameLeft, costsLeft);
this.createRightHalfCard(nameRight, costsRight);
this.splitCard = true;
@ -69,7 +61,6 @@ public abstract class SplitCard<T extends SplitCard<T>> extends CardImpl<T> {
super(card);
this.leftHalfCard = card.leftHalfCard.copy();
this.rightHalfCard = card.rightHalfCard.copy();
this.activeCardHalf = card.activeCardHalf;
}
private Card createLeftHalfCard (String nameLeft, String costsLeft) {
@ -83,7 +74,7 @@ public abstract class SplitCard<T extends SplitCard<T>> extends CardImpl<T> {
private Card createRightHalfCard (String nameRight, String costsRight) {
CardType[] cardTypes = new CardType[getCardType().size()];
this.getCardType().toArray(cardTypes);
rightHalfCard = new LeftHalfCard(this.getOwnerId(), this.getCardNumber(), nameRight, this.rarity, cardTypes, costsRight);
rightHalfCard = new RightHalfCard(this.getOwnerId(), this.getCardNumber(), nameRight, this.rarity, cardTypes, costsRight);
rightHalfCard.getAbilities().setSourceId(objectId);
return rightHalfCard;
}
@ -98,96 +89,28 @@ public abstract class SplitCard<T extends SplitCard<T>> extends CardImpl<T> {
return rightHalfCard;
}
public ActiveCardHalf getActiveCardHalf() {
return activeCardHalf;
}
@Override
public boolean cast(Game game, Constants.Zone fromZone, SpellAbility ability, UUID controllerId) {
if (this.getAbilities().contains(ability)) {
activeCardHalf = ActiveCardHalf.BOTH;
} else if (leftHalfCard.getAbilities().contains(ability)) {
activeCardHalf = ActiveCardHalf.LEFT;
} else if (rightHalfCard.getAbilities().contains(ability)) {
activeCardHalf = ActiveCardHalf.RIGHT;
} else {
activeCardHalf = ActiveCardHalf.NONE;
}
if (super.cast(game, fromZone, ability, controllerId)) {
return true;
}
activeCardHalf = ActiveCardHalf.NONE;
return false;
}
@Override
public boolean moveToZone(Constants.Zone toZone, UUID sourceId, Game game, boolean flag, ArrayList<UUID> appliedEffects) {
if (super.moveToZone(toZone, sourceId, game, flag, appliedEffects)) {
if (!toZone.equals(Zone.STACK)) {
activeCardHalf = ActiveCardHalf.NONE;
}
return true;
}
return false;
}
@Override
public boolean moveToExile(UUID exileId, String name, UUID sourceId, Game game, ArrayList<UUID> appliedEffects) {
if(super.moveToExile(exileId, name, sourceId, game, appliedEffects)) {
activeCardHalf = ActiveCardHalf.NONE;
return true;
}
return false;
}
@Override
public Abilities<Ability> getAbilities(){
Abilities<Ability> allAbilites = new AbilitiesImpl<Ability>();
if (activeCardHalf.equals(ActiveCardHalf.NONE) || activeCardHalf.equals(ActiveCardHalf.LEFT)) {
allAbilites.addAll(leftHalfCard.getAbilities());
}
if (activeCardHalf.equals(ActiveCardHalf.NONE) || activeCardHalf.equals(ActiveCardHalf.RIGHT)) {
allAbilites.addAll(rightHalfCard.getAbilities());
}
for (Ability ability: super.getAbilities()) {
if (ability instanceof FuseAbility) {
for (Ability ability : super.getAbilities()) {
if (ability instanceof SpellAbility && !((SpellAbility)ability).getSpellAbilityType().equals(SpellAbilityType.SPLIT)) {
allAbilites.add(ability);
}
}
allAbilites.addAll(super.getAbilities());
allAbilites.addAll(leftHalfCard.getAbilities());
allAbilites.addAll(rightHalfCard.getAbilities());
return allAbilites;
}
@Override
public SpellAbility getSpellAbility() {
switch (activeCardHalf) {
case LEFT:
return leftHalfCard.getSpellAbility();
case RIGHT:
return rightHalfCard.getSpellAbility();
}
return null;
}
@Override
public List<String> getRules() {
List<String> rules = new ArrayList<String>();
if (activeCardHalf.equals(ActiveCardHalf.NONE) || activeCardHalf.equals(ActiveCardHalf.LEFT)) {
rules.add(new StringBuilder("<b>").append(leftHalfCard.getName()).append("<b/>").toString());
rules.addAll(leftHalfCard.getRules());
}
if (activeCardHalf.equals(ActiveCardHalf.NONE)) {
rules.add("<br/>");
}
if (activeCardHalf.equals(ActiveCardHalf.NONE) || activeCardHalf.equals(ActiveCardHalf.RIGHT)) {
rules.add(new StringBuilder("<b>").append(rightHalfCard.getName()).append("<b/>").toString());
rules.addAll(rightHalfCard.getRules());
}
for (Ability ability: super.getAbilities()) {
if (ability instanceof FuseAbility) {
rules.add("<br/>------------------------------------------------------------");
rules.add(ability.getRule());
}
rules.addAll(leftHalfCard.getRules());
rules.addAll(rightHalfCard.getRules());
if (getSpellAbility().getSpellAbilityType().equals(SpellAbilityType.SPLIT_FUSED)) {
rules.add("------------------------------------------------------------------------");
rules.add("Fuse (You may cast one or both halves of this card from your hand.)");
}
return rules;
}
@ -213,14 +136,9 @@ public abstract class SplitCard<T extends SplitCard<T>> extends CardImpl<T> {
@Override
public List<Watcher> getWatchers() {
List<Watcher> allWatchers = new ArrayList<Watcher>();
switch (activeCardHalf) {
case LEFT:
allWatchers.addAll(leftHalfCard.getWatchers());
break;
case RIGHT:
allWatchers.addAll(rightHalfCard.getWatchers());
break;
}
allWatchers.addAll(super.getWatchers());
allWatchers.addAll(leftHalfCard.getWatchers());
allWatchers.addAll(rightHalfCard.getWatchers());
return allWatchers;
}
}
@ -230,8 +148,8 @@ public abstract class SplitCard<T extends SplitCard<T>> extends CardImpl<T> {
*/
class LeftHalfCard extends CardImpl<LeftHalfCard> {
public LeftHalfCard(UUID ownerId, int cardNumber, String name, Rarity rarity, CardType[] cardTypes, String costs) {
super(ownerId, cardNumber, name, rarity, cardTypes, costs);
public LeftHalfCard(UUID ownerId, int cardNumber, String name, Rarity rarity, CardType[] cardTypes, String costs ) {
super(ownerId, cardNumber, name, rarity, cardTypes, costs, SpellAbilityType.SPLIT_LEFT);
}
public LeftHalfCard(final LeftHalfCard card) {
@ -242,6 +160,16 @@ class LeftHalfCard extends CardImpl<LeftHalfCard> {
public LeftHalfCard copy() {
return new LeftHalfCard(this);
}
@Override
public List<String> getRules() {
List<String> rules = new ArrayList<String>();
rules.add(new StringBuilder("<b>").append(this.getName()).append("<b/>").toString());
rules.addAll(super.getRules());
return rules;
}
}
/*
@ -250,7 +178,7 @@ class LeftHalfCard extends CardImpl<LeftHalfCard> {
class RightHalfCard extends CardImpl<RightHalfCard> {
public RightHalfCard(UUID ownerId, int cardNumber, String name, Rarity rarity, CardType[] cardTypes, String costs) {
super(ownerId, cardNumber, name, rarity, cardTypes, costs);
super(ownerId, cardNumber, name, rarity, cardTypes, costs, SpellAbilityType.SPLIT_RIGHT);
}
public RightHalfCard(final RightHalfCard card) {
@ -261,4 +189,12 @@ class RightHalfCard extends CardImpl<RightHalfCard> {
public RightHalfCard copy() {
return new RightHalfCard(this);
}
@Override
public List<String> getRules() {
List<String> rules = new ArrayList<String>();
rules.add(new StringBuilder("<b>").append(this.getName()).append("<b/>").toString());
rules.addAll(super.getRules());
return rules;
}
}

View file

@ -37,6 +37,7 @@ import mage.abilities.effects.ContinuousEffect;
import mage.abilities.effects.ContinuousEffects;
import mage.abilities.effects.Effect;
import mage.cards.Card;
import mage.cards.SplitCard;
import mage.choices.Choice;
import mage.game.combat.Combat;
import mage.game.combat.CombatGroup;
@ -454,6 +455,10 @@ public class GameState implements Serializable, Copyable<GameState> {
for (Ability ability: card.getAbilities()) {
addAbility(ability, card);
}
if (card.isSplitCard()) {
addCard( ((SplitCard)card).getLeftHalfCard());
addCard( ((SplitCard)card).getRightHalfCard());
}
}
@Deprecated

View file

@ -55,6 +55,10 @@ import mage.watchers.Watcher;
import java.util.ArrayList;
import java.util.List;
import java.util.UUID;
import mage.Constants;
import static mage.Constants.SpellAbilityType.SPLIT_LEFT;
import static mage.Constants.SpellAbilityType.SPLIT_RIGHT;
import mage.cards.SplitCard;
/**
*
@ -62,35 +66,87 @@ import java.util.UUID;
*/
public class Spell<T extends Spell<T>> implements StackObject, Card {
private List<Card> spellCards = new ArrayList<Card>();
private List<SpellAbility> spellAbilities = new ArrayList<SpellAbility>();
private Card card;
private SpellAbility ability;
private UUID controllerId;
private boolean copiedSpell;
private Zone fromZone;
private UUID id;
public Spell(Card card, SpellAbility ability, UUID controllerId, Zone fromZone) {
this.card = card;
id = ability.getId();
this.ability = ability;
this.ability.setControllerId(controllerId);
if (ability.getSpellAbilityType().equals(Constants.SpellAbilityType.SPLIT_FUSED)) {
spellCards.add(((SplitCard) card).getLeftHalfCard());
spellAbilities.add(((SplitCard) card).getLeftHalfCard().getSpellAbility().copy());
spellCards.add(((SplitCard) card).getRightHalfCard());
spellAbilities.add(((SplitCard) card).getRightHalfCard().getSpellAbility().copy());
} else {
spellCards.add(card);
spellAbilities.add(ability);
}
this.controllerId = controllerId;
this.fromZone = fromZone;
}
public Spell(final Spell<T> spell) {
this.card = spell.card.copy();
this.ability = spell.ability.copy();
this.id = spell.id;
for (SpellAbility spellAbility: spell.spellAbilities) {
this.spellAbilities.add(spellAbility.copy());
}
for (Card spellCard: spell.spellCards) {
this.spellCards.add(spellCard.copy());
}
if (spell.spellAbilities.get(0).equals(spell.ability)) {
this.ability = spellAbilities.get(0);
} else {
this.ability = spell.ability.copy();
}
if (spell.spellCards.get(0).equals(spell.card)) {
this.card = spellCards.get(0);
} else {
this.card = spell.card.copy();
}
this.controllerId = spell.controllerId;
this.fromZone = spell.fromZone;
this.copiedSpell = spell.copiedSpell;
}
public boolean activate(Game game, boolean noMana) {
for (SpellAbility spellAbility: spellAbilities) {
if (!spellAbility.activate(game, noMana)) {
return false;
}
}
return true;
}
public String getActivatedMessage(Game game) {
return ability.getActivatedMessage(game);
}
@Override
public boolean resolve(Game game) {
boolean result;
if (card.getCardType().contains(CardType.INSTANT) || card.getCardType().contains(CardType.SORCERY)) {
if (ability.getTargets().stillLegal(ability, game)) {
updateOptionalCosts();
result = ability.resolve(game);
int index = 0;
result = false;
boolean legalParts = false;
for(SpellAbility spellAbility: this.spellAbilities) {
if (spellAbility.getTargets().stillLegal(ability, game)) {
legalParts = true;
updateOptionalCosts(index);
result |= spellAbility.resolve(game);
}
index++;
}
if (legalParts) {
if (!copiedSpell) {
for (Effect effect : ability.getEffects()) {
if (effect instanceof PostResolveEffect) {
@ -102,7 +158,6 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
card.moveToZone(Zone.GRAVEYARD, ability.getId(), game, false);
}
}
return result;
}
//20091005 - 608.2b
@ -111,7 +166,7 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
return false;
} else if (card.getCardType().contains(CardType.ENCHANTMENT) && card.getSubtype().contains("Aura")) {
if (ability.getTargets().stillLegal(ability, game)) {
updateOptionalCosts();
updateOptionalCosts(0);
if (card.putOntoBattlefield(game, Zone.HAND, ability.getId(), controllerId)) {
return ability.resolve(game);
}
@ -122,7 +177,7 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
counter(null, game);
return false;
} else {
updateOptionalCosts();
updateOptionalCosts(0);
result = card.putOntoBattlefield(game, Zone.HAND, ability.getId(), controllerId);
return result;
}
@ -133,10 +188,10 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
* This information will be used later by effects, e.g. to determine whether card was kicked or not.
* E.g. Desolation Angel
*/
private void updateOptionalCosts() {
Ability abilityOrig = card.getAbilities().get(ability.getId());
private void updateOptionalCosts(int index) {
Ability abilityOrig = spellCards.get(index).getAbilities().get(spellAbilities.get(index).getId());
if (abilityOrig != null) {
for (Object object : ability.getOptionalCosts()) {
for (Object object : spellAbilities.get(index).getOptionalCosts()) {
Cost cost = (Cost) object;
for (Cost costOrig : abilityOrig.getOptionalCosts()) {
if (cost.getId().equals(costOrig.getId())) {
@ -163,29 +218,31 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
public boolean chooseNewTargets(Game game, UUID playerId) {
Player player = game.getPlayer(playerId);
if (player != null) {
for (Target target: ability.getTargets()) {
Target newTarget = target.copy();
newTarget.clearChosen();
for (UUID targetId: target.getTargets()) {
MageObject object = game.getObject(targetId);
String name = null;
if (object == null) {
Player targetPlayer = game.getPlayer(targetId);
if (targetPlayer != null) name = targetPlayer.getName();
} else {
name = object.getName();
for(SpellAbility spellAbility: spellAbilities) {
for (Target target: spellAbility.getTargets()) {
Target newTarget = target.copy();
newTarget.clearChosen();
for (UUID targetId: target.getTargets()) {
MageObject object = game.getObject(targetId);
String name = null;
if (object == null) {
Player targetPlayer = game.getPlayer(targetId);
if (targetPlayer != null) name = targetPlayer.getName();
} else {
name = object.getName();
}
if (name != null && player.chooseUse(spellAbility.getEffects().get(0).getOutcome(), "Change target from " + name + "?", game)) {
if (!player.chooseTarget(spellAbility.getEffects().get(0).getOutcome(), newTarget, spellAbility, game))
newTarget.addTarget(targetId, spellAbility, game, false);
}
else {
newTarget.addTarget(targetId, spellAbility, game, false);
}
}
if (name != null && player.chooseUse(ability.getEffects().get(0).getOutcome(), "Change target from " + name + "?", game)) {
if (!player.chooseTarget(ability.getEffects().get(0).getOutcome(), newTarget, ability, game))
newTarget.addTarget(targetId, ability, game, false);
target.clearChosen();
for (UUID newTargetId: newTarget.getTargets()) {
target.addTarget(newTargetId, spellAbility, game, false);
}
else {
newTarget.addTarget(targetId, ability, game, false);
}
}
target.clearChosen();
for (UUID newTargetId: newTarget.getTargets()) {
target.addTarget(newTargetId, ability, game, false);
}
}
return true;
@ -271,7 +328,7 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
@Override
public UUID getId() {
return ability.getId();
return id;
}
@Override
@ -293,6 +350,9 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
@Override
public void setControllerId(UUID controllerId) {
this.ability.setControllerId(controllerId);
for (SpellAbility spellAbility: spellAbilities) {
spellAbility.setControllerId(controllerId);
}
this.controllerId = controllerId;
}
@ -301,12 +361,26 @@ public class Spell<T extends Spell<T>> implements StackObject, Card {
@Override
public List<String> getRules() {
return card.getRules();
switch (ability.getSpellAbilityType()) {
case SPLIT_LEFT:
return ((SplitCard)card).getLeftHalfCard().getRules();
case SPLIT_RIGHT:
return ((SplitCard)card).getRightHalfCard().getRules();
default:
return card.getRules();
}
}
@Override
public List<Watcher> getWatchers() {
return card.getWatchers();
switch (ability.getSpellAbilityType()) {
case SPLIT_LEFT:
return ((SplitCard)card).getLeftHalfCard().getWatchers();
case SPLIT_RIGHT:
return ((SplitCard)card).getLeftHalfCard().getWatchers();
default:
return card.getWatchers();
}
}
@Override

View file

@ -73,6 +73,10 @@ import org.apache.log4j.Logger;
import java.io.Serializable;
import java.util.*;
import mage.Constants;
import mage.Constants.SpellAbilityType;
import mage.cards.SplitCard;
import mage.game.stack.Spell;
public abstract class PlayerImpl<T extends PlayerImpl<T>> implements Player, Serializable {
@ -598,13 +602,12 @@ public abstract class PlayerImpl<T extends PlayerImpl<T>> implements Player, Ser
int bookmark = game.bookmarkState();
Zone fromZone = game.getState().getZone(card.getId());
card.cast(game, fromZone, ability, playerId);
SpellAbility spellAbility = game.getStack().getSpell(ability.getId()).getSpellAbility();
if (spellAbility.activate(game, noMana)) {
GameEvent event = GameEvent.getEvent(GameEvent.EventType.SPELL_CAST, spellAbility.getId(), spellAbility.getSourceId(), playerId);
Spell spell = game.getStack().getSpell(ability.getId());
if (spell.activate(game, noMana)) {
GameEvent event = GameEvent.getEvent(GameEvent.EventType.SPELL_CAST, spell.getSpellAbility().getId(), spell.getSpellAbility().getSourceId(), playerId);
event.setZone(fromZone);
game.fireEvent(event);
game.fireInformEvent(name + spellAbility.getActivatedMessage(game));
game.fireInformEvent(new StringBuilder(name).append(spell.getActivatedMessage(game)).toString());
game.removeBookmark(bookmark);
resetStoredBookmark(game);
return true;
@ -775,7 +778,7 @@ public abstract class PlayerImpl<T extends PlayerImpl<T>> implements Player, Ser
protected LinkedHashMap<UUID, ActivatedAbility> getUseableActivatedAbilities(MageObject object, Zone zone, Game game) {
LinkedHashMap<UUID, ActivatedAbility> useable = new LinkedHashMap<UUID, ActivatedAbility>();
for (ActivatedAbility ability: object.getAbilities().getActivatedAbilities(zone)) {
if (ability.canActivate(playerId, game)) {
useable.put(ability.getId(), ability);
}
@ -785,7 +788,7 @@ public abstract class PlayerImpl<T extends PlayerImpl<T>> implements Player, Ser
for (ActivatedAbility ability: object.getAbilities().getActivatedAbilities(Zone.HAND)) {
useable.put(ability.getId(), ability);
}
}
}
//Alternative cost are not use for Flashback. This lines alloews to play spell with Alternative cost (like Force of Will) from wrong zone
/*else {
// this allows alternative costs like Flashback work from other than hand zones
@ -801,11 +804,39 @@ public abstract class PlayerImpl<T extends PlayerImpl<T>> implements Player, Ser
Abilities<ActivatedAbility> otherAbilities = game.getState().getOtherAbilities(object.getId(), zone);
if (otherAbilities != null) {
for (ActivatedAbility ability: otherAbilities) {
useable.put(ability.getId(), ability);
Card card = game.getCard(ability.getSourceId());
if (card.isSplitCard() && ability instanceof FlashbackAbility) {
FlashbackAbility flashbackAbility;
if (card.getCardType().contains(Constants.CardType.INSTANT)) {
flashbackAbility = new FlashbackAbility(((SplitCard) card).getLeftHalfCard().getManaCost(), Constants.TimingRule.INSTANT);
}
else {
flashbackAbility = new FlashbackAbility(((SplitCard) card).getLeftHalfCard().getManaCost(), Constants.TimingRule.SORCERY);
}
flashbackAbility.setSourceId(card.getId());
flashbackAbility.setControllerId(card.getOwnerId());
flashbackAbility.setSpellAbilityType(SpellAbilityType.SPLIT_LEFT);
flashbackAbility.setAbilityName(((SplitCard) card).getLeftHalfCard().getName());
useable.put(flashbackAbility.getId(), flashbackAbility);
if (card.getCardType().contains(Constants.CardType.INSTANT)) {
flashbackAbility = new FlashbackAbility(((SplitCard) card).getRightHalfCard().getManaCost(), Constants.TimingRule.INSTANT);
}
else {
flashbackAbility = new FlashbackAbility(((SplitCard) card).getRightHalfCard().getManaCost(), Constants.TimingRule.SORCERY);
}
flashbackAbility.setSourceId(card.getId());
flashbackAbility.setControllerId(card.getOwnerId());
flashbackAbility.setSpellAbilityType(SpellAbilityType.SPLIT_RIGHT);
flashbackAbility.setAbilityName(((SplitCard) card).getRightHalfCard().getName());
useable.put(flashbackAbility.getId(), flashbackAbility);
} else {
useable.put(ability.getId(), ability);
}
}
}
return useable;
}
}
protected LinkedHashMap<UUID, ManaAbility> getUseableManaAbilities(MageObject object, Zone zone, Game game) {
LinkedHashMap<UUID, ManaAbility> useable = new LinkedHashMap<UUID, ManaAbility>();