mirror of
https://github.com/correl/mage.git
synced 2025-01-12 11:08:01 +00:00
[CMR] Implemented Averna, the Chaos Bloom
This commit is contained in:
parent
c451436cb7
commit
ee03e4e862
4 changed files with 114 additions and 24 deletions
74
Mage.Sets/src/mage/cards/a/AvernaTheChaosBloom.java
Normal file
74
Mage.Sets/src/mage/cards/a/AvernaTheChaosBloom.java
Normal file
|
@ -0,0 +1,74 @@
|
|||
package mage.cards.a;
|
||||
|
||||
import mage.MageInt;
|
||||
import mage.abilities.Ability;
|
||||
import mage.abilities.common.SimpleStaticAbility;
|
||||
import mage.abilities.effects.ReplacementEffectImpl;
|
||||
import mage.cards.CardImpl;
|
||||
import mage.cards.CardSetInfo;
|
||||
import mage.constants.*;
|
||||
import mage.game.Game;
|
||||
import mage.game.events.GameEvent;
|
||||
|
||||
import java.util.UUID;
|
||||
|
||||
/**
|
||||
* @author TheElk801
|
||||
*/
|
||||
public final class AvernaTheChaosBloom extends CardImpl {
|
||||
|
||||
public AvernaTheChaosBloom(UUID ownerId, CardSetInfo setInfo) {
|
||||
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{G}{U}{R}");
|
||||
|
||||
this.addSuperType(SuperType.LEGENDARY);
|
||||
this.subtype.add(SubType.ELEMENTAL);
|
||||
this.subtype.add(SubType.SHAMAN);
|
||||
this.power = new MageInt(4);
|
||||
this.toughness = new MageInt(2);
|
||||
|
||||
// As you cascade, you may put a land card from among the exiled cards onto the battlefield tapped.
|
||||
this.addAbility(new SimpleStaticAbility(new AvernaTheChaosBloomReplacementEffect()));
|
||||
}
|
||||
|
||||
private AvernaTheChaosBloom(final AvernaTheChaosBloom card) {
|
||||
super(card);
|
||||
}
|
||||
|
||||
@Override
|
||||
public AvernaTheChaosBloom copy() {
|
||||
return new AvernaTheChaosBloom(this);
|
||||
}
|
||||
}
|
||||
|
||||
class AvernaTheChaosBloomReplacementEffect extends ReplacementEffectImpl {
|
||||
|
||||
AvernaTheChaosBloomReplacementEffect() {
|
||||
super(Duration.WhileOnBattlefield, Outcome.Benefit);
|
||||
staticText = "As you cascade, you may put a land card from among the exiled cards onto the battlefield tapped.";
|
||||
}
|
||||
|
||||
private AvernaTheChaosBloomReplacementEffect(final AvernaTheChaosBloomReplacementEffect effect) {
|
||||
super(effect);
|
||||
}
|
||||
|
||||
@Override
|
||||
public AvernaTheChaosBloomReplacementEffect copy() {
|
||||
return new AvernaTheChaosBloomReplacementEffect(this);
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean checksEventType(GameEvent event, Game game) {
|
||||
return event.getType() == GameEvent.EventType.CASCADE_LAND;
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean applies(GameEvent event, Ability source, Game game) {
|
||||
return event.getPlayerId().equals(source.getControllerId());
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean replaceEvent(GameEvent event, Ability source, Game game) {
|
||||
event.setAmount(event.getAmount() + 1);
|
||||
return false;
|
||||
}
|
||||
}
|
|
@ -30,6 +30,7 @@ public final class CommanderLegends extends ExpansionSet {
|
|||
cards.add(new SetCardInfo("Acidic Slime", 421, Rarity.UNCOMMON, mage.cards.a.AcidicSlime.class));
|
||||
cards.add(new SetCardInfo("Alena, Kessig Trapper", 160, Rarity.UNCOMMON, mage.cards.a.AlenaKessigTrapper.class));
|
||||
cards.add(new SetCardInfo("Amareth, the Lustrous", 266, Rarity.RARE, mage.cards.a.AmarethTheLustrous.class));
|
||||
cards.add(new SetCardInfo("Averna, the Chaos Bloom", 269, Rarity.RARE, mage.cards.a.AvernaTheChaosBloom.class));
|
||||
cards.add(new SetCardInfo("Bladegriff Prototype", 300, Rarity.RARE, mage.cards.b.BladegriffPrototype.class));
|
||||
cards.add(new SetCardInfo("Brazen Freebooter", 164, Rarity.COMMON, mage.cards.b.BrazenFreebooter.class));
|
||||
cards.add(new SetCardInfo("Briarblade Adept", 111, Rarity.COMMON, mage.cards.b.BriarbladeAdept.class));
|
||||
|
|
|
@ -5,14 +5,18 @@ import mage.abilities.Ability;
|
|||
import mage.abilities.TriggeredAbilityImpl;
|
||||
import mage.abilities.effects.OneShotEffect;
|
||||
import mage.cards.Card;
|
||||
import mage.cards.Cards;
|
||||
import mage.cards.CardsImpl;
|
||||
import mage.constants.Outcome;
|
||||
import mage.constants.Zone;
|
||||
import mage.game.ExileZone;
|
||||
import mage.filter.FilterCard;
|
||||
import mage.filter.StaticFilters;
|
||||
import mage.filter.common.FilterLandCard;
|
||||
import mage.game.Game;
|
||||
import mage.game.events.GameEvent;
|
||||
import mage.game.stack.Spell;
|
||||
import mage.players.Player;
|
||||
import mage.target.common.TargetCardInExile;
|
||||
|
||||
/**
|
||||
* @author BetaSteward_at_googlemail.com
|
||||
|
@ -25,7 +29,8 @@ public class CascadeAbility extends TriggeredAbilityImpl {
|
|||
+ "nonland card that costs less."
|
||||
+ " You may cast it without paying its mana cost. "
|
||||
+ "Put the exiled cards on the bottom in a random order.)</i>";
|
||||
private boolean withReminder;
|
||||
private final boolean withReminder;
|
||||
private static final FilterCard filter = new FilterLandCard("land card (to put onto the battlefield)");
|
||||
|
||||
public CascadeAbility() {
|
||||
this(true);
|
||||
|
@ -36,7 +41,7 @@ public class CascadeAbility extends TriggeredAbilityImpl {
|
|||
this.withReminder = withReminder;
|
||||
}
|
||||
|
||||
public CascadeAbility(final CascadeAbility ability) {
|
||||
private CascadeAbility(final CascadeAbility ability) {
|
||||
super(ability);
|
||||
this.withReminder = ability.withReminder;
|
||||
}
|
||||
|
@ -71,11 +76,11 @@ public class CascadeAbility extends TriggeredAbilityImpl {
|
|||
|
||||
class CascadeEffect extends OneShotEffect {
|
||||
|
||||
public CascadeEffect() {
|
||||
CascadeEffect() {
|
||||
super(Outcome.PutCardInPlay);
|
||||
}
|
||||
|
||||
public CascadeEffect(CascadeEffect effect) {
|
||||
private CascadeEffect(CascadeEffect effect) {
|
||||
super(effect);
|
||||
}
|
||||
|
||||
|
@ -86,8 +91,7 @@ class CascadeEffect extends OneShotEffect {
|
|||
if (controller == null) {
|
||||
return false;
|
||||
}
|
||||
ExileZone exile = game.getExile().createZone(source.getSourceId(),
|
||||
controller.getName() + " Cascade");
|
||||
Cards cards = new CardsImpl();
|
||||
card = game.getCard(source.getSourceId());
|
||||
if (card == null) {
|
||||
return false;
|
||||
|
@ -98,33 +102,43 @@ class CascadeEffect extends OneShotEffect {
|
|||
if (card == null) {
|
||||
break;
|
||||
}
|
||||
controller.moveCardsToExile(card, source, game, true, exile.getId(), exile.getName());
|
||||
cards.add(card);
|
||||
controller.moveCards(card, Zone.EXILED, source, game);
|
||||
} while (controller.canRespond()
|
||||
&& (card.isLand()
|
||||
|| !cardThatCostsLess(sourceCost, card, game)));
|
||||
&& (card.isLand() || card.getConvertedManaCost() >= sourceCost));
|
||||
|
||||
controller.getLibrary().reset(); // set back empty draw state if that caused an empty draw
|
||||
|
||||
if (card != null) {
|
||||
if (controller.chooseUse(outcome, "Use cascade effect on " + card.getLogName() + '?', source, game)) {
|
||||
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), Boolean.TRUE);
|
||||
controller.cast(controller.chooseAbilityForCast(card, game, true),
|
||||
game, true, new ApprovingObject(source, game));
|
||||
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), null);
|
||||
}
|
||||
GameEvent event = GameEvent.getEvent(
|
||||
GameEvent.EventType.CASCADE_LAND, source.getSourceId(),
|
||||
source.getSourceId(), source.getControllerId(), 0
|
||||
);
|
||||
game.replaceEvent(event);
|
||||
if (event.getAmount() > 0) {
|
||||
TargetCardInExile target = new TargetCardInExile(
|
||||
0, event.getAmount(), StaticFilters.FILTER_CARD_LAND, null, true
|
||||
);
|
||||
controller.choose(Outcome.PutCardInPlay, cards, target, game);
|
||||
controller.moveCards(
|
||||
new CardsImpl(target.getTargets()).getCards(game), Zone.BATTLEFIELD,
|
||||
source, game, true, false, false, null
|
||||
);
|
||||
}
|
||||
if (card != null && controller.chooseUse(
|
||||
outcome, "Use cascade effect on " + card.getLogName() + '?', source, game
|
||||
)) {
|
||||
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), Boolean.TRUE);
|
||||
controller.cast(controller.chooseAbilityForCast(card, game, true),
|
||||
game, true, new ApprovingObject(source, game));
|
||||
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), null);
|
||||
}
|
||||
// Move the remaining cards to the buttom of the library in a random order
|
||||
return controller.putCardsOnBottomOfLibrary(new CardsImpl(exile), game, source, false);
|
||||
cards.removeIf(uuid -> game.getState().getZone(uuid) != Zone.EXILED);
|
||||
return controller.putCardsOnBottomOfLibrary(cards, game, source, false);
|
||||
}
|
||||
|
||||
@Override
|
||||
public CascadeEffect copy() {
|
||||
return new CascadeEffect(this);
|
||||
}
|
||||
|
||||
private boolean cardThatCostsLess(int value, Card card, Game game) {
|
||||
|
||||
return card.getConvertedManaCost() < value;
|
||||
|
||||
}
|
||||
}
|
||||
|
|
|
@ -256,6 +256,7 @@ public class GameEvent implements Serializable {
|
|||
PAID_CUMULATIVE_UPKEEP,
|
||||
DIDNT_PAY_CUMULATIVE_UPKEEP,
|
||||
LIFE_PAID,
|
||||
CASCADE_LAND,
|
||||
//permanent events
|
||||
ENTERS_THE_BATTLEFIELD_SELF, /* 616.1a If any of the replacement and/or prevention effects are self-replacement effects (see rule 614.15),
|
||||
one of them must be chosen. If not, proceed to rule 616.1b. */
|
||||
|
|
Loading…
Reference in a new issue