From feb35dda81941c13d1394781d3f152047f1543e9 Mon Sep 17 00:00:00 2001 From: Duncan Townsend Date: Sat, 24 Jan 2015 19:44:16 -0500 Subject: [PATCH 1/6] Renamed mage.util.SpellTargetAddress to mage.util.TargetAddress and made it handle non-Spell Cards. --- .../sets/guildpact/InkTreaderNephilim.java | 8 +-- .../sets/scarsofmirrodin/PrecursorGolem.java | 8 +-- ...lTargetAddress.java => TargetAddress.java} | 62 +++++++++++++------ 3 files changed, 51 insertions(+), 27 deletions(-) rename Mage/src/mage/util/{SpellTargetAddress.java => TargetAddress.java} (70%) diff --git a/Mage.Sets/src/mage/sets/guildpact/InkTreaderNephilim.java b/Mage.Sets/src/mage/sets/guildpact/InkTreaderNephilim.java index 960c5b6a8d..639d6afbf7 100644 --- a/Mage.Sets/src/mage/sets/guildpact/InkTreaderNephilim.java +++ b/Mage.Sets/src/mage/sets/guildpact/InkTreaderNephilim.java @@ -55,7 +55,7 @@ import mage.target.Target; import mage.filter.predicate.mageobject.FromSetPredicate; import mage.players.Player; import mage.target.TargetPermanent; -import mage.util.SpellTargetAddress; +import mage.util.TargetAddress; /** * @author duncancmt @@ -128,7 +128,7 @@ class InkTreaderNephilimTriggeredAbility extends TriggeredAbilityImpl { if (spell != null) { boolean allTargetsInkTreaderNephilim = true; boolean atLeastOneTargetsInkTreaderNephilim = false; - for (SpellTargetAddress addr : SpellTargetAddress.walk(spell)) { + for (TargetAddress addr : TargetAddress.walk(spell)) { Target targetInstance = addr.getTarget(spell); for (UUID target : targetInstance.getTargets()) { allTargetsInkTreaderNephilim &= target.equals(sourceId); @@ -183,12 +183,12 @@ class InkTreaderNephilimEffect extends OneShotEffect { continue; // copy only for other creatures } boolean legal = true; - for (SpellTargetAddress addr : SpellTargetAddress.walk(copy)) { + for (TargetAddress addr : TargetAddress.walk(copy)) { Target targetInstance = addr.getTarget(copy); legal &= targetInstance.canTarget(permanent.getId(), addr.getSpellAbility(copy), game); } if (legal) { - for (SpellTargetAddress addr : SpellTargetAddress.walk(copy)) { + for (TargetAddress addr : TargetAddress.walk(copy)) { Target targetInstance = addr.getTarget(copy); int numTargets = targetInstance.getNumberOfTargets(); targetInstance.clearChosen(); diff --git a/Mage.Sets/src/mage/sets/scarsofmirrodin/PrecursorGolem.java b/Mage.Sets/src/mage/sets/scarsofmirrodin/PrecursorGolem.java index 34067fe1ac..1760502e9c 100644 --- a/Mage.Sets/src/mage/sets/scarsofmirrodin/PrecursorGolem.java +++ b/Mage.Sets/src/mage/sets/scarsofmirrodin/PrecursorGolem.java @@ -57,7 +57,7 @@ import mage.target.Target; import java.util.UUID; import mage.filter.predicate.mageobject.FromSetPredicate; import mage.target.TargetPermanent; -import mage.util.SpellTargetAddress; +import mage.util.TargetAddress; /** * @author nantuko @@ -127,7 +127,7 @@ class PrecursorGolemCopyTriggeredAbility extends TriggeredAbilityImpl { if (spell != null && (spell.getCardType().contains(CardType.INSTANT) || spell.getCardType().contains(CardType.SORCERY))) { UUID targetGolem = null; - for (SpellTargetAddress addr : SpellTargetAddress.walk(spell)) { + for (TargetAddress addr : TargetAddress.walk(spell)) { Target targetInstance = addr.getTarget(spell); for (UUID target : targetInstance.getTargets()) { Permanent permanent = game.getPermanent(target); @@ -186,7 +186,7 @@ class PrecursorGolemCopySpellEffect extends OneShotEffect { continue; // copy only for other golems } boolean legal = true; - for (SpellTargetAddress addr : SpellTargetAddress.walk(spell)) { + for (TargetAddress addr : TargetAddress.walk(spell)) { Target target = addr.getTarget(spell); if (!target.canTarget(permanent.getId(), game)) { legal = false; @@ -196,7 +196,7 @@ class PrecursorGolemCopySpellEffect extends OneShotEffect { if (legal) { Spell copy = spell.copySpell(); copy.setCopiedSpell(true); - for (SpellTargetAddress addr : SpellTargetAddress.walk(copy)) { + for (TargetAddress addr : TargetAddress.walk(copy)) { Target target = addr.getTarget(copy); target.clearChosen(); target.add(permanent.getId(), game); diff --git a/Mage/src/mage/util/SpellTargetAddress.java b/Mage/src/mage/util/TargetAddress.java similarity index 70% rename from Mage/src/mage/util/SpellTargetAddress.java rename to Mage/src/mage/util/TargetAddress.java index 9220afab77..da86a808d6 100644 --- a/Mage/src/mage/util/SpellTargetAddress.java +++ b/Mage/src/mage/util/TargetAddress.java @@ -32,6 +32,7 @@ import java.util.UUID; import mage.abilities.Mode; import mage.abilities.Modes; import mage.abilities.SpellAbility; +import mage.cards.Card; import mage.game.stack.Spell; import mage.target.Target; @@ -39,30 +40,30 @@ import mage.target.Target; /** * @author duncancmt */ -public class SpellTargetAddress { +public class TargetAddress { protected int spellAbilityIndex; protected UUID mode; protected int targetIndex; - public SpellTargetAddress(int spellAbilityIndex, UUID mode, int targetIndex) { + public TargetAddress(int spellAbilityIndex, UUID mode, int targetIndex) { this.spellAbilityIndex = spellAbilityIndex; this.mode = mode; this.targetIndex = targetIndex; } - protected static class SpellTargetAddressIterable implements Iterable { - protected final Spell spell; + protected static class TargetAddressIterable implements Iterable { + protected final Card card; - public SpellTargetAddressIterable(Spell spell) { - this.spell = spell; + public TargetAddressIterable(Card card) { + this.card = card; } - public Iterator iterator() { - return new SpellTargetAddressIterator(spell); + public Iterator iterator() { + return new TargetAddressIterator(card); } } - protected static class SpellTargetAddressIterator implements Iterator { + protected static class TargetAddressIterator implements Iterator { protected Iterator spellAbilityIterator; protected Integer lastSpellAbilityIndex = null; protected Iterator modeIterator = null; @@ -71,19 +72,27 @@ public class SpellTargetAddress { protected Iterator targetIterator = null; protected Integer lastTargetIndex = null; - public SpellTargetAddressIterator(Spell spell) { + public TargetAddressIterator(Spell spell) { this.spellAbilityIterator = spell.getSpellAbilities().iterator(); calcNext(); } + public TargetAddressIterator(Card card) { + this.lastSpellAbilityIndex = 0; + this.spellAbilityIterator = null; + this.modes = card.getSpellAbility().getModes(); + this.modeIterator = this.modes.getSelectedModes().iterator(); + calcNext(); + } + public boolean hasNext() { return lastTargetIndex != null; } - public SpellTargetAddress next() { - SpellTargetAddress ret = new SpellTargetAddress(lastSpellAbilityIndex, - lastMode, - lastTargetIndex); + public TargetAddress next() { + TargetAddress ret = new TargetAddress(lastSpellAbilityIndex, + lastMode, + lastTargetIndex); calcNext(); return ret; @@ -96,7 +105,7 @@ public class SpellTargetAddress { protected void calcNext() { if (targetIterator == null) { if (modeIterator == null) { - if (spellAbilityIterator.hasNext()) { + if (spellAbilityIterator != null && spellAbilityIterator.hasNext()) { if (lastSpellAbilityIndex == null) { lastSpellAbilityIndex = 0; } else { @@ -137,19 +146,34 @@ public class SpellTargetAddress { } - public static Iterable walk(Spell spell) { - return new SpellTargetAddressIterable(spell); + public static Iterable walk(Card card) { + return new TargetAddressIterable(card); } public Target getTarget(Spell spell) { - return spell.getSpellAbilities().get(spellAbilityIndex).getModes().get(mode).getTargets().get(targetIndex); + return getMode(spell).getTargets().get(targetIndex); + } + + public Target getTarget(Card card) { + return getMode(card).getTargets().get(targetIndex); } public Mode getMode(Spell spell) { - return spell.getSpellAbilities().get(spellAbilityIndex).getModes().get(mode); + return getSpellAbility(spell).getModes().get(mode); + } + + public Mode getMode(Card card) { + return getSpellAbility(card).getModes().get(mode); } public SpellAbility getSpellAbility(Spell spell) { return spell.getSpellAbilities().get(spellAbilityIndex); } + + public SpellAbility getSpellAbility(Card card) { + if (spellAbilityIndex > 0) { + throw new IndexOutOfBoundsException("SpellAbility index " + spellAbilityIndex + " is out of bounds."); + } + return card.getSpellAbility(); + } } From 080790cf2556d8007c1aab6743917192945c0a6e Mon Sep 17 00:00:00 2001 From: Duncan Townsend Date: Sat, 24 Jan 2015 19:45:04 -0500 Subject: [PATCH 2/6] Comment in mage.target.Target was backwards. Fixed. --- Mage/src/mage/target/Target.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Mage/src/mage/target/Target.java b/Mage/src/mage/target/Target.java index 6ce070943a..bdd210b8dd 100644 --- a/Mage/src/mage/target/Target.java +++ b/Mage/src/mage/target/Target.java @@ -52,7 +52,7 @@ public interface Target extends Serializable { /** * controlls if it will be checked, if the target can be targeted from source - * @param notTarget true = check for protection, false = do not check for protection + * @param notTarget true = do not check for protection, false = check for protection */ void setNotTarget(boolean notTarget); From aa8554e82bb2054fe1d4ba07b218b18c58c17dea Mon Sep 17 00:00:00 2001 From: Duncan Townsend Date: Sat, 24 Jan 2015 19:45:47 -0500 Subject: [PATCH 3/6] Added mage.filter.predicate.permanent.CanBeEnchantedByPredicate --- .../src/mage/sets/ravnika/BreathOfFury.java | 25 +-------- .../permanent/CanBeEnchantedByPredicate.java | 55 +++++++++++++++++++ 2 files changed, 57 insertions(+), 23 deletions(-) create mode 100644 Mage/src/mage/filter/predicate/permanent/CanBeEnchantedByPredicate.java diff --git a/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java b/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java index 9a77b764fc..9c49603f81 100644 --- a/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java +++ b/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java @@ -49,9 +49,7 @@ import mage.target.Target; import mage.filter.common.FilterControlledCreaturePermanent; import mage.game.turn.TurnMod; import mage.constants.TurnPhase; -import mage.MageObject; -import mage.filter.predicate.Predicate; - +import mage.filter.predicate.permanent.CanBeEnchantedByPredicate; /** * @author duncancmt */ @@ -142,7 +140,7 @@ class BreathOfFuryEffect extends OneShotEffect { Permanent enchantedCreature = game.getPermanent(enchantment.getAttachedTo()); Player controller = game.getPlayer(source.getControllerId()); FilterControlledCreaturePermanent filter = new FilterControlledCreaturePermanent("creature you control that could be enchanted by " + enchantment.getName()); - filter.add(new CanBeEnchantedPredicate(enchantment)); + filter.add(new CanBeEnchantedByPredicate(enchantment)); Target target = new TargetControlledCreaturePermanent(filter); target.setNotTarget(true); // It's important to check that the creature was successfully sacrificed here. Effects that prevent sacrifice will also prevent Breath of Fury's effect from working. @@ -165,22 +163,3 @@ class BreathOfFuryEffect extends OneShotEffect { return false; } } - -class CanBeEnchantedPredicate implements Predicate { - - private final MageObject auraEnchantment; - - public CanBeEnchantedPredicate(MageObject auraEnchantment){ - this.auraEnchantment = auraEnchantment; - } - - @Override - public boolean apply(Permanent input, Game game) { - return !input.cantBeEnchantedBy(auraEnchantment, game); - } - - @Override - public String toString() { - return "CanBeEnchanted(" + auraEnchantment.toString() + ")"; - } -} diff --git a/Mage/src/mage/filter/predicate/permanent/CanBeEnchantedByPredicate.java b/Mage/src/mage/filter/predicate/permanent/CanBeEnchantedByPredicate.java new file mode 100644 index 0000000000..f11f4354bf --- /dev/null +++ b/Mage/src/mage/filter/predicate/permanent/CanBeEnchantedByPredicate.java @@ -0,0 +1,55 @@ +/* + * 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.filter.predicate.permanent; + +import mage.MageObject; +import mage.filter.predicate.Predicate; +import mage.game.Game; +import mage.game.permanent.Permanent; + +/** + * @author duncancmt + */ +public class CanBeEnchantedByPredicate implements Predicate { + + private final MageObject auraEnchantment; + + public CanBeEnchantedByPredicate(MageObject auraEnchantment){ + this.auraEnchantment = auraEnchantment; + } + + @Override + public boolean apply(Permanent input, Game game) { + return !input.cantBeEnchantedBy(auraEnchantment, game); + } + + @Override + public String toString() { + return "CanBeEnchanted(" + auraEnchantment.toString() + ")"; + } +} From ec917c7a50972a7db29c244bac9fa9d2d41da2d1 Mon Sep 17 00:00:00 2001 From: Duncan Townsend Date: Sat, 24 Jan 2015 19:48:19 -0500 Subject: [PATCH 4/6] In Breath of Fury, remember the creature enchanted by Breath of Fury when its ability triggers. --- Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java | 13 ++++++++++--- 1 file changed, 10 insertions(+), 3 deletions(-) diff --git a/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java b/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java index 9c49603f81..51cbbf19bb 100644 --- a/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java +++ b/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java @@ -38,6 +38,7 @@ import mage.constants.Outcome; import mage.abilities.Ability; import mage.abilities.keyword.EnchantAbility; import mage.abilities.TriggeredAbilityImpl; +import mage.abilities.effects.Effect; import mage.constants.Zone; import mage.game.events.GameEvent; import mage.game.Game; @@ -103,7 +104,13 @@ class BreathOfFuryAbility extends TriggeredAbilityImpl { if (damageEvent.isCombatDamage() && enchantment != null && enchantment.getAttachedTo().equals(event.getSourceId())) { - return true; + Permanent creature = game.getPermanent(enchantment.getAttachedTo()); + if (creature != null) { + for (Effect effect : getEffects()) { + effect.setValue("TriggeringCreatureId", creature.getId()); + } + return true; + } } } return false; @@ -137,7 +144,7 @@ class BreathOfFuryEffect extends OneShotEffect { if (enchantment == null) { return false; } - Permanent enchantedCreature = game.getPermanent(enchantment.getAttachedTo()); + Permanent enchantedCreature = game.getPermanent((UUID) getValue("TriggeringCreatureId")); Player controller = game.getPlayer(source.getControllerId()); FilterControlledCreaturePermanent filter = new FilterControlledCreaturePermanent("creature you control that could be enchanted by " + enchantment.getName()); filter.add(new CanBeEnchantedByPredicate(enchantment)); @@ -147,7 +154,7 @@ class BreathOfFuryEffect extends OneShotEffect { // Commanders going to the command zone and Rest in Peace style replacement effects don't make Permanent.sacrifice return false. if (enchantedCreature != null && controller != null && enchantedCreature.sacrifice(source.getSourceId(), game) - && target.canChoose(source.getSourceId(), source.getControllerId(), game)) { + && target.canChoose(source.getSourceId(), controller.getId(), game)) { controller.choose(outcome, target, source.getSourceId(), game); Permanent newCreature = game.getPermanent(target.getFirstTarget()); if (newCreature != null && From 2463418ecac51803d902574f70700ca1ccdb3014 Mon Sep 17 00:00:00 2001 From: Duncan Townsend Date: Sat, 24 Jan 2015 20:23:48 -0500 Subject: [PATCH 5/6] Breath of Fury had strange behavior when it was attached to a creature other than the one that caused its trigger when its trigger resolved. Fixed. --- .../src/mage/sets/ravnika/BreathOfFury.java | 21 +++++++++++++++++-- 1 file changed, 19 insertions(+), 2 deletions(-) diff --git a/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java b/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java index 51cbbf19bb..c8b526f2ac 100644 --- a/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java +++ b/Mage.Sets/src/mage/sets/ravnika/BreathOfFury.java @@ -157,8 +157,25 @@ class BreathOfFuryEffect extends OneShotEffect { && target.canChoose(source.getSourceId(), controller.getId(), game)) { controller.choose(outcome, target, source.getSourceId(), game); Permanent newCreature = game.getPermanent(target.getFirstTarget()); - if (newCreature != null && - newCreature.addAttachment(enchantment.getId(), game)) { + boolean success = false; + if (newCreature != null) { + Permanent oldCreature = game.getPermanent(enchantment.getAttachedTo()); + if (oldCreature != null) { + if (oldCreature.getId().equals(newCreature.getId())) { + success = true; + } else { + if (oldCreature.removeAttachment(enchantment.getId(), game) + && newCreature.addAttachment(enchantment.getId(), game)) { + game.informPlayers(enchantment.getLogName() + " was unattached from " + oldCreature.getLogName() + " and attached to " + newCreature.getLogName()); + success = true; + } + } + } else if (newCreature.addAttachment(enchantment.getId(), game)) { + game.informPlayers(enchantment.getLogName() + " was attached to " + newCreature.getLogName()); + success = true; + } + } + if (success) { for (Permanent permanent : game.getBattlefield().getAllActivePermanents(new FilterControlledCreaturePermanent(), controller.getId(), game)) { permanent.untap(game); } From 11a81e80477b4a96978758b428c4d43b8ba713db Mon Sep 17 00:00:00 2001 From: Duncan Townsend Date: Sat, 24 Jan 2015 20:24:55 -0500 Subject: [PATCH 6/6] Added implementation of Aura Graft --- Mage.Sets/src/mage/sets/tenth/AuraGraft.java | 164 +++++++++++++++++++ 1 file changed, 164 insertions(+) create mode 100644 Mage.Sets/src/mage/sets/tenth/AuraGraft.java diff --git a/Mage.Sets/src/mage/sets/tenth/AuraGraft.java b/Mage.Sets/src/mage/sets/tenth/AuraGraft.java new file mode 100644 index 0000000000..5dc14c83bd --- /dev/null +++ b/Mage.Sets/src/mage/sets/tenth/AuraGraft.java @@ -0,0 +1,164 @@ +/* + * 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.sets.tenth; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.effects.Effect; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.continious.GainControlTargetEffect; +import mage.constants.CardType; +import mage.constants.Rarity; +import mage.cards.CardImpl; +import mage.constants.Duration; +import mage.constants.Outcome; +import mage.filter.Filter; +import mage.filter.FilterPermanent; +import mage.filter.predicate.ObjectPlayer; +import mage.filter.predicate.ObjectPlayerPredicate; +import mage.filter.predicate.mageobject.SubtypePredicate; +import mage.filter.predicate.permanent.AnotherPredicate; +import mage.game.Game; +import mage.game.permanent.Permanent; +import mage.players.Player; +import mage.target.Target; +import mage.target.TargetPermanent; +import mage.util.TargetAddress; + +/** + * @author duncancmt + */ +public class AuraGraft extends CardImpl { + + public AuraGraft(UUID ownerId) { + super(ownerId, 67, "Aura Graft", Rarity.UNCOMMON, new CardType[]{CardType.INSTANT}, "{1}{U}"); + this.expansionSetCode = "10E"; + + FilterPermanent filter = new FilterPermanent("Aura that's attached to a permanent"); + filter.add(new SubtypePredicate("Aura")); + filter.add(new AttachedToPermanentPredicate()); + this.getSpellAbility().addTarget(new TargetPermanent(filter)); + + Effect gainControlEffect = new GainControlTargetEffect(Duration.EndOfGame); + //gainControlEffect.setText("Gain control of target Aura that's attached to a permanent"); + this.getSpellAbility().addEffect(gainControlEffect); + + this.getSpellAbility().addEffect(new MoveTargetAuraEffect()); + } + + public AuraGraft(final AuraGraft card) { + super(card); + } + + @Override + public AuraGraft copy() { + return new AuraGraft(this); + } +} + + +class AttachedToPermanentPredicate implements ObjectPlayerPredicate> { + + public AttachedToPermanentPredicate() { + super(); + } + + public boolean apply(ObjectPlayer input, Game game) { + Permanent attached = input.getObject(); + return attached != null && game.getPermanent(attached.getAttachedTo()) != null; + } +} + +class PermanentCanBeAttachedToPredicate implements ObjectPlayerPredicate> { + protected Permanent aura; + + public PermanentCanBeAttachedToPredicate(Permanent aura) { + super(); + this.aura = aura; + } + + public boolean apply(ObjectPlayer input, Game game) { + Permanent potentialAttachment = input.getObject(); + for (TargetAddress addr : TargetAddress.walk(aura)) { + Target target = addr.getTarget(aura); + Filter filter = target.getFilter(); + return filter.match(potentialAttachment, game); + } + return false; + } +} + +class MoveTargetAuraEffect extends OneShotEffect { + + public MoveTargetAuraEffect() { + super(Outcome.Benefit); + staticText = "Attach it to another permanent it can enchant"; + } + + public MoveTargetAuraEffect(final MoveTargetAuraEffect effect) { + super(effect); + } + + @Override + public MoveTargetAuraEffect copy() { + return new MoveTargetAuraEffect(this); + } + + @Override + public boolean apply(Game game, Ability source){ + Permanent enchantment = game.getPermanent(targetPointer.getFirst(game, source)); + if (enchantment == null) { + return false; + } + Permanent oldAttachment = game.getPermanent(enchantment.getAttachedTo()); + if (oldAttachment == null) { + return false; + } + Player controller = game.getPlayer(source.getControllerId()); + if (controller == null) { + return false; + } + + FilterPermanent filter = new FilterPermanent("another permanent " + enchantment.getLogName() + " can enchant"); + filter.add(new AnotherPredicate()); + filter.add(new PermanentCanBeAttachedToPredicate(enchantment)); + Target target = new TargetPermanent(filter); + target.setNotTarget(true); + if (target.canChoose(oldAttachment.getId(), controller.getId(), game) + && controller.choose(outcome, target, oldAttachment.getId(), game)) { + Permanent newAttachment = game.getPermanent(target.getFirstTarget()); + if (newAttachment != null && + oldAttachment.removeAttachment(enchantment.getId(), game)) { + newAttachment.addAttachment(enchantment.getId(), game); + game.informPlayers(enchantment.getLogName() + " was unattached from " + oldAttachment.getLogName() + " and attached to " + newAttachment.getLogName()); + return true; + } + } + return false; + } +}