From 55a02a6ee32b35fc64542c4fb333df40ed6be8ba Mon Sep 17 00:00:00 2001 From: emerald000 Date: Tue, 10 Feb 2015 06:36:10 -0500 Subject: [PATCH 1/3] Added Dimir Machinations and Vigor Mortis. --- .../mage/sets/izzetvsgolgari/VigorMortis.java | 52 +++++++ .../mage/sets/ravnika/DimirMachinations.java | 133 ++++++++++++++++++ .../src/mage/sets/ravnika/VigorMortis.java | 98 +++++++++++++ 3 files changed, 283 insertions(+) create mode 100644 Mage.Sets/src/mage/sets/izzetvsgolgari/VigorMortis.java create mode 100644 Mage.Sets/src/mage/sets/ravnika/DimirMachinations.java create mode 100644 Mage.Sets/src/mage/sets/ravnika/VigorMortis.java diff --git a/Mage.Sets/src/mage/sets/izzetvsgolgari/VigorMortis.java b/Mage.Sets/src/mage/sets/izzetvsgolgari/VigorMortis.java new file mode 100644 index 00000000000..2f51d970395 --- /dev/null +++ b/Mage.Sets/src/mage/sets/izzetvsgolgari/VigorMortis.java @@ -0,0 +1,52 @@ +/* + * 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.izzetvsgolgari; + +import java.util.UUID; + +/** + * + * @author emerald000 + */ +public class VigorMortis extends mage.sets.ravnika.VigorMortis { + + public VigorMortis(UUID ownerId) { + super(ownerId); + this.cardNumber = 74; + this.expansionSetCode = "DDJ"; + } + + public VigorMortis(final VigorMortis card) { + super(card); + } + + @Override + public VigorMortis copy() { + return new VigorMortis(this); + } +} diff --git a/Mage.Sets/src/mage/sets/ravnika/DimirMachinations.java b/Mage.Sets/src/mage/sets/ravnika/DimirMachinations.java new file mode 100644 index 00000000000..1a5841a4a95 --- /dev/null +++ b/Mage.Sets/src/mage/sets/ravnika/DimirMachinations.java @@ -0,0 +1,133 @@ +/* + * 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.ravnika; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.keyword.TransmuteAbility; +import mage.cards.Card; +import mage.cards.CardImpl; +import mage.cards.CardsImpl; +import mage.constants.CardType; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.filter.FilterCard; +import mage.game.Game; +import mage.players.Player; +import mage.target.TargetCard; +import mage.target.TargetPlayer; + +/** + * + * @author emerald000 + */ +public class DimirMachinations extends CardImpl { + + public DimirMachinations(UUID ownerId) { + super(ownerId, 84, "Dimir Machinations", Rarity.UNCOMMON, new CardType[]{CardType.SORCERY}, "{2}{B}"); + this.expansionSetCode = "RAV"; + + // Look at the top three cards of target player's library. Exile any number of those cards, then put the rest back in any order. + this.getSpellAbility().addEffect(new DimirMachinationsEffect()); + this.getSpellAbility().addTarget(new TargetPlayer()); + + // Transmute {1}{B}{B} + this.addAbility(new TransmuteAbility("{1}{B}{B}")); + + } + + public DimirMachinations(final DimirMachinations card) { + super(card); + } + + @Override + public DimirMachinations copy() { + return new DimirMachinations(this); + } +} + +class DimirMachinationsEffect extends OneShotEffect { + + DimirMachinationsEffect() { + super(Outcome.Neutral); + this.staticText = "Look at the top three cards of target player's library. Exile any number of those cards, then put the rest back in any order"; + } + + DimirMachinationsEffect(final DimirMachinationsEffect effect) { + super(effect); + } + + @Override + public DimirMachinationsEffect copy() { + return new DimirMachinationsEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player controller = game.getPlayer(source.getControllerId()); + Player targetPlayer = game.getPlayer(this.getTargetPointer().getFirst(game, source)); + if (controller != null && targetPlayer != null) { + int numCardsToLookAt = Math.min(3, targetPlayer.getLibrary().size()); + if (numCardsToLookAt > 0) { + CardsImpl cards = new CardsImpl(); + for (int i = 0; i < numCardsToLookAt; i++) { + cards.add(targetPlayer.getLibrary().removeFromTop(game)); + } + TargetCard targetExile = new TargetCard(0, numCardsToLookAt, Zone.LIBRARY, new FilterCard("card to exile")); + if (controller.choose(Outcome.Exile, cards, targetExile, game)) { + for (UUID cardId : targetExile.getTargets()) { + Card card = cards.get(cardId, game); + if (card != null) { + controller.moveCardToExileWithInfo(card, null, "", source.getSourceId(), game, Zone.LIBRARY); + cards.remove(card); + } + } + while (cards.size() > 0) { + if (cards.size() == 1) { + Card card = cards.get(cards.iterator().next(), game); + controller.moveCardToLibraryWithInfo(card, source.getSourceId(), game, Zone.LIBRARY, true, false); + cards.remove(card); + } + else { + TargetCard targetTop = new TargetCard(Zone.LIBRARY, new FilterCard("card to put on top of library")); + if (controller.choose(Outcome.Neutral, cards, targetTop, game)) { + Card card = cards.get(targetTop.getFirstTarget(), game); + controller.moveCardToLibraryWithInfo(card, source.getSourceId(), game, Zone.LIBRARY, true, false); + cards.remove(card); + } + } + } + } + } + return true; + } + return false; + } +} diff --git a/Mage.Sets/src/mage/sets/ravnika/VigorMortis.java b/Mage.Sets/src/mage/sets/ravnika/VigorMortis.java new file mode 100644 index 00000000000..4982189bb95 --- /dev/null +++ b/Mage.Sets/src/mage/sets/ravnika/VigorMortis.java @@ -0,0 +1,98 @@ +/* + * 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.ravnika; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.condition.common.ManaWasSpentCondition; +import mage.abilities.decorator.ConditionalOneShotEffect; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.ReturnFromGraveyardToBattlefieldTargetEffect; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.ColoredManaSymbol; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.counters.CounterType; +import mage.filter.common.FilterCreatureCard; +import mage.game.Game; +import mage.game.permanent.Permanent; +import mage.target.common.TargetCardInYourGraveyard; + +/** + * + * @author emerald000 + */ +public class VigorMortis extends CardImpl { + + public VigorMortis(UUID ownerId) { + super(ownerId, 111, "Vigor Mortis", Rarity.UNCOMMON, new CardType[]{CardType.SORCERY}, "{2}{B}{B}"); + this.expansionSetCode = "RAV"; + + // Return target creature card from your graveyard to the battlefield. If {G} was spent to cast Vigor Mortis, that creature enters the battlefield with an additional +1/+1 counter on it. + this.getSpellAbility().addEffect(new ReturnFromGraveyardToBattlefieldTargetEffect()); + this.getSpellAbility().addTarget(new TargetCardInYourGraveyard(new FilterCreatureCard())); + this.getSpellAbility().addEffect(new ConditionalOneShotEffect(new VigorMortisAddCounterEffect(), new ManaWasSpentCondition(ColoredManaSymbol.G))); + } + + public VigorMortis(final VigorMortis card) { + super(card); + } + + @Override + public VigorMortis copy() { + return new VigorMortis(this); + } +} + +class VigorMortisAddCounterEffect extends OneShotEffect { + + VigorMortisAddCounterEffect() { + super(Outcome.BoostCreature); + this.staticText = "that creature enters the battlefield with an additional +1/+1 counter on it"; + } + + VigorMortisAddCounterEffect(final VigorMortisAddCounterEffect effect) { + super(effect); + } + + @Override + public VigorMortisAddCounterEffect copy() { + return new VigorMortisAddCounterEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + // targetPointer can't be used because target moved from graveyard to battlefield + Permanent permanent = game.getPermanent(source.getFirstTarget()); + if (permanent != null) { + permanent.addCounters(CounterType.P1P1.createInstance(), game); + } + return false; + } +} From 9a20bdeff3216d5321593fbee692857568070328 Mon Sep 17 00:00:00 2001 From: emerald000 Date: Tue, 10 Feb 2015 07:11:39 -0500 Subject: [PATCH 2/3] Added Contamination. Hopefully not breaking anything. --- .../mage/sets/urzassaga/Contamination.java | 118 ++++++++++++++++++ Mage/src/mage/abilities/AbilityImpl.java | 48 ++++--- .../abilities/effects/common/ManaEffect.java | 5 +- 3 files changed, 155 insertions(+), 16 deletions(-) create mode 100644 Mage.Sets/src/mage/sets/urzassaga/Contamination.java diff --git a/Mage.Sets/src/mage/sets/urzassaga/Contamination.java b/Mage.Sets/src/mage/sets/urzassaga/Contamination.java new file mode 100644 index 00000000000..b65b9ff781e --- /dev/null +++ b/Mage.Sets/src/mage/sets/urzassaga/Contamination.java @@ -0,0 +1,118 @@ +/* + * 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.urzassaga; + +import java.util.UUID; +import mage.Mana; +import mage.abilities.Ability; +import mage.abilities.common.BeginningOfUpkeepTriggeredAbility; +import mage.abilities.common.SimpleStaticAbility; +import mage.abilities.costs.common.SacrificeTargetCost; +import mage.abilities.effects.ReplacementEffectImpl; +import mage.abilities.effects.common.SacrificeSourceUnlessPaysEffect; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Duration; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.TargetController; +import mage.constants.Zone; +import mage.filter.common.FilterControlledCreaturePermanent; +import mage.game.Game; +import mage.game.events.GameEvent; +import mage.game.events.GameEvent.EventType; +import mage.game.events.ManaEvent; +import mage.target.common.TargetControlledCreaturePermanent; + +/** + * + * @author emerald000 + */ +public class Contamination extends CardImpl { + + public Contamination(UUID ownerId) { + super(ownerId, 123, "Contamination", Rarity.RARE, new CardType[]{CardType.ENCHANTMENT}, "{2}{B}"); + this.expansionSetCode = "USG"; + + // At the beginning of your upkeep, sacrifice Contamination unless you sacrifice a creature. + this.addAbility(new BeginningOfUpkeepTriggeredAbility(new SacrificeSourceUnlessPaysEffect(new SacrificeTargetCost(new TargetControlledCreaturePermanent(new FilterControlledCreaturePermanent("a creature")))), TargetController.YOU, false)); + + // If a land is tapped for mana, it produces {B} instead of any other type and amount. + this.addAbility(new SimpleStaticAbility(Zone.BATTLEFIELD, new ContaminationReplacementEffect())); + } + + public Contamination(final Contamination card) { + super(card); + } + + @Override + public Contamination copy() { + return new Contamination(this); + } +} + +class ContaminationReplacementEffect extends ReplacementEffectImpl { + + ContaminationReplacementEffect() { + super(Duration.WhileOnBattlefield, Outcome.Neutral); + staticText = "If a land is tapped for mana, it produces {B} instead of any other type and amount"; + } + + ContaminationReplacementEffect(final ContaminationReplacementEffect effect) { + super(effect); + } + + @Override + public ContaminationReplacementEffect copy() { + return new ContaminationReplacementEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + return true; + } + + @Override + public boolean replaceEvent(GameEvent event, Ability source, Game game) { + ManaEvent manaEvent = (ManaEvent) event; + Mana mana = manaEvent.getMana(); + mana.setToMana(Mana.BlackMana); + return false; + } + + @Override + public boolean checksEventType(GameEvent event, Game game) { + return event.getType() == EventType.TAPPED_FOR_MANA; + } + + + @Override + public boolean applies(GameEvent event, Ability source, Game game) { + return true; + } +} \ No newline at end of file diff --git a/Mage/src/mage/abilities/AbilityImpl.java b/Mage/src/mage/abilities/AbilityImpl.java index 7be497ba262..618fa05bcad 100644 --- a/Mage/src/mage/abilities/AbilityImpl.java +++ b/Mage/src/mage/abilities/AbilityImpl.java @@ -28,21 +28,47 @@ package mage.abilities; +import java.util.ArrayList; +import java.util.List; +import java.util.UUID; import mage.MageObject; -import mage.abilities.costs.*; +import mage.Mana; +import mage.abilities.costs.AdjustingSourceCosts; +import mage.abilities.costs.AlternativeCost; +import mage.abilities.costs.AlternativeSourceCosts; +import mage.abilities.costs.Cost; +import mage.abilities.costs.Costs; +import mage.abilities.costs.CostsImpl; +import mage.abilities.costs.OptionalAdditionalSourceCosts; +import mage.abilities.costs.VariableCost; +import mage.abilities.costs.common.TapSourceCost; import mage.abilities.costs.mana.ManaCost; import mage.abilities.costs.mana.ManaCosts; import mage.abilities.costs.mana.ManaCostsImpl; import mage.abilities.costs.mana.VariableManaCost; -import mage.abilities.effects.*; +import mage.abilities.effects.ContinuousEffect; +import mage.abilities.effects.Effect; +import mage.abilities.effects.Effects; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.PostResolveEffect; +import mage.abilities.effects.common.BasicManaEffect; +import mage.abilities.effects.common.DynamicManaEffect; import mage.abilities.keyword.FlashbackAbility; import mage.abilities.mana.ManaAbility; import mage.cards.Card; import mage.choices.Choice; import mage.choices.Choices; -import mage.constants.*; +import mage.constants.AbilityType; +import mage.constants.AbilityWord; +import mage.constants.EffectType; +import mage.constants.Outcome; +import mage.constants.SpellAbilityType; +import mage.constants.Zone; import mage.game.Game; import mage.game.command.Emblem; +import mage.game.events.GameEvent; +import mage.game.events.ManaEvent; +import mage.game.permanent.Permanent; import mage.game.permanent.PermanentCard; import mage.game.stack.Spell; import mage.game.stack.StackAbility; @@ -51,17 +77,6 @@ import mage.target.Target; import mage.target.Targets; import org.apache.log4j.Logger; -import java.util.ArrayList; -import java.util.List; -import java.util.UUID; -import mage.Mana; -import mage.abilities.costs.common.TapSourceCost; -import mage.abilities.effects.common.BasicManaEffect; -import mage.abilities.effects.common.DynamicManaEffect; -import mage.game.events.GameEvent; -import mage.game.events.ManaEvent; -import mage.game.permanent.Permanent; - /** * * @author BetaSteward_at_googlemail.com @@ -360,7 +375,10 @@ public abstract class AbilityImpl implements Ability { mana = ((DynamicManaEffect)effect).getMana(game, this); } if (mana != null) { // if mana == null the event has to be fired in the mana effect - game.fireEvent(new ManaEvent(GameEvent.EventType.TAPPED_FOR_MANA, sourceId, sourceId, controllerId, mana)); + ManaEvent event = new ManaEvent(GameEvent.EventType.TAPPED_FOR_MANA, sourceId, sourceId, controllerId, mana); + if (!game.replaceEvent(event)) { + game.fireEvent(event); + } } break; } diff --git a/Mage/src/mage/abilities/effects/common/ManaEffect.java b/Mage/src/mage/abilities/effects/common/ManaEffect.java index 26f70d03602..c19e070a519 100644 --- a/Mage/src/mage/abilities/effects/common/ManaEffect.java +++ b/Mage/src/mage/abilities/effects/common/ManaEffect.java @@ -66,7 +66,10 @@ public abstract class ManaEffect extends OneShotEffect { if (source.getAbilityType().equals(AbilityType.MANA)) { for (Cost cost: source.getCosts()) { if (cost instanceof TapSourceCost) { - game.fireEvent(new ManaEvent(GameEvent.EventType.TAPPED_FOR_MANA, source.getSourceId(), source.getSourceId(), source.getControllerId(), mana)); + ManaEvent event = new ManaEvent(GameEvent.EventType.TAPPED_FOR_MANA, source.getSourceId(), source.getSourceId(), source.getControllerId(), mana); + if (!game.replaceEvent(event)) { + game.fireEvent(event); + } } } } From f4ce9d85cc83f1e6a1388207b28d4745c21000ce Mon Sep 17 00:00:00 2001 From: emerald000 Date: Tue, 10 Feb 2015 08:12:42 -0500 Subject: [PATCH 3/3] Added Nether Traitor, Scapegoat and Thran Foundry. --- .../src/mage/sets/stronghold/Scapegoat.java | 68 +++++++++++ .../mage/sets/timespiral/NetherTraitor.java | 114 ++++++++++++++++++ .../mage/sets/urzasdestiny/ThranFoundry.java | 101 ++++++++++++++++ 3 files changed, 283 insertions(+) create mode 100644 Mage.Sets/src/mage/sets/stronghold/Scapegoat.java create mode 100644 Mage.Sets/src/mage/sets/timespiral/NetherTraitor.java create mode 100644 Mage.Sets/src/mage/sets/urzasdestiny/ThranFoundry.java diff --git a/Mage.Sets/src/mage/sets/stronghold/Scapegoat.java b/Mage.Sets/src/mage/sets/stronghold/Scapegoat.java new file mode 100644 index 00000000000..62342d91e32 --- /dev/null +++ b/Mage.Sets/src/mage/sets/stronghold/Scapegoat.java @@ -0,0 +1,68 @@ +/* + * 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.stronghold; + +import java.util.UUID; +import mage.abilities.costs.common.SacrificeTargetCost; +import mage.abilities.effects.Effect; +import mage.abilities.effects.common.ReturnToHandTargetEffect; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Rarity; +import mage.filter.common.FilterControlledCreaturePermanent; +import mage.target.common.TargetControlledCreaturePermanent; + +/** + * + * @author emerald000 + */ +public class Scapegoat extends CardImpl { + + public Scapegoat(UUID ownerId) { + super(ownerId, 114, "Scapegoat", Rarity.UNCOMMON, new CardType[]{CardType.INSTANT}, "{W}"); + this.expansionSetCode = "STH"; + + // As an additional cost to cast Scapegoat, sacrifice a creature. + this.getSpellAbility().addCost(new SacrificeTargetCost(new TargetControlledCreaturePermanent())); + + // Return any number of target creatures you control to their owner's hand. + Effect effect = new ReturnToHandTargetEffect(); + effect.setText("Return any number of target creatures you control to their owner's hand"); + this.getSpellAbility().addEffect(effect); + this.getSpellAbility().addTarget(new TargetControlledCreaturePermanent(0, Integer.MAX_VALUE, new FilterControlledCreaturePermanent(), false)); + } + + public Scapegoat(final Scapegoat card) { + super(card); + } + + @Override + public Scapegoat copy() { + return new Scapegoat(this); + } +} diff --git a/Mage.Sets/src/mage/sets/timespiral/NetherTraitor.java b/Mage.Sets/src/mage/sets/timespiral/NetherTraitor.java new file mode 100644 index 00000000000..40b541e0ce0 --- /dev/null +++ b/Mage.Sets/src/mage/sets/timespiral/NetherTraitor.java @@ -0,0 +1,114 @@ +/* + * 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.timespiral; + +import java.util.UUID; +import mage.MageInt; +import mage.abilities.TriggeredAbilityImpl; +import mage.abilities.costs.mana.ColoredManaCost; +import mage.abilities.effects.common.DoIfCostPaid; +import mage.abilities.effects.common.ReturnSourceFromGraveyardToBattlefieldEffect; +import mage.abilities.keyword.HasteAbility; +import mage.abilities.keyword.ShadowAbility; +import mage.cards.Card; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.ColoredManaSymbol; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.game.Game; +import mage.game.events.GameEvent; +import mage.game.events.ZoneChangeEvent; + +/** + * + * @author emerald000 + */ +public class NetherTraitor extends CardImpl { + + public NetherTraitor(UUID ownerId) { + super(ownerId, 120, "Nether Traitor", Rarity.RARE, new CardType[]{CardType.CREATURE}, "{B}{B}"); + this.expansionSetCode = "TSP"; + this.subtype.add("Spirit"); + this.power = new MageInt(1); + this.toughness = new MageInt(1); + + // Haste + this.addAbility(HasteAbility.getInstance()); + + // Shadow + this.addAbility(ShadowAbility.getInstance()); + + // Whenever another creature is put into your graveyard from the battlefield, you may pay {B}. If you do, return Nether Traitor from your graveyard to the battlefield. + this.addAbility(new NetherTraitorTriggeredAbility()); + } + + public NetherTraitor(final NetherTraitor card) { + super(card); + } + + @Override + public NetherTraitor copy() { + return new NetherTraitor(this); + } +} + +class NetherTraitorTriggeredAbility extends TriggeredAbilityImpl { + + NetherTraitorTriggeredAbility(){ + super(Zone.GRAVEYARD, new DoIfCostPaid(new ReturnSourceFromGraveyardToBattlefieldEffect(), new ColoredManaCost(ColoredManaSymbol.B))); + } + + NetherTraitorTriggeredAbility(final NetherTraitorTriggeredAbility ability) { + super(ability); + } + + @Override + public NetherTraitorTriggeredAbility copy(){ + return new NetherTraitorTriggeredAbility(this); + } + + @Override + public boolean checkTrigger(GameEvent event, Game game) { + if (event.getType() == GameEvent.EventType.ZONE_CHANGE) { + ZoneChangeEvent zEvent = (ZoneChangeEvent) event; + if (zEvent.getFromZone() == Zone.BATTLEFIELD && zEvent.getToZone() == Zone.GRAVEYARD) { + Card card = game.getCard(event.getTargetId()); + if (card != null && card.getOwnerId().equals(this.getControllerId()) && !card.getId().equals(this.getSourceId())) { + return true; + } + } + } + return false; + } + + @Override + public String getRule() { + return "Whenever another creature is put into your graveyard from the battlefield, you may pay {B}. If you do, return {this} from your graveyard to the battlefield."; + } +} diff --git a/Mage.Sets/src/mage/sets/urzasdestiny/ThranFoundry.java b/Mage.Sets/src/mage/sets/urzasdestiny/ThranFoundry.java new file mode 100644 index 00000000000..9c60e6f35c3 --- /dev/null +++ b/Mage.Sets/src/mage/sets/urzasdestiny/ThranFoundry.java @@ -0,0 +1,101 @@ +/* + * 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.urzasdestiny; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.common.SimpleActivatedAbility; +import mage.abilities.costs.common.ExileSourceCost; +import mage.abilities.costs.common.TapSourceCost; +import mage.abilities.costs.mana.GenericManaCost; +import mage.abilities.effects.OneShotEffect; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.game.Game; +import mage.players.Player; +import mage.target.TargetPlayer; + +/** + * + * @author emerald000 + */ +public class ThranFoundry extends CardImpl { + + public ThranFoundry(UUID ownerId) { + super(ownerId, 140, "Thran Foundry", Rarity.UNCOMMON, new CardType[]{CardType.ARTIFACT}, "{1}"); + this.expansionSetCode = "UDS"; + + // {1}, {tap}, Exile Thran Foundry: Target player shuffles his or her graveyard into his or her library. + Ability ability = new SimpleActivatedAbility(Zone.BATTLEFIELD, new ThranFoundryEffect(), new GenericManaCost(1)); + ability.addCost(new TapSourceCost()); + ability.addCost(new ExileSourceCost()); + ability.addTarget(new TargetPlayer()); + this.addAbility(ability); + } + + public ThranFoundry(final ThranFoundry card) { + super(card); + } + + @Override + public ThranFoundry copy() { + return new ThranFoundry(this); + } +} + +class ThranFoundryEffect extends OneShotEffect { + + ThranFoundryEffect() { + super(Outcome.Neutral); + this.staticText = "Target player shuffles his or her graveyard into his or her library"; + } + + ThranFoundryEffect(final ThranFoundryEffect effect) { + super(effect); + } + + @Override + public ThranFoundryEffect copy() { + return new ThranFoundryEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player player = game.getPlayer(this.getTargetPointer().getFirst(game, source)); + if (player != null) { + player.getLibrary().addAll(player.getGraveyard().getCards(game), game); + player.getGraveyard().clear(); + player.shuffleLibrary(game); + return true; + } + return false; + } +}