From 306a0874dd38bf6e02010e887fa84759a5c10bbb Mon Sep 17 00:00:00 2001 From: LevelX2 Date: Fri, 20 Feb 2015 02:16:58 +0100 Subject: [PATCH] * Reworked Delve and Convoke to be more rule conform. --- .../abilities/keyword/ConvokeAbility.java | 260 +++++++++++------- .../mage/abilities/keyword/DelveAbility.java | 132 +++++---- 2 files changed, 250 insertions(+), 142 deletions(-) diff --git a/Mage/src/mage/abilities/keyword/ConvokeAbility.java b/Mage/src/mage/abilities/keyword/ConvokeAbility.java index b56631c0cc..9db9db18ed 100644 --- a/Mage/src/mage/abilities/keyword/ConvokeAbility.java +++ b/Mage/src/mage/abilities/keyword/ConvokeAbility.java @@ -28,28 +28,33 @@ package mage.abilities.keyword; +import java.util.ArrayList; +import java.util.List; import java.util.UUID; +import mage.Mana; import mage.ObjectColor; import mage.abilities.Ability; -import mage.abilities.SpellAbility; +import mage.abilities.SpecialAction; import mage.abilities.common.SimpleStaticAbility; -import mage.abilities.costs.AdjustingSourceCosts; +import mage.abilities.costs.mana.AlternateManaPaymentAbility; import mage.abilities.costs.mana.ManaCost; -import mage.abilities.costs.mana.ManaCosts; -import mage.abilities.costs.mana.ManaCostsImpl; +import mage.abilities.effects.OneShotEffect; import mage.choices.Choice; import mage.choices.ChoiceImpl; +import mage.constants.AbilityType; +import mage.constants.ManaType; import mage.constants.Outcome; import mage.constants.Zone; import mage.filter.common.FilterControlledCreaturePermanent; import mage.filter.predicate.Predicates; +import mage.filter.predicate.mageobject.ColorPredicate; import mage.filter.predicate.permanent.TappedPredicate; import mage.game.Game; import mage.game.permanent.Permanent; +import mage.players.ManaPool; import mage.players.Player; import mage.target.Target; import mage.target.common.TargetControlledCreaturePermanent; -import mage.util.CardUtil; /* * 502.46. Convoke @@ -84,13 +89,7 @@ import mage.util.CardUtil; * * @author LevelX2 */ -public class ConvokeAbility extends SimpleStaticAbility implements AdjustingSourceCosts { - - private static final FilterControlledCreaturePermanent filter = new FilterControlledCreaturePermanent(); - - static { - filter.add(Predicates.not(new TappedPredicate())); - } +public class ConvokeAbility extends SimpleStaticAbility implements AlternateManaPaymentAbility { public ConvokeAbility() { super(Zone.STACK, null); @@ -107,96 +106,171 @@ public class ConvokeAbility extends SimpleStaticAbility implements AdjustingSour } @Override - public void adjustCosts(Ability ability, Game game) { - Player player = game.getPlayer(controllerId); - if (player == null || !(ability instanceof SpellAbility)) { - return; - } - Target target = new TargetControlledCreaturePermanent(1, Integer.MAX_VALUE, filter,true); - target.setTargetName("creatures to convoke"); - if (!target.canChoose(sourceId, controllerId, game)) { - return; - } - if (player.chooseUse(Outcome.Detriment, "Convoke creatures?", game)) { - player.chooseTarget(Outcome.Tap, target, ability, game); - if (target.getTargets().size() > 0) { - for (UUID creatureId: target.getTargets()) { - Permanent perm = game.getPermanent(creatureId); - if (perm == null || ability.getManaCostsToPay().convertedManaCost() == 0) { - continue; + public void addSpecialAction(Ability source, Game game, ManaCost unpaid) { + Player controller = game.getPlayer(source.getControllerId()); + if (controller != null && controller.getGraveyard().size() > 0) { + if (unpaid.getMana().getColorless() > 0 && source.getAbilityType().equals(AbilityType.SPELL)) { + SpecialAction specialAction = new ConvokeSpecialAction(unpaid); + specialAction.setControllerId(source.getControllerId()); + specialAction.setSourceId(source.getSourceId()); + // create filter for possible creatures to tap + FilterControlledCreaturePermanent filter = new FilterControlledCreaturePermanent(); + filter.add(Predicates.not(new TappedPredicate())); + if (unpaid.getMana().getColorless() == 0) { + List colorPredicates = new ArrayList<>(); + if (unpaid.getMana().getBlack() > 0) { + colorPredicates.add(new ColorPredicate(ObjectColor.BLACK)); } - if (!perm.isTapped() && perm.tap(game)) { - ManaCosts manaCostsToReduce = new ManaCostsImpl<>(); - int costBefore = ability.getManaCostsToPay().convertedManaCost(); - Choice chooseManaType = buildChoice(perm.getColor(), ability.getManaCostsToPay()); - if (chooseManaType.getChoices().size() > 0) { - if (chooseManaType.getChoices().size() > 1) { - chooseManaType.getChoices().add("Colorless"); - chooseManaType.setMessage("Choose mana color to reduce from " + perm.getName()); - while (!chooseManaType.isChosen()) { - player.choose(Outcome.Benefit, chooseManaType, game); - } - } else { - chooseManaType.setChoice(chooseManaType.getChoices().iterator().next()); - } - if (chooseManaType.getChoice().equals("Black")) { - manaCostsToReduce.load("{B}"); - } - if (chooseManaType.getChoice().equals("Blue")) { - manaCostsToReduce.load("{U}"); - } - if (chooseManaType.getChoice().equals("Green")) { - manaCostsToReduce.load("{G}"); - } - if (chooseManaType.getChoice().equals("White")) { - manaCostsToReduce.load("{W}"); - } - if (chooseManaType.getChoice().equals("Red")) { - manaCostsToReduce.load("{R}"); - } - if (chooseManaType.getChoice().equals("Colorless")) { - manaCostsToReduce.load("{1}"); - } - CardUtil.reduceCost((SpellAbility)ability, manaCostsToReduce); - } else { - manaCostsToReduce.load("{1}"); - CardUtil.reduceCost((SpellAbility)ability, manaCostsToReduce); - } - if (costBefore == ability.getManaCostsToPay().convertedManaCost()) { - // creature could not reduce mana costs so tap must be reverted - perm.untap(game); - } else { - game.informPlayers("Convoke: " + player.getName() + " taps " + perm.getLogName() + " to reduce mana costs by " + manaCostsToReduce.getText()); - } + if (unpaid.getMana().getBlue() > 0) { + colorPredicates.add(new ColorPredicate(ObjectColor.BLUE)); } + if (unpaid.getMana().getRed() > 0) { + colorPredicates.add(new ColorPredicate(ObjectColor.RED)); + } + if (unpaid.getMana().getGreen() > 0) { + colorPredicates.add(new ColorPredicate(ObjectColor.GREEN)); + } + if (unpaid.getMana().getWhite() > 0) { + colorPredicates.add(new ColorPredicate(ObjectColor.WHITE)); + } + filter.add(Predicates.or(colorPredicates)); + } + Target target = new TargetControlledCreaturePermanent(1, 1, filter, true); + target.setTargetName("creature to convoke"); + specialAction.addTarget(target); + if (specialAction.canActivate(source.getControllerId(), game)) { + game.getState().getSpecialActions().add(specialAction); } } } } - private Choice buildChoice(ObjectColor creatureColor, ManaCosts manaCostsSpell) { - Choice choice = new ChoiceImpl(); - String spellCosts = manaCostsSpell.getText(); - if (creatureColor.isBlack() && spellCosts.contains("B")) { - choice.getChoices().add("Black"); - } - if (creatureColor.isBlue() && spellCosts.contains("U")) { - choice.getChoices().add("Blue"); - } - if (creatureColor.isGreen() && spellCosts.contains("G")) { - choice.getChoices().add("Green"); - } - if (creatureColor.isRed() && spellCosts.contains("R")) { - choice.getChoices().add("Red"); - } - if (creatureColor.isWhite() && spellCosts.contains("W")) { - choice.getChoices().add("White"); - } - return choice; - } - @Override public String getRule() { return "Convoke (Your creatures can help cast this spell. Each creature you tap while casting this spell pays for {1} or one mana of that creature's color.)"; } } + +class ConvokeSpecialAction extends SpecialAction { + + public ConvokeSpecialAction(ManaCost unpaid) { + super(Zone.ALL, true); + setRuleVisible(false); + this.addEffect(new ConvokeEffect(unpaid)); + } + + public ConvokeSpecialAction(final ConvokeSpecialAction ability) { + super(ability); + } + + @Override + public ConvokeSpecialAction copy() { + return new ConvokeSpecialAction(this); + } +} + +class ConvokeEffect extends OneShotEffect { + + private final ManaCost unpaid; + + public ConvokeEffect(ManaCost unpaid) { + super(Outcome.Benefit); + this.unpaid = unpaid; + this.staticText = "Convoke (Your creatures can help cast this spell. Each creature you tap while casting this spell pays for {1} or one mana of that creature's color.)"; + } + + public ConvokeEffect(final ConvokeEffect effect) { + super(effect); + this.unpaid = effect.unpaid; + } + + @Override + public ConvokeEffect copy() { + return new ConvokeEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player controller = game.getPlayer(source.getControllerId()); + if (controller != null) { + for (UUID creatureId: this.getTargetPointer().getTargets(game, source)) { + Permanent perm = game.getPermanent(creatureId); + if (perm == null) { + continue; + } + String manaName; + if (!perm.isTapped() && perm.tap(game)) { + ManaPool manaPool = controller.getManaPool(); + Choice chooseManaType = buildChoice(perm.getColor(), unpaid.getMana()); + if (chooseManaType.getChoices().size() > 0) { + if (chooseManaType.getChoices().size() > 1) { + chooseManaType.getChoices().add("Colorless"); + chooseManaType.setMessage("Choose mana color to reduce from " + perm.getName()); + while (!chooseManaType.isChosen()) { + controller.choose(Outcome.Benefit, chooseManaType, game); + if (!controller.isInGame()) { + return false; + } + } + } else { + chooseManaType.setChoice(chooseManaType.getChoices().iterator().next()); + } + if (chooseManaType.getChoice().equals("Black")) { + manaPool.addMana(Mana.BlackMana, game, source); + manaPool.unlockManaType(ManaType.BLACK); + } + if (chooseManaType.getChoice().equals("Blue")) { + manaPool.addMana(Mana.BlueMana, game, source); + manaPool.unlockManaType(ManaType.BLUE); + } + if (chooseManaType.getChoice().equals("Green")) { + manaPool.addMana(Mana.GreenMana, game, source); + manaPool.unlockManaType(ManaType.GREEN); + } + if (chooseManaType.getChoice().equals("White")) { + manaPool.addMana(Mana.WhiteMana, game, source); + manaPool.unlockManaType(ManaType.WHITE); + } + if (chooseManaType.getChoice().equals("Red")) { + manaPool.addMana(Mana.RedMana, game, source); + manaPool.unlockManaType(ManaType.RED); + } + if (chooseManaType.getChoice().equals("Colorless")) { + manaPool.addMana(Mana.ColorlessMana, game, source); + manaPool.unlockManaType(ManaType.COLORLESS); + } + manaName = chooseManaType.getChoice().toLowerCase(); + } else { + manaPool.addMana(Mana.ColorlessMana, game, source); + manaPool.unlockManaType(ManaType.COLORLESS); + manaName = "colorless"; + } + game.informPlayers("Convoke: " + controller.getName() + " taps " + perm.getLogName() + " to pay one " + manaName + " mana"); + } + + } + return true; + } + return false; + } + + private Choice buildChoice(ObjectColor creatureColor, Mana mana) { + Choice choice = new ChoiceImpl(); + if (creatureColor.isBlack() && mana.getBlack() > 0) { + choice.getChoices().add("Black"); + } + if (creatureColor.isBlue() && mana.getBlue() > 0) { + choice.getChoices().add("Blue"); + } + if (creatureColor.isGreen() && mana.getGreen() > 0) { + choice.getChoices().add("Green"); + } + if (creatureColor.isRed() && mana.getRed() > 0) { + choice.getChoices().add("Red"); + } + if (creatureColor.isWhite() && mana.getWhite() > 0) { + choice.getChoices().add("White"); + } + return choice; + } +} diff --git a/Mage/src/mage/abilities/keyword/DelveAbility.java b/Mage/src/mage/abilities/keyword/DelveAbility.java index 5f2106dc0c..492de5d9c3 100644 --- a/Mage/src/mage/abilities/keyword/DelveAbility.java +++ b/Mage/src/mage/abilities/keyword/DelveAbility.java @@ -29,18 +29,23 @@ package mage.abilities.keyword; import java.util.ArrayList; import java.util.List; -import java.util.UUID; +import mage.Mana; import mage.abilities.Ability; -import mage.abilities.SpellAbility; +import mage.abilities.SpecialAction; import mage.abilities.common.SimpleStaticAbility; -import mage.abilities.costs.AdjustingSourceCosts; +import mage.abilities.costs.common.ExileFromGraveCost; +import mage.abilities.costs.mana.AlternateManaPaymentAbility; +import mage.abilities.costs.mana.ManaCost; +import mage.abilities.effects.OneShotEffect; import mage.cards.Card; +import mage.constants.AbilityType; +import mage.constants.ManaType; import mage.constants.Outcome; import mage.constants.Zone; import mage.filter.FilterCard; import mage.game.Game; +import mage.players.ManaPool; import mage.players.Player; -import mage.target.Target; import mage.target.common.TargetCardInYourGraveyard; import mage.util.CardUtil; @@ -71,22 +76,16 @@ import mage.util.CardUtil; * increase the mana costs. */ - public class DelveAbility extends SimpleStaticAbility implements AdjustingSourceCosts { + public class DelveAbility extends SimpleStaticAbility implements AlternateManaPaymentAbility { - private List delvedCards; public DelveAbility() { super(Zone.STACK, null); this.setRuleAtTheTop(true); - this.delvedCards = null; } public DelveAbility(final DelveAbility ability) { super(ability); - if (ability.delvedCards != null) { - this.delvedCards = new ArrayList<>(); - this.delvedCards.addAll(ability.delvedCards); - } } @Override @@ -95,46 +94,81 @@ import mage.util.CardUtil; } @Override - public void adjustCosts(Ability ability, Game game) { - Player player = game.getPlayer(controllerId); - if (player == null || !(ability instanceof SpellAbility)) { - return; - } - Target target = new TargetCardInYourGraveyard(1, Integer.MAX_VALUE, new FilterCard()); - target.setTargetName("cards to delve from your graveyard"); - target.setNotTarget(true); - if (!target.canChoose(sourceId, controllerId, game)) { - return; - } - if (!CardUtil.isCheckPlayableMode(ability) && - player.chooseUse(Outcome.Detriment, "Delve cards from your graveyard?", game)) { - player.chooseTarget(Outcome.Detriment, target, ability, game); - if (target.getTargets().size() > 0) { - delvedCards = new ArrayList<>(); - int adjCost = 0; - for (UUID cardId: target.getTargets()) { - Card card = game.getCard(cardId); - if (card == null) { - continue; - } - delvedCards.add(card); - player.moveCardToExileWithInfo(card, null, "", getSourceId(), game, Zone.GRAVEYARD); - ++adjCost; - } - game.informPlayers(new StringBuilder("Delve: ").append(player.getName()).append(" exiled ") - .append(adjCost).append(" card").append(adjCost != 1?"s":"").append(" from his or her graveyard").toString()); - CardUtil.adjustCost((SpellAbility)ability, adjCost); - } - } + public String getRule() { + return "Delve (Each card you exile from your graveyard while casting this spell pays for {1})"; } @Override - public String getRule() { - return "Delve (You may exile any number of cards from your graveyard as you cast this spell. It costs {1} less to cast for each card exiled this way.)"; + public void addSpecialAction(Ability source, Game game, ManaCost unpaid) { + Player controller = game.getPlayer(source.getControllerId()); + if (controller != null && controller.getGraveyard().size() > 0) { + if (unpaid.getMana().getColorless() > 0 && source.getAbilityType().equals(AbilityType.SPELL)) { + SpecialAction specialAction = new DelveSpecialAction(); + specialAction.setControllerId(source.getControllerId()); + specialAction.setSourceId(source.getSourceId()); + specialAction.addCost(new ExileFromGraveCost(new TargetCardInYourGraveyard( + 0, Math.min(controller.getGraveyard().size(), unpaid.getMana().getColorless()), new FilterCard()))); + if (specialAction.canActivate(source.getControllerId(), game)) { + game.getState().getSpecialActions().add(specialAction); + } + } + } + } +} + +class DelveSpecialAction extends SpecialAction { + + public DelveSpecialAction() { + super(Zone.ALL, true); + this.addEffect(new DelveEffect()); + } + + public DelveSpecialAction(final DelveSpecialAction ability) { + super(ability); + } + + @Override + public DelveSpecialAction copy() { + return new DelveSpecialAction(this); + } +} + +class DelveEffect extends OneShotEffect { + + public DelveEffect() { + super(Outcome.Benefit); + this.staticText = "Delve (Each card you exile from your graveyard while casting this spell pays for {1}.)"; + } + + public DelveEffect(final DelveEffect effect) { + super(effect); + } + + @Override + public DelveEffect copy() { + return new DelveEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player controller = game.getPlayer(source.getControllerId()); + if (controller != null) { + ExileFromGraveCost exileFromGraveCost = (ExileFromGraveCost) source.getCosts().get(0); + List exiledCards = exileFromGraveCost.getExiledCards(); + if (exiledCards.size() > 0) { + ManaPool manaPool = controller.getManaPool(); + manaPool.addMana(new Mana(0,0,0,0,0,exiledCards.size(),0), game, source); + manaPool.unlockManaType(ManaType.COLORLESS); + String keyString = CardUtil.getCardZoneString("delvedCards", source.getSourceId(), game); + List delvedCards = (List) game.getState().getValue(keyString); + if (delvedCards == null) { + game.getState().setValue(keyString, exiledCards); + } else { + delvedCards.addAll(exiledCards); + } + } + return true; + } + return false; } - - public List getDelvedCards() { - return delvedCards; - } - }