[SPM] implement The Spot, Living Portal

This commit is contained in:
jmlundeen 2025-09-06 22:11:47 -05:00
parent d0ee2fef06
commit 8f1050a834
4 changed files with 203 additions and 3 deletions

View file

@ -0,0 +1,82 @@
package mage.cards.t;
import mage.MageInt;
import mage.abilities.Ability;
import mage.abilities.common.DiesSourceTriggeredAbility;
import mage.abilities.common.EntersBattlefieldTriggeredAbility;
import mage.abilities.costs.common.PutSourceOnBottomOwnerLibraryCost;
import mage.abilities.effects.common.DoIfCostPaid;
import mage.abilities.effects.common.ExileTargetForSourceEffect;
import mage.abilities.effects.common.ReturnFromExileForSourceEffect;
import mage.cards.CardImpl;
import mage.cards.CardSetInfo;
import mage.constants.CardType;
import mage.constants.SubType;
import mage.constants.SuperType;
import mage.constants.Zone;
import mage.filter.FilterCard;
import mage.filter.predicate.Predicates;
import mage.target.common.TargetCardInGraveyard;
import mage.target.common.TargetNonlandPermanent;
import mage.target.targetpointer.EachTargetPointer;
import java.util.Arrays;
import java.util.UUID;
import java.util.stream.Collectors;
/**
*
* @author Jmlundeen
*/
public final class TheSpotLivingPortal extends CardImpl {
private static final FilterCard filter = new FilterCard("nonland permanent card from a graveyard");
static {
filter.add(Predicates.or(
Arrays.stream(CardType.values())
.filter(CardType::isPermanentType)
.filter(type -> type != CardType.LAND)
.map(CardType::getPredicate)
.collect(Collectors.toSet()))
);
}
public TheSpotLivingPortal(UUID ownerId, CardSetInfo setInfo) {
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{3}{W}{B}");
this.supertype.add(SuperType.LEGENDARY);
this.subtype.add(SubType.HUMAN);
this.subtype.add(SubType.SCIENTIST);
this.subtype.add(SubType.VILLAIN);
this.power = new MageInt(4);
this.toughness = new MageInt(4);
// When The Spot enters, exile up to one target nonland permanent and up to one target nonland permanent card from a graveyard.
Ability ability = new EntersBattlefieldTriggeredAbility(new ExileTargetForSourceEffect()
.setTargetPointer(new EachTargetPointer()));
ability.addTarget(new TargetNonlandPermanent(0, 1));
ability.addTarget(new TargetCardInGraveyard(0, 1, filter));
this.addAbility(ability);
// When The Spot dies, put him on the bottom of his owner's library. If you do, return the exiled cards to their owners' hands.
DoIfCostPaid effect = new DoIfCostPaid(
new ReturnFromExileForSourceEffect(Zone.HAND)
.withText(true, true, false),
null,
new PutSourceOnBottomOwnerLibraryCost()
.setText("put him on the bottom of his owner's library"),
false
);
this.addAbility(new DiesSourceTriggeredAbility(effect, false));
}
private TheSpotLivingPortal(final TheSpotLivingPortal card) {
super(card);
}
@Override
public TheSpotLivingPortal copy() {
return new TheSpotLivingPortal(this);
}
}

View file

@ -273,6 +273,8 @@ public final class MarvelsSpiderMan extends ExpansionSet {
cards.add(new SetCardInfo("The Clone Saga", 219, Rarity.RARE, mage.cards.t.TheCloneSaga.class, NON_FULL_USE_VARIOUS)); cards.add(new SetCardInfo("The Clone Saga", 219, Rarity.RARE, mage.cards.t.TheCloneSaga.class, NON_FULL_USE_VARIOUS));
cards.add(new SetCardInfo("The Clone Saga", 28, Rarity.RARE, mage.cards.t.TheCloneSaga.class, NON_FULL_USE_VARIOUS)); cards.add(new SetCardInfo("The Clone Saga", 28, Rarity.RARE, mage.cards.t.TheCloneSaga.class, NON_FULL_USE_VARIOUS));
cards.add(new SetCardInfo("The Spot's Portal", 68, Rarity.UNCOMMON, mage.cards.t.TheSpotsPortal.class)); cards.add(new SetCardInfo("The Spot's Portal", 68, Rarity.UNCOMMON, mage.cards.t.TheSpotsPortal.class));
cards.add(new SetCardInfo("The Spot, Living Portal", 153, Rarity.RARE, mage.cards.t.TheSpotLivingPortal.class, NON_FULL_USE_VARIOUS));
cards.add(new SetCardInfo("The Spot, Living Portal", 231, Rarity.RARE, mage.cards.t.TheSpotLivingPortal.class, NON_FULL_USE_VARIOUS));
cards.add(new SetCardInfo("Thwip!", 20, Rarity.COMMON, mage.cards.t.Thwip.class)); cards.add(new SetCardInfo("Thwip!", 20, Rarity.COMMON, mage.cards.t.Thwip.class));
cards.add(new SetCardInfo("Tombstone, Career Criminal", 70, Rarity.UNCOMMON, mage.cards.t.TombstoneCareerCriminal.class)); cards.add(new SetCardInfo("Tombstone, Career Criminal", 70, Rarity.UNCOMMON, mage.cards.t.TombstoneCareerCriminal.class));
cards.add(new SetCardInfo("Ultimate Green Goblin", 157, Rarity.RARE, mage.cards.u.UltimateGreenGoblin.class, NON_FULL_USE_VARIOUS)); cards.add(new SetCardInfo("Ultimate Green Goblin", 157, Rarity.RARE, mage.cards.u.UltimateGreenGoblin.class, NON_FULL_USE_VARIOUS));

View file

@ -0,0 +1,115 @@
package org.mage.test.cards.single.spm;
import mage.constants.PhaseStep;
import mage.constants.Zone;
import org.junit.Test;
import org.mage.test.player.TestPlayer;
import org.mage.test.serverside.base.CardTestPlayerBase;
/**
*
* @author Jmlundeen
*/
public class TheSpotLivingPortalTest extends CardTestPlayerBase {
/*
The Spot, Living Portal
{3}{W}{B}
Legendary Creature - Human Scientist Villain
When The Spot enters, exile up to one target nonland permanent and up to one target nonland permanent card from a graveyard.
When The Spot dies, put him on the bottom of his owner's library. If you do, return the exiled cards to their owners' hands.
4/4
*/
private static final String theSpotLivingPortal = "The Spot, Living Portal";
/*
Bear Cub
{1}{G}
Creature - Bear
2/2
*/
private static final String bearCub = "Bear Cub";
/*
Fugitive Wizard
{U}
Creature - Human Wizard
1/1
*/
private static final String fugitiveWizard = "Fugitive Wizard";
@Test
public void testTheSpotLivingPortal() {
setStrictChooseMode(true);
addCustomEffect_TargetDestroy(playerB);
addCard(Zone.HAND, playerA, theSpotLivingPortal);
addCard(Zone.GRAVEYARD, playerA, bearCub);
addCard(Zone.BATTLEFIELD, playerA, "Scrubland", 5);
addCard(Zone.BATTLEFIELD, playerB, fugitiveWizard);
castSpell(1, PhaseStep.PRECOMBAT_MAIN, playerA, theSpotLivingPortal);
addTarget(playerA, bearCub);
addTarget(playerA, fugitiveWizard);
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "target destroy");
addTarget(playerB, theSpotLivingPortal);
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
execute();
assertHandCount(playerA, bearCub, 1);
assertHandCount(playerB, fugitiveWizard, 1);
assertLibraryCount(playerA, theSpotLivingPortal, 1);
}
@Test
public void testOnlyGraveyard() {
setStrictChooseMode(true);
addCustomEffect_TargetDestroy(playerB);
addCard(Zone.HAND, playerA, theSpotLivingPortal);
addCard(Zone.GRAVEYARD, playerA, bearCub);
addCard(Zone.BATTLEFIELD, playerA, "Scrubland", 5);
addCard(Zone.BATTLEFIELD, playerB, fugitiveWizard);
castSpell(1, PhaseStep.PRECOMBAT_MAIN, playerA, theSpotLivingPortal);
addTarget(playerA, TestPlayer.TARGET_SKIP);
addTarget(playerA, bearCub);
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "target destroy");
addTarget(playerB, theSpotLivingPortal);
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
execute();
assertHandCount(playerA, bearCub, 1);
assertLibraryCount(playerA, theSpotLivingPortal, 1);
}
@Test
public void testOnlyBattlefield() {
setStrictChooseMode(true);
addCustomEffect_TargetDestroy(playerB);
addCard(Zone.HAND, playerA, theSpotLivingPortal);
addCard(Zone.GRAVEYARD, playerA, bearCub);
addCard(Zone.BATTLEFIELD, playerA, "Scrubland", 5);
addCard(Zone.BATTLEFIELD, playerB, fugitiveWizard);
castSpell(1, PhaseStep.PRECOMBAT_MAIN, playerA, theSpotLivingPortal);
addTarget(playerA, fugitiveWizard);
addTarget(playerA, TestPlayer.TARGET_SKIP);
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "target destroy");
addTarget(playerB, theSpotLivingPortal);
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
execute();
assertHandCount(playerB, fugitiveWizard, 1);
assertLibraryCount(playerA, theSpotLivingPortal, 1);
}
}

View file

@ -1,7 +1,6 @@
package mage.abilities.costs.common; package mage.abilities.costs.common;
import java.util.UUID;
import mage.abilities.Ability; import mage.abilities.Ability;
import mage.abilities.costs.Cost; import mage.abilities.costs.Cost;
import mage.abilities.costs.CostImpl; import mage.abilities.costs.CostImpl;
@ -10,6 +9,8 @@ import mage.game.Game;
import mage.game.permanent.Permanent; import mage.game.permanent.Permanent;
import mage.players.Player; import mage.players.Player;
import java.util.UUID;
/** /**
* *
* @author LevelX2 * @author LevelX2
@ -27,7 +28,7 @@ public class PutSourceOnBottomOwnerLibraryCost extends CostImpl {
@Override @Override
public boolean pay(Ability ability, Game game, Ability source, UUID controllerId, boolean noMana, Cost costToPay) { public boolean pay(Ability ability, Game game, Ability source, UUID controllerId, boolean noMana, Cost costToPay) {
Player player = game.getPlayer(controllerId); Player player = game.getPlayer(controllerId);
Permanent sourcePermanent = game.getPermanent(source.getSourceId()); Permanent sourcePermanent = game.getPermanentOrLKIBattlefield(source.getSourceId());
if (player != null && sourcePermanent != null) { if (player != null && sourcePermanent != null) {
paid = true; paid = true;
player.putCardsOnBottomOfLibrary(new CardsImpl(sourcePermanent), game, ability, false); player.putCardsOnBottomOfLibrary(new CardsImpl(sourcePermanent), game, ability, false);
@ -37,7 +38,7 @@ public class PutSourceOnBottomOwnerLibraryCost extends CostImpl {
@Override @Override
public boolean canPay(Ability ability, Ability source, UUID controllerId, Game game) { public boolean canPay(Ability ability, Ability source, UUID controllerId, Game game) {
return game.getPermanent(source.getSourceId()) != null; return game.getPermanentOrLKIBattlefield(source.getSourceId()) != null;
} }
@Override @Override