From dbbf7062c88f4b527612648bcd54e073f6fbda92 Mon Sep 17 00:00:00 2001 From: LevelX2 Date: Sun, 24 Feb 2013 18:04:04 +0100 Subject: [PATCH] [CHK] Mana Seism, Cut the Tethers, Shimatsu, the Bloodcloaked, Squelch. --- .../championsofkamigawa/CutTheTethers.java | 112 +++++++++++++ .../sets/championsofkamigawa/ManaSeism.java | 108 ++++++++++++ .../ShimatsuTheBloodcloaked.java | 136 +++++++++++++++ .../sets/championsofkamigawa/Squelch.java | 158 ++++++++++++++++++ 4 files changed, 514 insertions(+) create mode 100644 Mage.Sets/src/mage/sets/championsofkamigawa/CutTheTethers.java create mode 100644 Mage.Sets/src/mage/sets/championsofkamigawa/ManaSeism.java create mode 100644 Mage.Sets/src/mage/sets/championsofkamigawa/ShimatsuTheBloodcloaked.java create mode 100644 Mage.Sets/src/mage/sets/championsofkamigawa/Squelch.java diff --git a/Mage.Sets/src/mage/sets/championsofkamigawa/CutTheTethers.java b/Mage.Sets/src/mage/sets/championsofkamigawa/CutTheTethers.java new file mode 100644 index 00000000000..155f60c72ac --- /dev/null +++ b/Mage.Sets/src/mage/sets/championsofkamigawa/CutTheTethers.java @@ -0,0 +1,112 @@ +/* + * 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.Constants; +import mage.Constants.CardType; +import mage.Constants.Outcome; +import mage.Constants.Rarity; +import mage.abilities.Ability; +import mage.abilities.costs.Cost; +import mage.abilities.costs.mana.GenericManaCost; +import mage.abilities.effects.OneShotEffect; +import mage.cards.CardImpl; +import mage.filter.common.FilterCreaturePermanent; +import mage.filter.predicate.mageobject.SubtypePredicate; +import mage.game.Game; +import mage.game.permanent.Permanent; +import mage.players.Player; + +/** + * + * @author LevelX2 + */ +public class CutTheTethers extends CardImpl { + + public CutTheTethers(UUID ownerId) { + super(ownerId, 56, "Cut the Tethers", Rarity.UNCOMMON, new CardType[]{CardType.SORCERY}, "{2}{U}{U}"); + this.expansionSetCode = "CHK"; + + this.color.setBlue(true); + + // For each Spirit, return it to its owner's hand unless that player pays {3}. + this.getSpellAbility().addEffect(new CutTheTethersEffect()); + } + + public CutTheTethers(final CutTheTethers card) { + super(card); + } + + @Override + public CutTheTethers copy() { + return new CutTheTethers(this); + } +} + +class CutTheTethersEffect extends OneShotEffect { + + private static final FilterCreaturePermanent filter = new FilterCreaturePermanent("Spirit creatures"); + static { + filter.add(new SubtypePredicate("Spirit")); + } + + public CutTheTethersEffect() { + super(Outcome.ReturnToHand); + this.staticText = "For each Spirit, return it to its owner's hand unless that player pays {3}"; + } + + public CutTheTethersEffect(final CutTheTethersEffect effect) { + super(effect); + } + + @Override + public CutTheTethersEffect copy() { + return new CutTheTethersEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + for (Permanent creature: game.getState().getBattlefield().getActivePermanents(filter, source.getControllerId(), source.getSourceId(), game)) { + Player player = game.getPlayer(creature.getControllerId()); + if (player != null) { + boolean paid = false; + if (player.chooseUse(outcome, new StringBuilder("Pay {3} to keep ").append(creature.getName()).append(" on the battlefield?").toString(), game)) { + Cost cost = new GenericManaCost(3); + if (!cost.pay(source, game, source.getSourceId(), creature.getControllerId(), false)) { + paid = true; + } + if (!paid) { + creature.moveToZone(Constants.Zone.HAND, source.getSourceId(), game, true); + } + } + } + } + return true; + } +} diff --git a/Mage.Sets/src/mage/sets/championsofkamigawa/ManaSeism.java b/Mage.Sets/src/mage/sets/championsofkamigawa/ManaSeism.java new file mode 100644 index 00000000000..23d79651d2f --- /dev/null +++ b/Mage.Sets/src/mage/sets/championsofkamigawa/ManaSeism.java @@ -0,0 +1,108 @@ +/* + * 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.Constants.CardType; +import mage.Constants.Outcome; +import mage.Constants.Rarity; +import mage.Mana; +import mage.abilities.Ability; +import mage.abilities.effects.OneShotEffect; +import mage.cards.CardImpl; +import mage.filter.common.FilterControlledLandPermanent; +import mage.game.Game; +import mage.game.permanent.Permanent; +import mage.players.Player; +import mage.target.common.TargetControlledPermanent; + +/** + * + * @author LevelX2 + */ +public class ManaSeism extends CardImpl { + + public ManaSeism(UUID ownerId) { + super(ownerId, 179, "Mana Seism", Rarity.UNCOMMON, new CardType[]{CardType.SORCERY}, "{1}{R}"); + this.expansionSetCode = "CHK"; + + this.color.setRed(true); + + // Sacrifice any number of lands. Add {1} to your mana pool for each land sacrificed this way. + this.getSpellAbility().addEffect(new ManaSeismEffect()); + + } + + public ManaSeism(final ManaSeism card) { + super(card); + } + + @Override + public ManaSeism copy() { + return new ManaSeism(this); + } +} + +class ManaSeismEffect extends OneShotEffect { + + public ManaSeismEffect() { + super(Outcome.Neutral); + staticText = "Sacrifice any number of lands. Add {1} to your mana pool for each land sacrificed this way"; + } + + public ManaSeismEffect(final ManaSeismEffect effect) { + super(effect); + } + + @Override + public ManaSeismEffect copy() { + return new ManaSeismEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player player = game.getPlayer(source.getControllerId()); + if (player == null){ + return false; + } + int amount = 0; + TargetControlledPermanent sacrificeLand = new TargetControlledPermanent(0, Integer.MAX_VALUE, new FilterControlledLandPermanent(), true); + if(player.chooseTarget(Outcome.Sacrifice, sacrificeLand, source, game)){ + for(Object uuid : sacrificeLand.getTargets()){ + Permanent land = game.getPermanent((UUID)uuid); + if(land != null){ + land.sacrifice(source.getId(), game); + amount++; + } + } + } + player.getManaPool().addMana(Mana.ColorlessMana(amount), game, source); + return true; + } + +} diff --git a/Mage.Sets/src/mage/sets/championsofkamigawa/ShimatsuTheBloodcloaked.java b/Mage.Sets/src/mage/sets/championsofkamigawa/ShimatsuTheBloodcloaked.java new file mode 100644 index 00000000000..2b315db0730 --- /dev/null +++ b/Mage.Sets/src/mage/sets/championsofkamigawa/ShimatsuTheBloodcloaked.java @@ -0,0 +1,136 @@ +/* + * 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.Constants; +import mage.Constants.CardType; +import mage.Constants.Outcome; +import mage.Constants.Rarity; +import mage.MageInt; +import mage.abilities.Ability; +import mage.abilities.common.SimpleStaticAbility; +import mage.abilities.effects.ReplacementEffectImpl; +import mage.cards.CardImpl; +import mage.counters.CounterType; +import mage.filter.FilterPermanent; +import mage.game.Game; +import mage.game.events.GameEvent; +import mage.game.permanent.Permanent; +import mage.players.Player; +import mage.target.Target; +import mage.target.common.TargetControlledPermanent; + +/** + * + * @author LevelX2 + */ +public class ShimatsuTheBloodcloaked extends CardImpl { + + public ShimatsuTheBloodcloaked(UUID ownerId) { + super(ownerId, 186, "Shimatsu the Bloodcloaked", Rarity.RARE, new CardType[]{CardType.CREATURE}, "{3}{R}"); + this.expansionSetCode = "CHK"; + this.supertype.add("Legendary"); + this.subtype.add("Demon"); + this.subtype.add("Spirit"); + + this.color.setRed(true); + this.power = new MageInt(0); + this.toughness = new MageInt(0); + + // As Shimatsu the Bloodcloaked enters the battlefield, sacrifice any number of permanents. Shimatsu enters the battlefield with that many +1/+1 counters on it. + this.addAbility(new SimpleStaticAbility(Constants.Zone.BATTLEFIELD, new ShimatsuTheBloodcloakedEffect())); + } + + public ShimatsuTheBloodcloaked(final ShimatsuTheBloodcloaked card) { + super(card); + } + + @Override + public ShimatsuTheBloodcloaked copy() { + return new ShimatsuTheBloodcloaked(this); + } +} + +class ShimatsuTheBloodcloakedEffect extends ReplacementEffectImpl { + + public ShimatsuTheBloodcloakedEffect() { + super(Constants.Duration.WhileOnBattlefield, Outcome.BoostCreature); + this.staticText = "As Shimatsu the Bloodcloaked enters the battlefield, sacrifice any number of permanents. Shimatsu enters the battlefield with that many +1/+1 counters on it"; + } + + public ShimatsuTheBloodcloakedEffect(final ShimatsuTheBloodcloakedEffect effect) { + super(effect); + } + + @Override + public ShimatsuTheBloodcloakedEffect copy() { + return new ShimatsuTheBloodcloakedEffect(this); + } + + @Override + public boolean applies(GameEvent event, Ability source, Game game) { + if (event.getType() == GameEvent.EventType.ENTERS_THE_BATTLEFIELD + && event.getTargetId().equals(source.getSourceId())) { + return true; + } + return false; + + } + + @Override + public boolean replaceEvent(GameEvent event, Ability source, Game game) { + Permanent creature = game.getPermanent(event.getTargetId()); + Player controller = game.getPlayer(source.getControllerId()); + if (creature != null && controller != null) { + Target target = new TargetControlledPermanent(0, Integer.MAX_VALUE, new FilterPermanent(), true); + target.setRequired(false); + if (!target.canChoose(source.getSourceId(), source.getControllerId(), game)) { + return false; + } + controller.chooseTarget(Outcome.Detriment, target, source, game); + if (target.getTargets().size() > 0) { + int sacrificedCreatures = target.getTargets().size(); + game.informPlayers(new StringBuilder(controller.getName()).append(" sacrifices ").append(sacrificedCreatures).append(" creatures for ").append(creature.getName()).toString()); + for (UUID targetId: target.getTargets()) { + Permanent targetCreature = game.getPermanent(targetId); + if (targetCreature == null || !targetCreature.sacrifice(source.getSourceId(), game)) { + return false; + } + } + creature.addCounters(CounterType.P1P1.createInstance(sacrificedCreatures), game); + } + } + return false; + } + + @Override + public boolean apply(Game game, Ability source) { + return false; + } +} diff --git a/Mage.Sets/src/mage/sets/championsofkamigawa/Squelch.java b/Mage.Sets/src/mage/sets/championsofkamigawa/Squelch.java new file mode 100644 index 00000000000..ffc96fabbb4 --- /dev/null +++ b/Mage.Sets/src/mage/sets/championsofkamigawa/Squelch.java @@ -0,0 +1,158 @@ +/* + * 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.HashSet; +import java.util.Set; +import java.util.UUID; +import mage.Constants; +import mage.Constants.CardType; +import mage.Constants.Rarity; +import mage.abilities.Ability; +import mage.abilities.ActivatedAbility; +import mage.abilities.effects.common.CounterTargetEffect; +import mage.abilities.effects.common.DrawCardControllerEffect; +import mage.cards.CardImpl; +import mage.filter.Filter; +import mage.filter.FilterAbility; +import mage.game.Game; +import mage.game.stack.StackAbility; +import mage.game.stack.StackObject; +import mage.target.TargetObject; + +/** + * + * @author LevelX2 + */ +public class Squelch extends CardImpl { + + public Squelch(UUID ownerId) { + super(ownerId, 92, "Squelch", Rarity.UNCOMMON, new CardType[]{CardType.INSTANT}, "{1}{U}"); + this.expansionSetCode = "CHK"; + + this.color.setBlue(true); + + // Counter target activated ability. + this.getSpellAbility().addEffect(new CounterTargetEffect()); + this.getSpellAbility().addTarget(new ActivatedAbilityTarget()); + // Draw a card. + this.getSpellAbility().addEffect(new DrawCardControllerEffect(1)); + } + + public Squelch(final Squelch card) { + super(card); + } + + @Override + public Squelch copy() { + return new Squelch(this); + } +} + +class ActivatedAbilityTarget extends TargetObject { + + public ActivatedAbilityTarget() { + this.minNumberOfTargets = 1; + this.maxNumberOfTargets = 1; + this.zone = Constants.Zone.STACK; + this.targetName = "activated ability"; + } + + public ActivatedAbilityTarget(final ActivatedAbilityTarget target) { + super(target); + } + + + @Override + public boolean canTarget(UUID id, Ability source, Game game) { + if (source != null && source.getId().equals(id)) { + return false; + } + + StackObject stackObject = game.getStack().getStackObject(id); + if (stackObject.getStackAbility() != null && (stackObject.getStackAbility() instanceof ActivatedAbility)) { + return true; + } + return false; + } + + @Override + public boolean canChoose(UUID sourceId, UUID sourceControllerId, Game game) { + return canChoose(sourceControllerId, game); + } + + @Override + public boolean canChoose(UUID sourceControllerId, Game game) { + for (StackObject stackObject : game.getStack()) { + if (stackObject.getStackAbility() != null && (stackObject.getStackAbility() instanceof ActivatedAbility) && game.getPlayer(sourceControllerId).getInRange().contains(stackObject.getStackAbility().getControllerId())) { + return true; + } + } + return false; + } + + @Override + public Set possibleTargets(UUID sourceId, UUID sourceControllerId, Game game) { + return possibleTargets(sourceControllerId, game); + } + + @Override + public Set possibleTargets(UUID sourceControllerId, Game game) { + Set possibleTargets = new HashSet(); + for (StackObject stackObject : game.getStack()) { + if (stackObject.getStackAbility() != null && (stackObject.getStackAbility() instanceof ActivatedAbility) && game.getPlayer(sourceControllerId).getInRange().contains(stackObject.getStackAbility().getControllerId())) { + possibleTargets.add(stackObject.getStackAbility().getId()); + } + } + return possibleTargets; + } + + @Override + public ActivatedAbilityTarget copy() { + return new ActivatedAbilityTarget(this); + } + + @Override + public Filter getFilter() { + return new FilterAbility(); + } + + @Override + public String getTargetedName(Game game) { + StringBuilder sb = new StringBuilder("activated ability ("); + for (UUID targetId: getTargets()) { + StackAbility object = (StackAbility) game.getObject(targetId); + if (object != null) { + sb.append(object.getRule()).append(" "); + } + } + sb.append(")"); + return sb.toString(); + } + +}