mirror of
https://github.com/correl/mage.git
synced 2024-12-26 03:00:11 +00:00
Merge branch 'master' into master
This commit is contained in:
commit
4d5e8d4a91
3 changed files with 338 additions and 1 deletions
335
Mage.Sets/src/mage/cards/m/MairsilThePretender.java
Normal file
335
Mage.Sets/src/mage/cards/m/MairsilThePretender.java
Normal file
|
@ -0,0 +1,335 @@
|
||||||
|
/*
|
||||||
|
* 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.cards.m;
|
||||||
|
|
||||||
|
import java.util.HashMap;
|
||||||
|
import java.util.HashSet;
|
||||||
|
import java.util.Map;
|
||||||
|
import java.util.Map.Entry;
|
||||||
|
import java.util.Optional;
|
||||||
|
import java.util.Set;
|
||||||
|
import java.util.UUID;
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.ActivatedAbility;
|
||||||
|
import mage.abilities.common.EntersBattlefieldTriggeredAbility;
|
||||||
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
|
import mage.abilities.effects.ContinuousEffectImpl;
|
||||||
|
import mage.abilities.effects.ContinuousRuleModifyingEffectImpl;
|
||||||
|
import mage.abilities.effects.Effect;
|
||||||
|
import mage.abilities.effects.OneShotEffect;
|
||||||
|
import mage.cards.Card;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.constants.Layer;
|
||||||
|
import mage.constants.Outcome;
|
||||||
|
import mage.constants.SubLayer;
|
||||||
|
import mage.constants.SuperType;
|
||||||
|
import mage.constants.TargetController;
|
||||||
|
import mage.constants.WatcherScope;
|
||||||
|
import mage.constants.Zone;
|
||||||
|
import mage.counters.CounterType;
|
||||||
|
import mage.filter.FilterCard;
|
||||||
|
import mage.filter.predicate.Predicates;
|
||||||
|
import mage.filter.predicate.mageobject.CardTypePredicate;
|
||||||
|
import mage.filter.predicate.other.CounterCardPredicate;
|
||||||
|
import mage.filter.predicate.other.OwnerPredicate;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.events.GameEvent;
|
||||||
|
import mage.game.events.ZoneChangeEvent;
|
||||||
|
import mage.game.permanent.Permanent;
|
||||||
|
import mage.game.stack.StackAbility;
|
||||||
|
import mage.players.Player;
|
||||||
|
import mage.target.Target;
|
||||||
|
import mage.target.common.TargetCardInHand;
|
||||||
|
import mage.target.common.TargetCardInYourGraveyard;
|
||||||
|
import mage.watchers.Watcher;
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class MairsilThePretender extends CardImpl {
|
||||||
|
|
||||||
|
public MairsilThePretender(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{1}{U}{B}{R}");
|
||||||
|
|
||||||
|
addSuperType(SuperType.LEGENDARY);
|
||||||
|
this.subtype.add("Human");
|
||||||
|
this.subtype.add("Wizard");
|
||||||
|
this.power = new MageInt(4);
|
||||||
|
this.toughness = new MageInt(4);
|
||||||
|
|
||||||
|
// When Mairsil, the Pretender enters the battlefield, you may exile an artifact or creature card from your hand or graveyard and put a cage counter on it.
|
||||||
|
this.addAbility(new EntersBattlefieldTriggeredAbility(new MairsilThePretenderExileEffect(), true));
|
||||||
|
|
||||||
|
// Mairsil, the Pretender has all activated abilities of all cards you own in exile with cage counters on them. You may activate each of those abilities only once each turn.
|
||||||
|
Ability ability = new SimpleStaticAbility(Zone.BATTLEFIELD, new MairsilThePretenderGainAbilitiesEffect());
|
||||||
|
ability.addEffect(new MairsilThePretenderRuleModifyingEffect());
|
||||||
|
this.addAbility(ability, new MairsilThePretenderWatcher());
|
||||||
|
}
|
||||||
|
|
||||||
|
public MairsilThePretender(final MairsilThePretender card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MairsilThePretender copy() {
|
||||||
|
return new MairsilThePretender(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MairsilThePretenderExileEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
private static final FilterCard filter = new FilterCard();
|
||||||
|
|
||||||
|
static {
|
||||||
|
filter.add(Predicates.or(new CardTypePredicate(CardType.ARTIFACT), new CardTypePredicate(CardType.CREATURE)));
|
||||||
|
}
|
||||||
|
|
||||||
|
MairsilThePretenderExileEffect() {
|
||||||
|
super(Outcome.Benefit);
|
||||||
|
this.staticText = "you may exile an artifact or creature card from your hand or graveyard and put a cage counter on it.";
|
||||||
|
}
|
||||||
|
|
||||||
|
MairsilThePretenderExileEffect(final MairsilThePretenderExileEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MairsilThePretenderExileEffect copy() {
|
||||||
|
return new MairsilThePretenderExileEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Player controller = game.getPlayer(source.getControllerId());
|
||||||
|
if (controller != null) {
|
||||||
|
if (controller.chooseUse(Outcome.PutCardInPlay, "Exile a card from your hand? (No = from graveyard)", source, game)) {
|
||||||
|
Target target = new TargetCardInHand(0, 1, filter);
|
||||||
|
controller.choose(outcome, target, source.getSourceId(), game);
|
||||||
|
Card card = controller.getHand().get(target.getFirstTarget(), game);
|
||||||
|
if (card != null) {
|
||||||
|
controller.moveCards(card, Zone.EXILED, source, game);
|
||||||
|
card.addCounters(CounterType.CAGE.createInstance(), source, game);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Target target = new TargetCardInYourGraveyard(0, 1, filter);
|
||||||
|
target.choose(Outcome.PutCardInPlay, source.getControllerId(), source.getSourceId(), game);
|
||||||
|
Card card = controller.getGraveyard().get(target.getFirstTarget(), game);
|
||||||
|
if (card != null) {
|
||||||
|
controller.moveCards(card, Zone.EXILED, source, game);
|
||||||
|
card.addCounters(CounterType.CAGE.createInstance(), source, game);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MairsilThePretenderGainAbilitiesEffect extends ContinuousEffectImpl {
|
||||||
|
|
||||||
|
private static final FilterCard filter = new FilterCard();
|
||||||
|
|
||||||
|
static {
|
||||||
|
filter.add(new CounterCardPredicate(CounterType.CAGE));
|
||||||
|
filter.add(new OwnerPredicate(TargetController.YOU));
|
||||||
|
}
|
||||||
|
|
||||||
|
public MairsilThePretenderGainAbilitiesEffect() {
|
||||||
|
super(Duration.WhileOnBattlefield, Layer.AbilityAddingRemovingEffects_6, SubLayer.NA, Outcome.AddAbility);
|
||||||
|
staticText = "{this} has all activated abilities of all cards you own in exile with cage counters on them. You may activate each of those abilities only once each turn";
|
||||||
|
}
|
||||||
|
|
||||||
|
public MairsilThePretenderGainAbilitiesEffect(final MairsilThePretenderGainAbilitiesEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Permanent perm = game.getPermanent(source.getSourceId());
|
||||||
|
if (perm != null) {
|
||||||
|
for (Card card : game.getExile().getAllCards(game)) {
|
||||||
|
if (filter.match(card, game)) {
|
||||||
|
for (Ability ability : card.getAbilities()) {
|
||||||
|
if (ability instanceof ActivatedAbility) {
|
||||||
|
UUID originaId = ability.getId();
|
||||||
|
ActivatedAbility copyAbility = (ActivatedAbility) ability.copy();
|
||||||
|
Effect effect = new DoNothingEffect();
|
||||||
|
effect.setValue("key", originaId);
|
||||||
|
copyAbility.addEffect(effect);
|
||||||
|
perm.addAbility(copyAbility, card.getId(), game);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MairsilThePretenderGainAbilitiesEffect copy() {
|
||||||
|
return new MairsilThePretenderGainAbilitiesEffect(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MairsilThePretenderWatcher extends Watcher {
|
||||||
|
|
||||||
|
public final Map<UUID, Set<UUID>> activatedThisTurnAbilities = new HashMap<>();
|
||||||
|
|
||||||
|
public MairsilThePretenderWatcher() {
|
||||||
|
super(MairsilThePretenderWatcher.class.getSimpleName(), WatcherScope.GAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
public MairsilThePretenderWatcher(final MairsilThePretenderWatcher watcher) {
|
||||||
|
super(watcher);
|
||||||
|
for (Entry<UUID, Set<UUID>> entry : watcher.activatedThisTurnAbilities.entrySet()) {
|
||||||
|
activatedThisTurnAbilities.put(entry.getKey(), entry.getValue());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void watch(GameEvent event, Game game) {
|
||||||
|
if (event.getType() == GameEvent.EventType.ZONE_CHANGE && event instanceof ZoneChangeEvent) {
|
||||||
|
ZoneChangeEvent zEvent = (ZoneChangeEvent) event;
|
||||||
|
if (zEvent.getFromZone() == Zone.BATTLEFIELD) {
|
||||||
|
Permanent permanent = zEvent.getTarget();
|
||||||
|
if (permanent != null) {
|
||||||
|
this.activatedThisTurnAbilities.remove(permanent.getId());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (event.getType() == GameEvent.EventType.ACTIVATED_ABILITY) {
|
||||||
|
Set<UUID> permAbilities;
|
||||||
|
if (activatedThisTurnAbilities.keySet().contains(event.getSourceId())) {
|
||||||
|
permAbilities = activatedThisTurnAbilities.get(event.getSourceId());
|
||||||
|
} else {
|
||||||
|
permAbilities = new HashSet<>();
|
||||||
|
}
|
||||||
|
StackAbility ability = (StackAbility) game.getStack().getStackObject(event.getSourceId());
|
||||||
|
if (ability != null && ability.getStackAbility().isActivated()) {
|
||||||
|
for (Effect effect : ability.getAllEffects()) {
|
||||||
|
if (effect instanceof DoNothingEffect) {
|
||||||
|
permAbilities.add((UUID) effect.getValue("key"));
|
||||||
|
this.activatedThisTurnAbilities.put(event.getSourceId(), permAbilities);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void reset() {
|
||||||
|
activatedThisTurnAbilities.clear();
|
||||||
|
}
|
||||||
|
|
||||||
|
public Map<UUID, Set<UUID>> getActivatedThisTurnAbilities() {
|
||||||
|
return this.activatedThisTurnAbilities;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MairsilThePretenderWatcher copy() {
|
||||||
|
return new MairsilThePretenderWatcher(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class MairsilThePretenderRuleModifyingEffect extends ContinuousRuleModifyingEffectImpl {
|
||||||
|
|
||||||
|
public MairsilThePretenderRuleModifyingEffect() {
|
||||||
|
super(Duration.WhileOnBattlefield, Outcome.Detriment);
|
||||||
|
}
|
||||||
|
|
||||||
|
public MairsilThePretenderRuleModifyingEffect(final MairsilThePretenderRuleModifyingEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MairsilThePretenderRuleModifyingEffect copy() {
|
||||||
|
return new MairsilThePretenderRuleModifyingEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean checksEventType(GameEvent event, Game game) {
|
||||||
|
return event.getType() == GameEvent.EventType.ACTIVATE_ABILITY;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean applies(GameEvent event, Ability source, Game game) {
|
||||||
|
Optional<Ability> ability = game.getAbility(event.getTargetId(), event.getSourceId());
|
||||||
|
MairsilThePretenderWatcher watcher = (MairsilThePretenderWatcher) game.getState().getWatchers().get(MairsilThePretenderWatcher.class.getSimpleName());
|
||||||
|
if (watcher != null && ability != null && ability.isPresent()) {
|
||||||
|
for (Effect effect : ability.get().getAllEffects()) {
|
||||||
|
if (effect instanceof DoNothingEffect) {
|
||||||
|
UUID originalID = (UUID) effect.getValue("key");
|
||||||
|
if (watcher.getActivatedThisTurnAbilities().keySet().contains(event.getSourceId())) {
|
||||||
|
if (watcher.getActivatedThisTurnAbilities().get(event.getSourceId()).contains(originalID)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getInfoMessage(Ability source, GameEvent event, Game game) {
|
||||||
|
return "This ability can only be activated once each turn.";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class DoNothingEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
DoNothingEffect() {
|
||||||
|
super(Outcome.Neutral);
|
||||||
|
}
|
||||||
|
|
||||||
|
DoNothingEffect(final DoNothingEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public DoNothingEffect copy() {
|
||||||
|
return new DoNothingEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
|
@ -79,6 +79,7 @@ public class Commander2017 extends ExpansionSet {
|
||||||
cards.add(new SetCardInfo("Kindred Summons", 32, Rarity.RARE, mage.cards.k.KindredSummons.class));
|
cards.add(new SetCardInfo("Kindred Summons", 32, Rarity.RARE, mage.cards.k.KindredSummons.class));
|
||||||
cards.add(new SetCardInfo("Licia, Sanguine Tribune", 40, Rarity.MYTHIC, mage.cards.l.LiciaSanguineTribune.class));
|
cards.add(new SetCardInfo("Licia, Sanguine Tribune", 40, Rarity.MYTHIC, mage.cards.l.LiciaSanguineTribune.class));
|
||||||
cards.add(new SetCardInfo("Magus of the Mind", 12, Rarity.RARE, mage.cards.m.MagusOfTheMind.class));
|
cards.add(new SetCardInfo("Magus of the Mind", 12, Rarity.RARE, mage.cards.m.MagusOfTheMind.class));
|
||||||
|
cards.add(new SetCardInfo("Mairsil, the Pretender", 41, Rarity.MYTHIC, mage.cards.m.MairsilThePretender.class));
|
||||||
cards.add(new SetCardInfo("Mathas, Fiend Seeker", 42, Rarity.MYTHIC, mage.cards.m.MathasFiendSeeker.class));
|
cards.add(new SetCardInfo("Mathas, Fiend Seeker", 42, Rarity.MYTHIC, mage.cards.m.MathasFiendSeeker.class));
|
||||||
cards.add(new SetCardInfo("Mirri, Weatherlight Duelist", 43, Rarity.MYTHIC, mage.cards.m.MirriWeatherlightDuelist.class));
|
cards.add(new SetCardInfo("Mirri, Weatherlight Duelist", 43, Rarity.MYTHIC, mage.cards.m.MirriWeatherlightDuelist.class));
|
||||||
cards.add(new SetCardInfo("Mirror of the Forebears", 54, Rarity.UNCOMMON, mage.cards.m.MirrorOfTheForebears.class));
|
cards.add(new SetCardInfo("Mirror of the Forebears", 54, Rarity.UNCOMMON, mage.cards.m.MirrorOfTheForebears.class));
|
||||||
|
@ -99,4 +100,4 @@ public class Commander2017 extends ExpansionSet {
|
||||||
cards.add(new SetCardInfo("Vindictive Lich", 21, Rarity.RARE, mage.cards.v.VindictiveLich.class));
|
cards.add(new SetCardInfo("Vindictive Lich", 21, Rarity.RARE, mage.cards.v.VindictiveLich.class));
|
||||||
cards.add(new SetCardInfo("Wasitora, Nekoru Queen", 49, Rarity.MYTHIC, mage.cards.w.WasitoraNekoruQueen.class));
|
cards.add(new SetCardInfo("Wasitora, Nekoru Queen", 49, Rarity.MYTHIC, mage.cards.w.WasitoraNekoruQueen.class));
|
||||||
}
|
}
|
||||||
}
|
}
|
|
@ -42,6 +42,7 @@ public enum CounterType {
|
||||||
BOUNTY("bounty"),
|
BOUNTY("bounty"),
|
||||||
BRIBERY("bribery"),
|
BRIBERY("bribery"),
|
||||||
BRICK("brick"),
|
BRICK("brick"),
|
||||||
|
CAGE("cage"),
|
||||||
CARRION("carrion"),
|
CARRION("carrion"),
|
||||||
CHARGE("charge"),
|
CHARGE("charge"),
|
||||||
CORPSE("corpse"),
|
CORPSE("corpse"),
|
||||||
|
|
Loading…
Reference in a new issue