From 9b0b777c3583d261aeb20f4b90433cbcd6fb712f Mon Sep 17 00:00:00 2001 From: LevelX2 Date: Sat, 20 Dec 2014 18:17:54 +0100 Subject: [PATCH] Added Primal Forcemage, Kaho Minamo Historian, Iname As One, Evermind. --- .../mage/sets/saviorsofkamigawa/Evermind.java | 74 ++++++++ .../sets/saviorsofkamigawa/InameAsOne.java | 132 +++++++++++++ .../KahoMinamoHistorian.java | 173 ++++++++++++++++++ .../mage/sets/timespiral/PrimalForcemage.java | 83 +++++++++ 4 files changed, 462 insertions(+) create mode 100644 Mage.Sets/src/mage/sets/saviorsofkamigawa/Evermind.java create mode 100644 Mage.Sets/src/mage/sets/saviorsofkamigawa/InameAsOne.java create mode 100644 Mage.Sets/src/mage/sets/saviorsofkamigawa/KahoMinamoHistorian.java create mode 100644 Mage.Sets/src/mage/sets/timespiral/PrimalForcemage.java diff --git a/Mage.Sets/src/mage/sets/saviorsofkamigawa/Evermind.java b/Mage.Sets/src/mage/sets/saviorsofkamigawa/Evermind.java new file mode 100644 index 00000000000..9422b4d68c5 --- /dev/null +++ b/Mage.Sets/src/mage/sets/saviorsofkamigawa/Evermind.java @@ -0,0 +1,74 @@ +/* + * 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.saviorsofkamigawa; + +import java.util.UUID; +import mage.abilities.Ability; +import mage.abilities.common.EmptyEffect; +import mage.abilities.common.SimpleStaticAbility; +import mage.abilities.effects.common.DrawCardSourceControllerEffect; +import mage.abilities.keyword.SpliceOntoArcaneAbility; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Rarity; +import mage.constants.Zone; + +/** + * + * @author LevelX2 + */ +public class Evermind extends CardImpl { + + public Evermind(UUID ownerId) { + super(ownerId, 37, "Evermind", Rarity.UNCOMMON, new CardType[]{CardType.INSTANT}, ""); + this.expansionSetCode = "SOK"; + this.subtype.add("Arcane"); + + this.color.setBlue(true); + + // (Nonexistent mana costs can't be paid.) + Ability ability = new SimpleStaticAbility(Zone.BATTLEFIELD, new EmptyEffect("(Nonexistent mana costs can't be paid.)")); + ability.setRuleAtTheTop(true); + this.addAbility(ability); + + // Draw a card. + this.getSpellAbility().addEffect(new DrawCardSourceControllerEffect(1)); + + // Splice onto Arcane {1}{U} + this.addAbility(new SpliceOntoArcaneAbility("{1}{U}")); + } + + public Evermind(final Evermind card) { + super(card); + } + + @Override + public Evermind copy() { + return new Evermind(this); + } +} diff --git a/Mage.Sets/src/mage/sets/saviorsofkamigawa/InameAsOne.java b/Mage.Sets/src/mage/sets/saviorsofkamigawa/InameAsOne.java new file mode 100644 index 00000000000..c48a4509917 --- /dev/null +++ b/Mage.Sets/src/mage/sets/saviorsofkamigawa/InameAsOne.java @@ -0,0 +1,132 @@ +/* + * 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.saviorsofkamigawa; + +import java.util.UUID; +import mage.MageInt; +import mage.MageObject; +import mage.abilities.Ability; +import mage.abilities.common.DiesTriggeredAbility; +import mage.abilities.common.EntersBattlefieldTriggeredAbility; +import mage.abilities.condition.common.CastFromHandCondition; +import mage.abilities.decorator.ConditionalOneShotEffect; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.common.ExileSourceEffect; +import mage.abilities.effects.common.ReturnFromGraveyardToBattlefieldTargetEffect; +import mage.abilities.effects.common.search.SearchLibraryPutInPlayEffect; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.filter.common.FilterPermanentCard; +import mage.filter.predicate.mageobject.SubtypePredicate; +import mage.game.Game; +import mage.players.Player; +import mage.target.common.TargetCardInLibrary; +import mage.target.common.TargetCardInYourGraveyard; +import mage.watchers.common.CastFromHandWatcher; + +/** + * + * @author LevelX2 + */ +public class InameAsOne extends CardImpl { + + private static final FilterPermanentCard filter = new FilterPermanentCard("Spirit permanent card"); + + static { + filter.add(new SubtypePredicate("Spirit")); + } + + public InameAsOne(UUID ownerId) { + super(ownerId, 151, "Iname as One", Rarity.RARE, new CardType[]{CardType.CREATURE}, "{8}{B}{B}{G}{G}"); + this.expansionSetCode = "SOK"; + this.supertype.add("Legendary"); + this.subtype.add("Spirit"); + this.power = new MageInt(8); + this.toughness = new MageInt(8); + + // When Iname as One enters the battlefield, if you cast it from your hand, you may search your library for a Spirit permanent card, put it onto the battlefield, then shuffle your library. + Ability ability = new EntersBattlefieldTriggeredAbility( + new ConditionalOneShotEffect(new SearchLibraryPutInPlayEffect(new TargetCardInLibrary(0, 1, filter), false), + new CastFromHandCondition())); + this.addAbility(ability); + this.addWatcher(new CastFromHandWatcher()); + + // When Iname as One dies, you may exile it. If you do, return target Spirit permanent card from your graveyard to the battlefield. + ability = new DiesTriggeredAbility(new InameAsOneEffect(), false); + ability.addTarget(new TargetCardInYourGraveyard(filter)); + this.addAbility(ability); + } + + public InameAsOne(final InameAsOne card) { + super(card); + } + + @Override + public InameAsOne copy() { + return new InameAsOne(this); + } +} + +class InameAsOneEffect extends OneShotEffect { + + public InameAsOneEffect() { + super(Outcome.Benefit); + this.staticText = "you may exile it. If you do, return target Spirit permanent card from your graveyard to the battlefield"; + } + + public InameAsOneEffect(final InameAsOneEffect effect) { + super(effect); + } + + @Override + public InameAsOneEffect copy() { + return new InameAsOneEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player controller = game.getPlayer(source.getControllerId()); + MageObject sourceObject = game.getObject(source.getSourceId()); + if (controller != null && sourceObject != null) { + if (controller.chooseUse(outcome, "Exile " + sourceObject.getLogName() + " to return Spirit card?", game)) { + // In a Commander game, you may send Iname to the Command Zone instead of exiling it during the resolution + // of its ability. If you do, its ability still works. Iname's ability only requires that you attempted to + // exile it, not that it actually gets to the exile zone. This is similar to how destroying a creature + // (with, for example, Rest in Peace) doesn't necessarily ensure that creature will end up in the graveyard; + // it just so happens that the action of exiling something and the exile zone both use the same word: "exile". + new ExileSourceEffect(Zone.GRAVEYARD).apply(game, source); + return new ReturnFromGraveyardToBattlefieldTargetEffect().apply(game, source); + } + return true; + } + return false; + } +} diff --git a/Mage.Sets/src/mage/sets/saviorsofkamigawa/KahoMinamoHistorian.java b/Mage.Sets/src/mage/sets/saviorsofkamigawa/KahoMinamoHistorian.java new file mode 100644 index 00000000000..41b39c26f91 --- /dev/null +++ b/Mage.Sets/src/mage/sets/saviorsofkamigawa/KahoMinamoHistorian.java @@ -0,0 +1,173 @@ +/* + * 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.saviorsofkamigawa; + +import java.util.UUID; +import mage.MageInt; +import mage.MageObject; +import mage.abilities.Ability; +import mage.abilities.common.EntersBattlefieldTriggeredAbility; +import mage.abilities.common.SimpleActivatedAbility; +import mage.abilities.costs.common.TapSourceCost; +import mage.abilities.costs.mana.ManaCostsImpl; +import mage.abilities.effects.OneShotEffect; +import mage.abilities.effects.SearchEffect; +import mage.cards.Card; +import mage.cards.CardImpl; +import mage.cards.Cards; +import mage.cards.CardsImpl; +import mage.constants.CardType; +import mage.constants.Outcome; +import mage.constants.Rarity; +import mage.constants.Zone; +import mage.filter.Filter; +import mage.filter.FilterCard; +import mage.filter.predicate.mageobject.CardTypePredicate; +import mage.filter.predicate.mageobject.ConvertedManaCostPredicate; +import mage.game.Game; +import mage.players.Player; +import mage.target.common.TargetCardInExile; +import mage.target.common.TargetCardInLibrary; +import mage.util.CardUtil; + +/** + * + * @author LevelX2 + */ +public class KahoMinamoHistorian extends CardImpl { + + public KahoMinamoHistorian(UUID ownerId) { + super(ownerId, 41, "Kaho, Minamo Historian", Rarity.RARE, new CardType[]{CardType.CREATURE}, "{2}{U}{U}"); + this.expansionSetCode = "SOK"; + this.supertype.add("Legendary"); + this.subtype.add("Human"); + this.subtype.add("Wizard"); + this.power = new MageInt(2); + this.toughness = new MageInt(2); + + // When Kaho, Minamo Historian enters the battlefield, search your library for up to three instant cards and exile them. Then shuffle your library. + this.addAbility(new EntersBattlefieldTriggeredAbility(new KahoMinamoHistorianEffect(), false)); + + // {X}, {tap}: You may cast a card with converted mana cost X exiled with Kaho without paying its mana cost. + Ability ability = new SimpleActivatedAbility(Zone.BATTLEFIELD, new KahoMinamoHistorianCastEffect(), new ManaCostsImpl("{X}")); + ability.addCost(new TapSourceCost()); + this.addAbility(ability); + } + + public KahoMinamoHistorian(final KahoMinamoHistorian card) { + super(card); + } + + @Override + public KahoMinamoHistorian copy() { + return new KahoMinamoHistorian(this); + } +} + +class KahoMinamoHistorianEffect extends SearchEffect { + + private static final FilterCard filter = new FilterCard("up to three instant cards"); + + static { + filter.add(new CardTypePredicate(CardType.INSTANT)); + } + + public KahoMinamoHistorianEffect() { + super(new TargetCardInLibrary(0, 3, filter), Outcome.Benefit); + this.staticText = "search your library for up to three instant cards and exile them. Then shuffle your library"; + } + + public KahoMinamoHistorianEffect(final KahoMinamoHistorianEffect effect) { + super(effect); + } + + @Override + public KahoMinamoHistorianEffect copy() { + return new KahoMinamoHistorianEffect(this); + } + + @Override + public boolean apply(Game game, Ability source) { + Player controller = game.getPlayer(source.getControllerId()); + MageObject sourceObject = game.getObject(source.getSourceId()); + if (controller != null && sourceObject != null) { + if (controller.searchLibrary(target, game)) { + UUID exileZone = CardUtil.getCardExileZoneId(game, source); + if (target.getTargets().size() > 0) { + for (UUID cardId : target.getTargets()) { + Card card = controller.getLibrary().getCard(cardId, game); + if (card != null) { + controller.moveCardToExileWithInfo(card, exileZone, sourceObject.getLogName(), source.getSourceId(), game, Zone.LIBRARY); + } + } + } + } + controller.shuffleLibrary(game); + return true; + + } + return false; + } +} + +class KahoMinamoHistorianCastEffect extends OneShotEffect { + + public KahoMinamoHistorianCastEffect() { + super(Outcome.PlayForFree); + this.staticText = "you may cast a card with converted mana cost X exiled with {this} without paying its mana cost"; + } + + public KahoMinamoHistorianCastEffect(final KahoMinamoHistorianCastEffect effect) { + super(effect); + } + + @Override + public KahoMinamoHistorianCastEffect copy() { + return new KahoMinamoHistorianCastEffect(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)); + FilterCard filter = new FilterCard(); + filter.add(new ConvertedManaCostPredicate(Filter.ComparisonType.Equal, source.getManaCostsToPay().getX())); + Cards cards = new CardsImpl(Zone.EXILED, game.getExile().getExileZone(CardUtil.getCardExileZoneId(game, source)).getCards(filter, game)); + if (cards.size() > 0 && controller.choose(Outcome.PlayForFree, cards, target, game)) { + Card card = game.getCard(target.getFirstTarget()); + if (card != null) { + game.getExile().removeCard(card, game); + controller.cast(card.getSpellAbility(), game, true); + } + } + return true; + } + return false; + } +} \ No newline at end of file diff --git a/Mage.Sets/src/mage/sets/timespiral/PrimalForcemage.java b/Mage.Sets/src/mage/sets/timespiral/PrimalForcemage.java new file mode 100644 index 00000000000..206e3a5f10d --- /dev/null +++ b/Mage.Sets/src/mage/sets/timespiral/PrimalForcemage.java @@ -0,0 +1,83 @@ +/* + * 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.common.EntersBattlefieldAllTriggeredAbility; +import mage.abilities.effects.common.continious.BoostTargetEffect; +import mage.cards.CardImpl; +import mage.constants.CardType; +import mage.constants.Duration; +import mage.constants.Rarity; +import mage.constants.SetTargetPointer; +import mage.constants.TargetController; +import mage.constants.Zone; +import mage.filter.common.FilterCreaturePermanent; +import mage.filter.predicate.permanent.AnotherPredicate; +import mage.filter.predicate.permanent.ControllerPredicate; + +/** + * + * @author LevelX2 + */ +public class PrimalForcemage extends CardImpl { + + private final static FilterCreaturePermanent filter = new FilterCreaturePermanent("another creature"); + + static { + filter.add(new ControllerPredicate(TargetController.YOU)); + filter.add(new AnotherPredicate()); + } + + private final String rule = "Whenever another creature enters the battlefield under your control, that creature gets +3/+3 until end of turn."; + + public PrimalForcemage(UUID ownerId) { + super(ownerId, 212, "Primal Forcemage", Rarity.UNCOMMON, new CardType[]{CardType.CREATURE}, "{2}{G}"); + this.expansionSetCode = "TSP"; + this.subtype.add("Elf"); + this.subtype.add("Shaman"); + this.power = new MageInt(2); + this.toughness = new MageInt(2); + + // Whenever another creature enters the battlefield under your control, that creature gets +3/+3 until end of turn. + this.addAbility(new EntersBattlefieldAllTriggeredAbility( + Zone.BATTLEFIELD, + new BoostTargetEffect(3, 3, Duration.EndOfTurn), + filter, false, SetTargetPointer.PERMANENT, rule, true)); + } + + public PrimalForcemage(final PrimalForcemage card) { + super(card); + } + + @Override + public PrimalForcemage copy() { + return new PrimalForcemage(this); + } +}