From 15796d999e7d5c9f354d5ea17a9fcb9dac4ab68f Mon Sep 17 00:00:00 2001 From: LevelX2 Date: Fri, 16 Aug 2013 10:15:48 +0200 Subject: [PATCH] [CHK] Added Kusari Gama and Shell of the last Kappa. --- .../sets/championsofkamigawa/KusariGama.java | 163 +++++++++++++++ .../ShellOfTheLastKappa.java | 192 ++++++++++++++++++ 2 files changed, 355 insertions(+) create mode 100644 Mage.Sets/src/mage/sets/championsofkamigawa/KusariGama.java create mode 100644 Mage.Sets/src/mage/sets/championsofkamigawa/ShellOfTheLastKappa.java diff --git a/Mage.Sets/src/mage/sets/championsofkamigawa/KusariGama.java b/Mage.Sets/src/mage/sets/championsofkamigawa/KusariGama.java new file mode 100644 index 0000000000..f4724f2c5d --- /dev/null +++ b/Mage.Sets/src/mage/sets/championsofkamigawa/KusariGama.java @@ -0,0 +1,163 @@ +/* + * 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.championsofkamigawa; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.TriggeredAbilityImpl; +import mage.abilities.common.SimpleActivatedAbility; +import mage.abilities.common.SimpleStaticAbility; +import mage.abilities.costs.mana.GenericManaCost; +import mage.abilities.effects.Effect; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.continious.BoostSourceEffect; +import mage.abilities.effects.common.continious.GainAbilityAttachedEffect; +import mage.abilities.keyword.EquipAbility; +import mage.cards.CardImpl; +import mage.constants.AttachmentType; +import mage.constants.CardType; +import mage.constants.Duration; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.filter.common.FilterBlockingCreature; +import mage.filter.common.FilterCreaturePermanent; +import mage.game.Game; +import mage.game.events.DamagedEvent; +import mage.game.events.GameEvent; +import mage.game.permanent.Permanent; + +/** + * + * @author LevelX2 + */ +public class KusariGama extends CardImpl { + + public KusariGama(UUID ownerId) { + super(ownerId, 260, "Kusari-Gama", Rarity.RARE, new CardType[]{CardType.ARTIFACT}, "{3}"); + this.expansionSetCode = "CHK"; + this.subtype.add("Equipment"); + + // Equipped creature has "{2}: This creature gets +1/+0 until end of turn." + Ability gainedAbility = new SimpleActivatedAbility(Zone.BATTLEFIELD, new BoostSourceEffect(1,0, Duration.EndOfTurn), new GenericManaCost(2)); + Effect effect = new GainAbilityAttachedEffect(gainedAbility, AttachmentType.EQUIPMENT); + effect.setText("Equipped creature has \"{2}: This creature gets +1/+0 until end of turn.\""); + Ability ability = new SimpleStaticAbility(Zone.BATTLEFIELD, effect); + this.addAbility(ability); + // Whenever equipped creature deals damage to a blocking creature, Kusari-Gama deals that much damage to each other creature defending player controls. + this.addAbility(new KusariGamaAbility()); + // Equip {3} + this.addAbility(new EquipAbility(Outcome.AddAbility, new GenericManaCost(3))); + } + + public KusariGama(final KusariGama card) { + super(card); + } + + @Override + public KusariGama copy() { + return new KusariGama(this); + } +} + +class KusariGamaAbility extends TriggeredAbilityImpl { + + private static final FilterCreaturePermanent filter = new FilterBlockingCreature(); + + public KusariGamaAbility() { + super(Zone.BATTLEFIELD, new KusariGamaDamageEffect()); + } + + public KusariGamaAbility(final KusariGamaAbility ability) { + super(ability); + } + + @Override + public KusariGamaAbility copy() { + return new KusariGamaAbility(this); + } + + @Override + public boolean checkTrigger(GameEvent event, Game game) { + if (event instanceof DamagedEvent) { + Permanent sourcePermanet = game.getPermanent(event.getSourceId()); + Permanent targetPermanet = game.getPermanent(event.getTargetId()); + if (sourcePermanet != null && targetPermanet != null && sourcePermanet.getAttachments().contains(this.getSourceId()) && filter.match(targetPermanet, game)) { + this.getEffects().get(0).setValue("damageAmount", event.getAmount()); + this.getEffects().get(0).setValue("damagedCreatureId", targetPermanet.getId()); + return true; + } + } + return false; + } + + @Override + public String getRule() { + return "Whenever equipped creature deals damage to a blocking creature, {this} deals that much damage to each other creature defending player controls."; + } +} + +class KusariGamaDamageEffect extends OneShotEffect { + + public KusariGamaDamageEffect() { + super(Outcome.Damage); + this.staticText = "{this} deals that much damage to each other creature defending player controls"; + } + + public KusariGamaDamageEffect(final KusariGamaDamageEffect effect) { + super(effect); + } + + @Override + public KusariGamaDamageEffect copy() { + return new KusariGamaDamageEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Integer damage = (Integer) this.getValue("damageAmount"); + if (damage != null && damage.intValue() > 0) { + UUID damagedCreatureId = (UUID) this.getValue("damagedCreatureId"); + Permanent creature = game.getPermanent(damagedCreatureId); + if (creature == null) { + creature = (Permanent) game.getLastKnownInformation(damagedCreatureId, Zone.BATTLEFIELD); + } + if (creature != null) { + for (UUID blockerId : game.getCombat().getBlockers()) { + if (!blockerId.equals(damagedCreatureId)) { + Permanent blockingCreature = game.getPermanent(blockerId); + if (blockingCreature != null && blockingCreature.getControllerId().equals(creature.getControllerId())) { + blockingCreature.damage(damage, source.getSourceId(), game, true, false); + } + } + } + } + } + return false; + } +} diff --git a/Mage.Sets/src/mage/sets/championsofkamigawa/ShellOfTheLastKappa.java b/Mage.Sets/src/mage/sets/championsofkamigawa/ShellOfTheLastKappa.java new file mode 100644 index 0000000000..31c1fb04f0 --- /dev/null +++ b/Mage.Sets/src/mage/sets/championsofkamigawa/ShellOfTheLastKappa.java @@ -0,0 +1,192 @@ +/* + * 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.championsofkamigawa; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.common.SimpleActivatedAbility; +import mage.abilities.costs.common.SacrificeSourceCost; +import mage.abilities.costs.common.TapSourceCost; +import mage.abilities.costs.mana.ManaCostsImpl; +import mage.abilities.effects.OneShotEffect; +import mage.cards.Card; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.filter.FilterCard; +import mage.filter.FilterSpell; +import mage.filter.predicate.ObjectPlayer; +import mage.filter.predicate.ObjectPlayerPredicate; +import mage.filter.predicate.Predicates; +import mage.filter.predicate.mageobject.CardTypePredicate; +import mage.game.Game; +import mage.game.permanent.Permanent; +import mage.game.stack.Spell; +import mage.game.stack.StackObject; +import mage.players.Player; +import mage.target.Target; +import mage.target.TargetSpell; +import mage.target.common.TargetCardInExile; +import mage.util.CardUtil; + +/** + * + * @author LevelX2 + */ +public class ShellOfTheLastKappa extends CardImpl { + + private static final FilterSpell filter = new FilterSpell("instant or sorcery spell that targets you"); + static { + filter.add(new TargetYouPredicate()); + filter.add(Predicates.or(new CardTypePredicate(CardType.INSTANT), new CardTypePredicate(CardType.SORCERY))); + } + + public ShellOfTheLastKappa(UUID ownerId) { + super(ownerId, 269, "Shell of the Last Kappa", Rarity.RARE, new CardType[]{CardType.ARTIFACT}, "{3}"); + this.expansionSetCode = "CHK"; + this.supertype.add("Legendary"); + + // {3}, {tap}: Exile target instant or sorcery spell that targets you. + Ability ability = new SimpleActivatedAbility(Zone.BATTLEFIELD, new ShellOfTheLastKappaEffect(), new ManaCostsImpl("{3}")); + ability.addCost(new TapSourceCost()); + Target target = new TargetSpell(filter); + target.setRequired(true); + ability.addTarget(target); + this.addAbility(ability); + // {3}, {tap}, Sacrifice Shell of the Last Kappa: You may cast a card exiled with Shell of the Last Kappa without paying its mana cost. + ability = new SimpleActivatedAbility(Zone.BATTLEFIELD, new ShellOfTheLastKappaCastEffect(), new ManaCostsImpl("{3}")); + ability.addCost(new TapSourceCost()); + ability.addCost(new SacrificeSourceCost()); + this.addAbility(ability); + + } + + public ShellOfTheLastKappa(final ShellOfTheLastKappa card) { + super(card); + } + + @Override + public ShellOfTheLastKappa copy() { + return new ShellOfTheLastKappa(this); + } +} + +class ShellOfTheLastKappaEffect extends OneShotEffect { + + public ShellOfTheLastKappaEffect() { + super(Outcome.Exile); + this.staticText = "Exile target instant or sorcery spell that targets you"; + } + + public ShellOfTheLastKappaEffect(final ShellOfTheLastKappaEffect effect) { + super(effect); + } + + @Override + public ShellOfTheLastKappaEffect copy() { + return new ShellOfTheLastKappaEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Spell spell = game.getStack().getSpell(targetPointer.getFirst(game, source)); + if (spell != null) { + Permanent sourcePermanent = game.getPermanent(source.getSourceId()); + if (sourcePermanent == null) { + sourcePermanent = (Permanent) game.getLastKnownInformation(source.getSourceId(), Zone.BATTLEFIELD); + } + if (sourcePermanent != null) { + game.getStack().counter(spell.getId(), source.getSourceId(), game); + Card card = spell.getCard(); + card.moveToExile(CardUtil.getCardExileZoneId(game, source), sourcePermanent.getName(), id, game); + } + } + return false; + } +} + +class ShellOfTheLastKappaCastEffect extends OneShotEffect { + + public ShellOfTheLastKappaCastEffect() { + super(Outcome.PlayForFree); + this.staticText = "You may cast a card exiled with {this} without paying its mana cost"; + } + + public ShellOfTheLastKappaCastEffect(final ShellOfTheLastKappaCastEffect effect) { + super(effect); + } + + @Override + public ShellOfTheLastKappaCastEffect copy() { + return new ShellOfTheLastKappaCastEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player controller = game.getPlayer(source.getControllerId()); + if (controller != null) { + TargetCardInExile target = new TargetCardInExile(new FilterCard(), CardUtil.getCardExileZoneId(game, source)); + if (controller.choose(Outcome.PlayForFree, game.getExile().getExileZone(CardUtil.getCardExileZoneId(game, source)), target, game)) { + Card card = game.getCard(target.getFirstTarget()); + if (card != null) { + game.getExile().removeCard(card, game); + return controller.cast(card.getSpellAbility(), game, true); + } + } + } + return false; + } +} + + +class TargetYouPredicate implements ObjectPlayerPredicate> { + + @Override + public boolean apply(ObjectPlayer input, Game game) { + UUID controllerId = input.getPlayerId(); + if (controllerId == null) { + return false; + } + + for (Target target : input.getObject().getStackAbility().getTargets()) { + for (UUID targetId : target.getTargets()) { + if (controllerId.equals(targetId)) { + return true; + } + } + } + return false; + } + + @Override + public String toString() { + return "spell that targets you"; + } +}