mirror of
https://github.com/correl/mage.git
synced 2024-11-25 03:00:11 +00:00
[AFR] Implementing dungeon mechanic (ready for review) (#7937)
* added dungeon and dungeon room class * [AFR] Implemented Tomb of Annihilation * [AFR] Implemented Shortcut Seeker * [AFR] Implemented Gloom Stalker * [AFR] Implemented Nadaar, Selfless Paladin * added room triggers * added more venturing code, currently untested * fixed error * moved venture into dungeon from player class to game class * removed unnecessary sourceobject from dungeon * fixed npe error * added dungeon completion * fixed concurrent modification exception * added logging * added proper copy methods * added views * updated room text generation * added some missing code * finished implementing CompletedDungeonCondition * [AFR] Implemented Ellywick Tumblestrum * [AFR] Implemented Lost Mine of Phandelver * added choice dialog for dungeons * [AFR] Implemented Dungeon of the Mad Mage * small text fix * added initial dungeon test * [AFR] Implemented Cloister Gargoyle * [AFR] Implemented Dungeon Crawler * small text change for dungeon rooms * added more tests * some simplification to dungeon props * updated testing helper functions * added currently failing test for venturing on separate steps and turns * added tests for dungeon completion * fixed missing trigger visual and dungeons not persisting through turns * some text updates * added rollback test * added a test for multiple dungeons at once * added one more condition test
This commit is contained in:
parent
c6d08ce344
commit
bb591dd038
42 changed files with 2481 additions and 144 deletions
|
@ -1,7 +1,10 @@
|
||||||
package mage.client.deckeditor.collection.viewer;
|
package mage.client.deckeditor.collection.viewer;
|
||||||
|
|
||||||
import mage.abilities.icon.CardIconRenderSettings;
|
import mage.abilities.icon.CardIconRenderSettings;
|
||||||
import mage.cards.*;
|
import mage.cards.CardDimensions;
|
||||||
|
import mage.cards.ExpansionSet;
|
||||||
|
import mage.cards.MageCard;
|
||||||
|
import mage.cards.Sets;
|
||||||
import mage.cards.repository.CardCriteria;
|
import mage.cards.repository.CardCriteria;
|
||||||
import mage.cards.repository.CardInfo;
|
import mage.cards.repository.CardInfo;
|
||||||
import mage.cards.repository.CardRepository;
|
import mage.cards.repository.CardRepository;
|
||||||
|
@ -18,15 +21,13 @@ import mage.client.util.audio.AudioManager;
|
||||||
import mage.client.util.sets.ConstructedFormats;
|
import mage.client.util.sets.ConstructedFormats;
|
||||||
import mage.components.ImagePanel;
|
import mage.components.ImagePanel;
|
||||||
import mage.components.ImagePanelStyle;
|
import mage.components.ImagePanelStyle;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
import mage.game.command.Emblem;
|
import mage.game.command.Emblem;
|
||||||
import mage.game.command.Plane;
|
import mage.game.command.Plane;
|
||||||
import mage.game.draft.RateCard;
|
import mage.game.draft.RateCard;
|
||||||
import mage.game.permanent.PermanentToken;
|
import mage.game.permanent.PermanentToken;
|
||||||
import mage.game.permanent.token.Token;
|
import mage.game.permanent.token.Token;
|
||||||
import mage.view.CardView;
|
import mage.view.*;
|
||||||
import mage.view.EmblemView;
|
|
||||||
import mage.view.PermanentView;
|
|
||||||
import mage.view.PlaneView;
|
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
import org.mage.card.arcane.ManaSymbols;
|
import org.mage.card.arcane.ManaSymbols;
|
||||||
import org.mage.plugins.card.images.CardDownloadData;
|
import org.mage.plugins.card.images.CardDownloadData;
|
||||||
|
@ -365,6 +366,8 @@ public class MageBook extends JComponent {
|
||||||
addToken((Token) item, bigCard, null, position);
|
addToken((Token) item, bigCard, null, position);
|
||||||
} else if (item instanceof Emblem) {
|
} else if (item instanceof Emblem) {
|
||||||
addEmblem((Emblem) item, bigCard, null, position);
|
addEmblem((Emblem) item, bigCard, null, position);
|
||||||
|
} else if (item instanceof Dungeon) {
|
||||||
|
addDungeon((Dungeon) item, bigCard, null, position);
|
||||||
} else if (item instanceof Plane) {
|
} else if (item instanceof Plane) {
|
||||||
addPlane((Plane) item, bigCard, null, position);
|
addPlane((Plane) item, bigCard, null, position);
|
||||||
} else {
|
} else {
|
||||||
|
@ -430,6 +433,11 @@ public class MageBook extends JComponent {
|
||||||
addCard(cardView, bigCard, gameId, rectangle, false);
|
addCard(cardView, bigCard, gameId, rectangle, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private void addDungeon(Dungeon dungeon, BigCard bigCard, UUID gameId, Rectangle rectangle) {
|
||||||
|
CardView cardView = new CardView(new DungeonView(dungeon));
|
||||||
|
addCard(cardView, bigCard, gameId, rectangle, false);
|
||||||
|
}
|
||||||
|
|
||||||
private void addPlane(Plane plane, BigCard bigCard, UUID gameId, Rectangle rectangle) {
|
private void addPlane(Plane plane, BigCard bigCard, UUID gameId, Rectangle rectangle) {
|
||||||
CardView cardView = new CardView(new PlaneView(plane));
|
CardView cardView = new CardView(new PlaneView(plane));
|
||||||
addCard(cardView, bigCard, gameId, rectangle, false);
|
addCard(cardView, bigCard, gameId, rectangle, false);
|
||||||
|
|
|
@ -3,9 +3,9 @@ package mage.client.dialog;
|
||||||
import mage.abilities.Ability;
|
import mage.abilities.Ability;
|
||||||
import mage.abilities.common.SimpleStaticAbility;
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
import mage.abilities.icon.CardIconImpl;
|
import mage.abilities.icon.CardIconImpl;
|
||||||
import mage.abilities.icon.CardIconType;
|
|
||||||
import mage.abilities.icon.CardIconOrder;
|
import mage.abilities.icon.CardIconOrder;
|
||||||
import mage.abilities.icon.CardIconPosition;
|
import mage.abilities.icon.CardIconPosition;
|
||||||
|
import mage.abilities.icon.CardIconType;
|
||||||
import mage.cards.*;
|
import mage.cards.*;
|
||||||
import mage.cards.decks.Deck;
|
import mage.cards.decks.Deck;
|
||||||
import mage.cards.repository.CardInfo;
|
import mage.cards.repository.CardInfo;
|
||||||
|
@ -15,13 +15,16 @@ import mage.cards.repository.ExpansionRepository;
|
||||||
import mage.client.MageFrame;
|
import mage.client.MageFrame;
|
||||||
import mage.client.cards.BigCard;
|
import mage.client.cards.BigCard;
|
||||||
import mage.client.themes.ThemeType;
|
import mage.client.themes.ThemeType;
|
||||||
import mage.client.util.*;
|
import mage.client.util.ClientEventType;
|
||||||
import mage.client.util.Event;
|
import mage.client.util.Event;
|
||||||
|
import mage.client.util.GUISizeHelper;
|
||||||
|
import mage.client.util.Listener;
|
||||||
import mage.constants.MultiplayerAttackOption;
|
import mage.constants.MultiplayerAttackOption;
|
||||||
import mage.constants.RangeOfInfluence;
|
import mage.constants.RangeOfInfluence;
|
||||||
import mage.constants.Zone;
|
import mage.constants.Zone;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
import mage.game.GameImpl;
|
import mage.game.GameImpl;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
import mage.game.command.Emblem;
|
import mage.game.command.Emblem;
|
||||||
import mage.game.command.Plane;
|
import mage.game.command.Plane;
|
||||||
import mage.game.match.MatchType;
|
import mage.game.match.MatchType;
|
||||||
|
@ -40,8 +43,8 @@ import org.mage.card.arcane.CardPanel;
|
||||||
import javax.swing.*;
|
import javax.swing.*;
|
||||||
import java.awt.*;
|
import java.awt.*;
|
||||||
import java.awt.event.KeyEvent;
|
import java.awt.event.KeyEvent;
|
||||||
import java.util.*;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* App GUI: debug only, testing card renders and manipulations
|
* App GUI: debug only, testing card renders and manipulations
|
||||||
|
@ -180,6 +183,12 @@ public class TestCardRenderDialog extends MageDialog {
|
||||||
return emblemView;
|
return emblemView;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private AbilityView createDungeon(Dungeon dungeon) {
|
||||||
|
AbilityView emblemView = new AbilityView(dungeon.getAbilities().get(0), dungeon.getName(), new CardView(new DungeonView(dungeon)));
|
||||||
|
emblemView.setName(dungeon.getName());
|
||||||
|
return emblemView;
|
||||||
|
}
|
||||||
|
|
||||||
private AbilityView createPlane(Plane plane) {
|
private AbilityView createPlane(Plane plane) {
|
||||||
AbilityView planeView = new AbilityView(plane.getAbilities().get(0), plane.getName(), new CardView(new PlaneView(plane)));
|
AbilityView planeView = new AbilityView(plane.getAbilities().get(0), plane.getName(), new CardView(new PlaneView(plane)));
|
||||||
planeView.setName(plane.getName());
|
planeView.setName(plane.getName());
|
||||||
|
@ -224,7 +233,7 @@ public class TestCardRenderDialog extends MageDialog {
|
||||||
// init card listener for clicks, menu and other events
|
// init card listener for clicks, menu and other events
|
||||||
if (this.cardListener == null) {
|
if (this.cardListener == null) {
|
||||||
this.cardListener = event -> {
|
this.cardListener = event -> {
|
||||||
switch(event.getEventType()) {
|
switch (event.getEventType()) {
|
||||||
case CARD_CLICK:
|
case CARD_CLICK:
|
||||||
case CARD_DOUBLE_CLICK:
|
case CARD_DOUBLE_CLICK:
|
||||||
handleCardClick(event);
|
handleCardClick(event);
|
||||||
|
@ -408,7 +417,7 @@ public class TestCardRenderDialog extends MageDialog {
|
||||||
|
|
||||||
labelRenderMode.setText("Render mode:");
|
labelRenderMode.setText("Render mode:");
|
||||||
|
|
||||||
comboRenderMode.setModel(new javax.swing.DefaultComboBoxModel<>(new String[] { "MTGO", "Image" }));
|
comboRenderMode.setModel(new javax.swing.DefaultComboBoxModel<>(new String[]{"MTGO", "Image"}));
|
||||||
comboRenderMode.addItemListener(new java.awt.event.ItemListener() {
|
comboRenderMode.addItemListener(new java.awt.event.ItemListener() {
|
||||||
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
||||||
comboRenderModeItemStateChanged(evt);
|
comboRenderModeItemStateChanged(evt);
|
||||||
|
@ -432,7 +441,7 @@ public class TestCardRenderDialog extends MageDialog {
|
||||||
|
|
||||||
labelCardIconsPosition.setText("Card icons position:");
|
labelCardIconsPosition.setText("Card icons position:");
|
||||||
|
|
||||||
comboCardIconsPosition.setModel(new javax.swing.DefaultComboBoxModel<>(new String[] { "TOP", "LEFT", "RIGHT", "BOTTOM", "CORNER_TOP_LEFT", "CORNER_TOP_RIGHT", "CORNER_BOTTOM_LEFT", "CORNER_BOTTOM_RIGHT" }));
|
comboCardIconsPosition.setModel(new javax.swing.DefaultComboBoxModel<>(new String[]{"TOP", "LEFT", "RIGHT", "BOTTOM", "CORNER_TOP_LEFT", "CORNER_TOP_RIGHT", "CORNER_BOTTOM_LEFT", "CORNER_BOTTOM_RIGHT"}));
|
||||||
comboCardIconsPosition.setSelectedIndex(1);
|
comboCardIconsPosition.setSelectedIndex(1);
|
||||||
comboCardIconsPosition.addItemListener(new java.awt.event.ItemListener() {
|
comboCardIconsPosition.addItemListener(new java.awt.event.ItemListener() {
|
||||||
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
||||||
|
@ -460,7 +469,7 @@ public class TestCardRenderDialog extends MageDialog {
|
||||||
|
|
||||||
labelCardIconsOrder.setText("Order:");
|
labelCardIconsOrder.setText("Order:");
|
||||||
|
|
||||||
comboCardIconsOrder.setModel(new javax.swing.DefaultComboBoxModel<>(new String[] { "START", "CENTER", "END" }));
|
comboCardIconsOrder.setModel(new javax.swing.DefaultComboBoxModel<>(new String[]{"START", "CENTER", "END"}));
|
||||||
comboCardIconsOrder.setSelectedIndex(2);
|
comboCardIconsOrder.setSelectedIndex(2);
|
||||||
comboCardIconsOrder.addItemListener(new java.awt.event.ItemListener() {
|
comboCardIconsOrder.addItemListener(new java.awt.event.ItemListener() {
|
||||||
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
||||||
|
@ -505,7 +514,7 @@ public class TestCardRenderDialog extends MageDialog {
|
||||||
|
|
||||||
labelTheme.setText("Theme:");
|
labelTheme.setText("Theme:");
|
||||||
|
|
||||||
comboTheme.setModel(new javax.swing.DefaultComboBoxModel<>(new String[] { "loading..." }));
|
comboTheme.setModel(new javax.swing.DefaultComboBoxModel<>(new String[]{"loading..."}));
|
||||||
comboTheme.setToolTipText("WARNING, selected theme will be applied to full app, not render dialog only");
|
comboTheme.setToolTipText("WARNING, selected theme will be applied to full app, not render dialog only");
|
||||||
comboTheme.addItemListener(new java.awt.event.ItemListener() {
|
comboTheme.addItemListener(new java.awt.event.ItemListener() {
|
||||||
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
public void itemStateChanged(java.awt.event.ItemEvent evt) {
|
||||||
|
|
|
@ -48,19 +48,21 @@ public final class CardsViewUtil {
|
||||||
|
|
||||||
public static CardsView convertCommandObject(List<CommandObjectView> view) {
|
public static CardsView convertCommandObject(List<CommandObjectView> view) {
|
||||||
CardsView cards = new CardsView();
|
CardsView cards = new CardsView();
|
||||||
|
|
||||||
for (CommandObjectView commandObject : view) {
|
for (CommandObjectView commandObject : view) {
|
||||||
|
CardView cardView;
|
||||||
if (commandObject instanceof EmblemView) {
|
if (commandObject instanceof EmblemView) {
|
||||||
CardView cardView = new CardView((EmblemView) commandObject);
|
cardView = new CardView((EmblemView) commandObject);
|
||||||
cards.put(commandObject.getId(), cardView);
|
} else if (commandObject instanceof DungeonView) {
|
||||||
|
cardView = new CardView((DungeonView) commandObject);
|
||||||
} else if (commandObject instanceof PlaneView) {
|
} else if (commandObject instanceof PlaneView) {
|
||||||
CardView cardView = new CardView((PlaneView) commandObject);
|
cardView = new CardView((PlaneView) commandObject);
|
||||||
cards.put(commandObject.getId(), cardView);
|
|
||||||
} else if (commandObject instanceof CommanderView) {
|
} else if (commandObject instanceof CommanderView) {
|
||||||
cards.put(commandObject.getId(), (CommanderView) commandObject);
|
cardView = (CommanderView) commandObject;
|
||||||
|
} else {
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
cards.put(commandObject.getId(), cardView);
|
||||||
}
|
}
|
||||||
|
|
||||||
return cards;
|
return cards;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,7 @@ import mage.counters.CounterType;
|
||||||
import mage.designations.Designation;
|
import mage.designations.Designation;
|
||||||
import mage.filter.FilterMana;
|
import mage.filter.FilterMana;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
import mage.game.command.Emblem;
|
import mage.game.command.Emblem;
|
||||||
import mage.game.command.Plane;
|
import mage.game.command.Plane;
|
||||||
import mage.game.permanent.Permanent;
|
import mage.game.permanent.Permanent;
|
||||||
|
@ -579,6 +580,11 @@ public class CardView extends SimpleCardView {
|
||||||
Emblem emblem = (Emblem) object;
|
Emblem emblem = (Emblem) object;
|
||||||
this.rarity = Rarity.SPECIAL;
|
this.rarity = Rarity.SPECIAL;
|
||||||
this.rules = emblem.getAbilities().getRules(emblem.getName());
|
this.rules = emblem.getAbilities().getRules(emblem.getName());
|
||||||
|
} else if (object instanceof Dungeon) {
|
||||||
|
this.mageObjectType = MageObjectType.DUNGEON;
|
||||||
|
Dungeon dungeon = (Dungeon) object;
|
||||||
|
this.rarity = Rarity.SPECIAL;
|
||||||
|
this.rules = dungeon.getRules();
|
||||||
} else if (object instanceof Plane) {
|
} else if (object instanceof Plane) {
|
||||||
this.mageObjectType = MageObjectType.PLANE;
|
this.mageObjectType = MageObjectType.PLANE;
|
||||||
Plane plane = (Plane) object;
|
Plane plane = (Plane) object;
|
||||||
|
@ -631,6 +637,21 @@ public class CardView extends SimpleCardView {
|
||||||
this.rarity = Rarity.COMMON;
|
this.rarity = Rarity.COMMON;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public CardView(DungeonView dungeon) {
|
||||||
|
this(true);
|
||||||
|
this.gameObject = true;
|
||||||
|
this.id = dungeon.getId();
|
||||||
|
this.mageObjectType = MageObjectType.DUNGEON;
|
||||||
|
this.name = dungeon.getName();
|
||||||
|
this.displayName = name;
|
||||||
|
this.displayFullName = name;
|
||||||
|
this.rules = dungeon.getRules();
|
||||||
|
// emblem images are always with common (black) symbol
|
||||||
|
this.frameStyle = FrameStyle.M15_NORMAL;
|
||||||
|
this.expansionSetCode = dungeon.getExpansionSetCode();
|
||||||
|
this.rarity = Rarity.COMMON;
|
||||||
|
}
|
||||||
|
|
||||||
public CardView(PlaneView plane) {
|
public CardView(PlaneView plane) {
|
||||||
this(true);
|
this(true);
|
||||||
this.gameObject = true;
|
this.gameObject = true;
|
||||||
|
|
|
@ -6,6 +6,7 @@ import mage.abilities.effects.Effect;
|
||||||
import mage.cards.Card;
|
import mage.cards.Card;
|
||||||
import mage.constants.Zone;
|
import mage.constants.Zone;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
import mage.game.command.Emblem;
|
import mage.game.command.Emblem;
|
||||||
import mage.game.command.Plane;
|
import mage.game.command.Plane;
|
||||||
import mage.game.permanent.Permanent;
|
import mage.game.permanent.Permanent;
|
||||||
|
@ -110,6 +111,9 @@ public class CardsView extends LinkedHashMap<UUID, CardView> {
|
||||||
abilityView = new AbilityView(ability, sourceObject.getName(), new CardView(new EmblemView((Emblem) sourceObject)));
|
abilityView = new AbilityView(ability, sourceObject.getName(), new CardView(new EmblemView((Emblem) sourceObject)));
|
||||||
abilityView.setName(sourceObject.getName());
|
abilityView.setName(sourceObject.getName());
|
||||||
// abilityView.setExpansionSetCode(sourceCard.getExpansionSetCode());
|
// abilityView.setExpansionSetCode(sourceCard.getExpansionSetCode());
|
||||||
|
} else if (sourceObject instanceof Dungeon) {
|
||||||
|
abilityView = new AbilityView(ability, sourceObject.getName(), new CardView(new DungeonView((Dungeon) sourceObject)));
|
||||||
|
abilityView.setName(sourceObject.getName());
|
||||||
} else if (sourceObject instanceof Plane) {
|
} else if (sourceObject instanceof Plane) {
|
||||||
abilityView = new AbilityView(ability, sourceObject.getName(), new CardView(new PlaneView((Plane) sourceObject)));
|
abilityView = new AbilityView(ability, sourceObject.getName(), new CardView(new PlaneView((Plane) sourceObject)));
|
||||||
abilityView.setName(sourceObject.getName());
|
abilityView.setName(sourceObject.getName());
|
||||||
|
|
84
Mage.Common/src/main/java/mage/view/DungeonView.java
Normal file
84
Mage.Common/src/main/java/mage/view/DungeonView.java
Normal file
|
@ -0,0 +1,84 @@
|
||||||
|
package mage.view;
|
||||||
|
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
|
import mage.players.PlayableObjectStats;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class DungeonView implements CommandObjectView, Serializable {
|
||||||
|
|
||||||
|
protected UUID id;
|
||||||
|
protected String name;
|
||||||
|
protected String expansionSetCode;
|
||||||
|
protected List<String> rules;
|
||||||
|
protected PlayableObjectStats playableStats = new PlayableObjectStats();
|
||||||
|
|
||||||
|
public DungeonView(Dungeon dungeon) {
|
||||||
|
this.id = dungeon.getId();
|
||||||
|
this.name = dungeon.getName();
|
||||||
|
this.expansionSetCode = dungeon.getExpansionSetCodeForImage();
|
||||||
|
this.rules = dungeon.getRules();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getExpansionSetCode() {
|
||||||
|
return expansionSetCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UUID getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<String> getRules() {
|
||||||
|
return rules;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isPlayable() {
|
||||||
|
return this.playableStats.getPlayableAmount() > 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setPlayableStats(PlayableObjectStats playableStats) {
|
||||||
|
this.playableStats = playableStats;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public PlayableObjectStats getPlayableStats() {
|
||||||
|
return this.playableStats;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isChoosable() {
|
||||||
|
// unsupported
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setChoosable(boolean isChoosable) {
|
||||||
|
// unsupported
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isSelected() {
|
||||||
|
// unsupported
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setSelected(boolean isSelected) {
|
||||||
|
// unsupported
|
||||||
|
}
|
||||||
|
}
|
|
@ -13,6 +13,7 @@ import mage.game.ExileZone;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
import mage.game.GameState;
|
import mage.game.GameState;
|
||||||
import mage.game.combat.CombatGroup;
|
import mage.game.combat.CombatGroup;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
import mage.game.command.Emblem;
|
import mage.game.command.Emblem;
|
||||||
import mage.game.command.Plane;
|
import mage.game.command.Plane;
|
||||||
import mage.game.permanent.Permanent;
|
import mage.game.permanent.Permanent;
|
||||||
|
@ -114,6 +115,12 @@ public class GameView implements Serializable {
|
||||||
stack.put(stackObject.getId(),
|
stack.put(stackObject.getId(),
|
||||||
new StackAbilityView(game, (StackAbility) stackObject, object.getName(), cardView));
|
new StackAbilityView(game, (StackAbility) stackObject, object.getName(), cardView));
|
||||||
checkPaid(stackObject.getId(), ((StackAbility) stackObject));
|
checkPaid(stackObject.getId(), ((StackAbility) stackObject));
|
||||||
|
} else if (object instanceof Dungeon) {
|
||||||
|
CardView cardView = new CardView(new DungeonView((Dungeon) object));
|
||||||
|
stackObject.setName(object.getName());
|
||||||
|
stack.put(stackObject.getId(),
|
||||||
|
new StackAbilityView(game, (StackAbility) stackObject, object.getName(), cardView));
|
||||||
|
checkPaid(stackObject.getId(), ((StackAbility) stackObject));
|
||||||
} else if (object instanceof Plane) {
|
} else if (object instanceof Plane) {
|
||||||
CardView cardView = new CardView(new PlaneView((Plane) object));
|
CardView cardView = new CardView(new PlaneView((Plane) object));
|
||||||
stackObject.setName(object.getName());
|
stackObject.setName(object.getName());
|
||||||
|
|
|
@ -6,10 +6,7 @@ import mage.designations.Designation;
|
||||||
import mage.game.ExileZone;
|
import mage.game.ExileZone;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
import mage.game.GameState;
|
import mage.game.GameState;
|
||||||
import mage.game.command.CommandObject;
|
import mage.game.command.*;
|
||||||
import mage.game.command.Commander;
|
|
||||||
import mage.game.command.Emblem;
|
|
||||||
import mage.game.command.Plane;
|
|
||||||
import mage.game.permanent.Permanent;
|
import mage.game.permanent.Permanent;
|
||||||
import mage.players.Player;
|
import mage.players.Player;
|
||||||
import mage.players.net.UserData;
|
import mage.players.net.UserData;
|
||||||
|
@ -113,6 +110,11 @@ public class PlayerView implements Serializable {
|
||||||
if (emblem.getControllerId().equals(this.playerId)) {
|
if (emblem.getControllerId().equals(this.playerId)) {
|
||||||
commandList.add(new EmblemView(emblem));
|
commandList.add(new EmblemView(emblem));
|
||||||
}
|
}
|
||||||
|
} else if (commandObject instanceof Dungeon) {
|
||||||
|
Dungeon dungeon = (Dungeon) commandObject;
|
||||||
|
if (dungeon.getControllerId().equals(this.playerId)) {
|
||||||
|
commandList.add(new DungeonView(dungeon));
|
||||||
|
}
|
||||||
} else if (commandObject instanceof Plane) {
|
} else if (commandObject instanceof Plane) {
|
||||||
Plane plane = (Plane) commandObject;
|
Plane plane = (Plane) commandObject;
|
||||||
// Planes are universal and all players can see them.
|
// Planes are universal and all players can see them.
|
||||||
|
|
56
Mage.Sets/src/mage/cards/c/CloisterGargoyle.java
Normal file
56
Mage.Sets/src/mage/cards/c/CloisterGargoyle.java
Normal file
|
@ -0,0 +1,56 @@
|
||||||
|
package mage.cards.c;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.common.EntersBattlefieldTriggeredAbility;
|
||||||
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
|
import mage.abilities.condition.common.CompletedDungeonCondition;
|
||||||
|
import mage.abilities.decorator.ConditionalContinuousEffect;
|
||||||
|
import mage.abilities.effects.common.continuous.BoostSourceEffect;
|
||||||
|
import mage.abilities.effects.common.continuous.GainAbilitySourceEffect;
|
||||||
|
import mage.abilities.effects.keyword.VentureIntoTheDungeonEffect;
|
||||||
|
import mage.abilities.keyword.FlyingAbility;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.watchers.common.CompletedDungeonWatcher;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class CloisterGargoyle extends CardImpl {
|
||||||
|
|
||||||
|
public CloisterGargoyle(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.ARTIFACT, CardType.CREATURE}, "{2}{W}");
|
||||||
|
|
||||||
|
this.subtype.add(SubType.GARGOYLE);
|
||||||
|
this.power = new MageInt(0);
|
||||||
|
this.toughness = new MageInt(4);
|
||||||
|
|
||||||
|
// When Cloister Gargoyle enters the battlefield, venture into the dungeon.
|
||||||
|
this.addAbility(new EntersBattlefieldTriggeredAbility(new VentureIntoTheDungeonEffect()));
|
||||||
|
|
||||||
|
// As long as you've completed a dungeon, Cloister Gargoyle gets +3/+0 and has flying.
|
||||||
|
Ability ability = new SimpleStaticAbility(new ConditionalContinuousEffect(
|
||||||
|
new BoostSourceEffect(3, 0, Duration.WhileOnBattlefield),
|
||||||
|
CompletedDungeonCondition.instance, "as long as you've completed a dungeon, {this} gets +3/+0"
|
||||||
|
));
|
||||||
|
ability.addEffect(new ConditionalContinuousEffect(new GainAbilitySourceEffect(
|
||||||
|
FlyingAbility.getInstance(), Duration.WhileOnBattlefield
|
||||||
|
), CompletedDungeonCondition.instance, "and has flying"));
|
||||||
|
this.addAbility(ability.addHint(CompletedDungeonCondition.getHint()), new CompletedDungeonWatcher());
|
||||||
|
}
|
||||||
|
|
||||||
|
private CloisterGargoyle(final CloisterGargoyle card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public CloisterGargoyle copy() {
|
||||||
|
return new CloisterGargoyle(this);
|
||||||
|
}
|
||||||
|
}
|
44
Mage.Sets/src/mage/cards/d/DungeonCrawler.java
Normal file
44
Mage.Sets/src/mage/cards/d/DungeonCrawler.java
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
package mage.cards.d;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.common.CompletedDungeonTriggeredAbility;
|
||||||
|
import mage.abilities.common.EntersBattlefieldTappedAbility;
|
||||||
|
import mage.abilities.effects.common.ReturnSourceFromGraveyardToHandEffect;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.constants.Zone;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class DungeonCrawler extends CardImpl {
|
||||||
|
|
||||||
|
public DungeonCrawler(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{B}");
|
||||||
|
|
||||||
|
this.subtype.add(SubType.ZOMBIE);
|
||||||
|
this.power = new MageInt(2);
|
||||||
|
this.toughness = new MageInt(1);
|
||||||
|
|
||||||
|
// Dungeon Crawler enters the battlefield tapped.
|
||||||
|
this.addAbility(new EntersBattlefieldTappedAbility());
|
||||||
|
|
||||||
|
// Whenever you complete a dungeon, you may return Dungeon Crawler from your graveyard to your hand.
|
||||||
|
this.addAbility(new CompletedDungeonTriggeredAbility(
|
||||||
|
Zone.GRAVEYARD, new ReturnSourceFromGraveyardToHandEffect(), true
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
private DungeonCrawler(final DungeonCrawler card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public DungeonCrawler copy() {
|
||||||
|
return new DungeonCrawler(this);
|
||||||
|
}
|
||||||
|
}
|
93
Mage.Sets/src/mage/cards/e/EllywickTumblestrum.java
Normal file
93
Mage.Sets/src/mage/cards/e/EllywickTumblestrum.java
Normal file
|
@ -0,0 +1,93 @@
|
||||||
|
package mage.cards.e;
|
||||||
|
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.LoyaltyAbility;
|
||||||
|
import mage.abilities.common.PlaneswalkerEntersWithLoyaltyCountersAbility;
|
||||||
|
import mage.abilities.effects.OneShotEffect;
|
||||||
|
import mage.abilities.effects.common.GetEmblemEffect;
|
||||||
|
import mage.abilities.effects.keyword.VentureIntoTheDungeonEffect;
|
||||||
|
import mage.cards.*;
|
||||||
|
import mage.constants.*;
|
||||||
|
import mage.filter.StaticFilters;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.command.emblems.EllywickTumblestrumEmblem;
|
||||||
|
import mage.players.Player;
|
||||||
|
import mage.target.common.TargetCardInLibrary;
|
||||||
|
import mage.watchers.common.CompletedDungeonWatcher;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class EllywickTumblestrum extends CardImpl {
|
||||||
|
|
||||||
|
public EllywickTumblestrum(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.PLANESWALKER}, "{2}{G}{G}");
|
||||||
|
|
||||||
|
this.addSuperType(SuperType.LEGENDARY);
|
||||||
|
this.subtype.add(SubType.ELLYWICK);
|
||||||
|
this.addAbility(new PlaneswalkerEntersWithLoyaltyCountersAbility(4));
|
||||||
|
|
||||||
|
// +1: Venture into the dungeon.
|
||||||
|
this.addAbility(new LoyaltyAbility(new VentureIntoTheDungeonEffect(), 1));
|
||||||
|
|
||||||
|
// −2: Look at the top six cards of your library. You may reveal a creature card from among them and put it into your hand. If it's legendary, you gain 3 life. Put the rest on the bottom of your library in a random order.
|
||||||
|
this.addAbility(new LoyaltyAbility(new EllywickTumblestrumEffect(), -2));
|
||||||
|
|
||||||
|
// −7: You get an emblem with "Creatures you control have trample and haste and get +2/+2 for each differently named dungeon you've completed."
|
||||||
|
this.addAbility(new LoyaltyAbility(
|
||||||
|
new GetEmblemEffect(new EllywickTumblestrumEmblem()), -7
|
||||||
|
), new CompletedDungeonWatcher());
|
||||||
|
}
|
||||||
|
|
||||||
|
private EllywickTumblestrum(final EllywickTumblestrum card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EllywickTumblestrum copy() {
|
||||||
|
return new EllywickTumblestrum(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class EllywickTumblestrumEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
EllywickTumblestrumEffect() {
|
||||||
|
super(Outcome.Benefit);
|
||||||
|
staticText = "look at the top six cards of your library. You may reveal a creature card " +
|
||||||
|
"from among them and put it into your hand. If it's legendary, you gain 3 life. " +
|
||||||
|
"Put the rest on the bottom of your library in a random order";
|
||||||
|
}
|
||||||
|
|
||||||
|
private EllywickTumblestrumEffect(final EllywickTumblestrumEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EllywickTumblestrumEffect copy() {
|
||||||
|
return new EllywickTumblestrumEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Player player = game.getPlayer(source.getControllerId());
|
||||||
|
if (player == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Cards cards = new CardsImpl(player.getLibrary().getTopCards(game, 6));
|
||||||
|
TargetCardInLibrary target = new TargetCardInLibrary(0, 1, StaticFilters.FILTER_CARD_CREATURE);
|
||||||
|
player.choose(outcome, cards, target, game);
|
||||||
|
Card card = cards.get(target.getFirstTarget(), game);
|
||||||
|
if (card != null) {
|
||||||
|
player.revealCards(source, new CardsImpl(card), game);
|
||||||
|
player.moveCards(card, Zone.HAND, source, game);
|
||||||
|
cards.remove(card);
|
||||||
|
if (card.isLegendary()) {
|
||||||
|
player.gainLife(3, game, source);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
player.putCardsOnBottomOfLibrary(cards, game, source, false);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
48
Mage.Sets/src/mage/cards/g/GloomStalker.java
Normal file
48
Mage.Sets/src/mage/cards/g/GloomStalker.java
Normal file
|
@ -0,0 +1,48 @@
|
||||||
|
package mage.cards.g;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
|
import mage.abilities.condition.common.CompletedDungeonCondition;
|
||||||
|
import mage.abilities.decorator.ConditionalContinuousEffect;
|
||||||
|
import mage.abilities.effects.common.continuous.GainAbilitySourceEffect;
|
||||||
|
import mage.abilities.keyword.DoubleStrikeAbility;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.watchers.common.CompletedDungeonWatcher;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class GloomStalker extends CardImpl {
|
||||||
|
|
||||||
|
public GloomStalker(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{2}{W}");
|
||||||
|
|
||||||
|
this.subtype.add(SubType.DWARF);
|
||||||
|
this.subtype.add(SubType.RANGER);
|
||||||
|
this.power = new MageInt(2);
|
||||||
|
this.toughness = new MageInt(3);
|
||||||
|
|
||||||
|
// As long as you've completed a dungeon, Gloom Stalker has double strike.
|
||||||
|
this.addAbility(new SimpleStaticAbility(new ConditionalContinuousEffect(
|
||||||
|
new GainAbilitySourceEffect(
|
||||||
|
DoubleStrikeAbility.getInstance(), Duration.WhileOnBattlefield
|
||||||
|
), CompletedDungeonCondition.instance,
|
||||||
|
"as long as you've completed a dungeon, {this} has double strike"
|
||||||
|
)).addHint(CompletedDungeonCondition.getHint()), new CompletedDungeonWatcher());
|
||||||
|
}
|
||||||
|
|
||||||
|
private GloomStalker(final GloomStalker card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public GloomStalker copy() {
|
||||||
|
return new GloomStalker(this);
|
||||||
|
}
|
||||||
|
}
|
58
Mage.Sets/src/mage/cards/n/NadaarSelflessPaladin.java
Normal file
58
Mage.Sets/src/mage/cards/n/NadaarSelflessPaladin.java
Normal file
|
@ -0,0 +1,58 @@
|
||||||
|
package mage.cards.n;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.common.EntersBattlefieldOrAttacksSourceTriggeredAbility;
|
||||||
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
|
import mage.abilities.condition.common.CompletedDungeonCondition;
|
||||||
|
import mage.abilities.decorator.ConditionalContinuousEffect;
|
||||||
|
import mage.abilities.effects.common.continuous.BoostControlledEffect;
|
||||||
|
import mage.abilities.effects.keyword.VentureIntoTheDungeonEffect;
|
||||||
|
import mage.abilities.keyword.VigilanceAbility;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.constants.SuperType;
|
||||||
|
import mage.watchers.common.CompletedDungeonWatcher;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class NadaarSelflessPaladin extends CardImpl {
|
||||||
|
|
||||||
|
public NadaarSelflessPaladin(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{2}{W}");
|
||||||
|
|
||||||
|
this.addSuperType(SuperType.LEGENDARY);
|
||||||
|
this.subtype.add(SubType.DRAGON);
|
||||||
|
this.subtype.add(SubType.KNIGHT);
|
||||||
|
this.power = new MageInt(3);
|
||||||
|
this.toughness = new MageInt(3);
|
||||||
|
|
||||||
|
// Vigilance
|
||||||
|
this.addAbility(VigilanceAbility.getInstance());
|
||||||
|
|
||||||
|
// Whenever Nadaar, Selfless Paladin enters the battlefield or attacks, venture into the dungeon.
|
||||||
|
this.addAbility(new EntersBattlefieldOrAttacksSourceTriggeredAbility(new VentureIntoTheDungeonEffect()));
|
||||||
|
|
||||||
|
// Other creatures you control get +1/+1 as long as you've completed a dungeon.
|
||||||
|
this.addAbility(new SimpleStaticAbility(new ConditionalContinuousEffect(
|
||||||
|
new BoostControlledEffect(
|
||||||
|
1, 1, Duration.WhileOnBattlefield, true
|
||||||
|
), CompletedDungeonCondition.instance,
|
||||||
|
"other creatures you control get +1/+1 as long as you've completed a dungeon"
|
||||||
|
)).addHint(CompletedDungeonCondition.getHint()), new CompletedDungeonWatcher());
|
||||||
|
}
|
||||||
|
|
||||||
|
private NadaarSelflessPaladin(final NadaarSelflessPaladin card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public NadaarSelflessPaladin copy() {
|
||||||
|
return new NadaarSelflessPaladin(this);
|
||||||
|
}
|
||||||
|
}
|
40
Mage.Sets/src/mage/cards/s/ShortcutSeeker.java
Normal file
40
Mage.Sets/src/mage/cards/s/ShortcutSeeker.java
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
package mage.cards.s;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.common.DealsCombatDamageToAPlayerTriggeredAbility;
|
||||||
|
import mage.abilities.effects.keyword.VentureIntoTheDungeonEffect;
|
||||||
|
import mage.cards.CardImpl;
|
||||||
|
import mage.cards.CardSetInfo;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class ShortcutSeeker extends CardImpl {
|
||||||
|
|
||||||
|
public ShortcutSeeker(UUID ownerId, CardSetInfo setInfo) {
|
||||||
|
super(ownerId, setInfo, new CardType[]{CardType.CREATURE}, "{3}{U}");
|
||||||
|
|
||||||
|
this.subtype.add(SubType.HUMAN);
|
||||||
|
this.subtype.add(SubType.ROGUE);
|
||||||
|
this.power = new MageInt(2);
|
||||||
|
this.toughness = new MageInt(5);
|
||||||
|
|
||||||
|
// Whenever Shortcut Seeker deals combat damage to a player, venture into the dungeon.
|
||||||
|
this.addAbility(new DealsCombatDamageToAPlayerTriggeredAbility(
|
||||||
|
new VentureIntoTheDungeonEffect(), false
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
private ShortcutSeeker(final ShortcutSeeker card) {
|
||||||
|
super(card);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ShortcutSeeker copy() {
|
||||||
|
return new ShortcutSeeker(this);
|
||||||
|
}
|
||||||
|
}
|
|
@ -26,17 +26,23 @@ public final class AdventuresInTheForgottenRealms extends ExpansionSet {
|
||||||
this.maxCardNumberInBooster = 275;
|
this.maxCardNumberInBooster = 275;
|
||||||
|
|
||||||
cards.add(new SetCardInfo("Bruenor Battlehammer", 219, Rarity.UNCOMMON, mage.cards.b.BruenorBattlehammer.class));
|
cards.add(new SetCardInfo("Bruenor Battlehammer", 219, Rarity.UNCOMMON, mage.cards.b.BruenorBattlehammer.class));
|
||||||
|
cards.add(new SetCardInfo("Cloister Gargoyle", 7, Rarity.UNCOMMON, mage.cards.c.CloisterGargoyle.class));
|
||||||
cards.add(new SetCardInfo("Drizzt Do'Urden", 220, Rarity.RARE, mage.cards.d.DrizztDoUrden.class));
|
cards.add(new SetCardInfo("Drizzt Do'Urden", 220, Rarity.RARE, mage.cards.d.DrizztDoUrden.class));
|
||||||
|
cards.add(new SetCardInfo("Dungeon Crawler", 99, Rarity.UNCOMMON, mage.cards.d.DungeonCrawler.class));
|
||||||
|
cards.add(new SetCardInfo("Ellywick Tumblestrum", 181, Rarity.MYTHIC, mage.cards.e.EllywickTumblestrum.class));
|
||||||
cards.add(new SetCardInfo("Evolving Wilds", 353, Rarity.COMMON, mage.cards.e.EvolvingWilds.class));
|
cards.add(new SetCardInfo("Evolving Wilds", 353, Rarity.COMMON, mage.cards.e.EvolvingWilds.class));
|
||||||
cards.add(new SetCardInfo("Flumph", 15, Rarity.RARE, mage.cards.f.Flumph.class));
|
cards.add(new SetCardInfo("Flumph", 15, Rarity.RARE, mage.cards.f.Flumph.class));
|
||||||
cards.add(new SetCardInfo("Forest", 278, Rarity.LAND, mage.cards.basiclands.Forest.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Forest", 278, Rarity.LAND, mage.cards.basiclands.Forest.class, NON_FULL_USE_VARIOUS));
|
||||||
|
cards.add(new SetCardInfo("Gloom Stalker", 16, Rarity.COMMON, mage.cards.g.GloomStalker.class));
|
||||||
cards.add(new SetCardInfo("Island", 266, Rarity.LAND, mage.cards.basiclands.Island.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Island", 266, Rarity.LAND, mage.cards.basiclands.Island.class, NON_FULL_USE_VARIOUS));
|
||||||
cards.add(new SetCardInfo("Lolth, Spider Queen", 112, Rarity.MYTHIC, mage.cards.l.LolthSpiderQueen.class));
|
cards.add(new SetCardInfo("Lolth, Spider Queen", 112, Rarity.MYTHIC, mage.cards.l.LolthSpiderQueen.class));
|
||||||
cards.add(new SetCardInfo("Mountain", 274, Rarity.LAND, mage.cards.basiclands.Mountain.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Mountain", 274, Rarity.LAND, mage.cards.basiclands.Mountain.class, NON_FULL_USE_VARIOUS));
|
||||||
|
cards.add(new SetCardInfo("Nadaar, Selfless Paladin", 27, Rarity.RARE, mage.cards.n.NadaarSelflessPaladin.class));
|
||||||
cards.add(new SetCardInfo("Plains", 262, Rarity.LAND, mage.cards.basiclands.Plains.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Plains", 262, Rarity.LAND, mage.cards.basiclands.Plains.class, NON_FULL_USE_VARIOUS));
|
||||||
cards.add(new SetCardInfo("Portable Hole", 33, Rarity.UNCOMMON, mage.cards.p.PortableHole.class));
|
cards.add(new SetCardInfo("Portable Hole", 33, Rarity.UNCOMMON, mage.cards.p.PortableHole.class));
|
||||||
cards.add(new SetCardInfo("Power Word Kill", 114, Rarity.UNCOMMON, mage.cards.p.PowerWordKill.class));
|
cards.add(new SetCardInfo("Power Word Kill", 114, Rarity.UNCOMMON, mage.cards.p.PowerWordKill.class));
|
||||||
cards.add(new SetCardInfo("Prosperous Innkeeper", 200, Rarity.UNCOMMON, mage.cards.p.ProsperousInnkeeper.class));
|
cards.add(new SetCardInfo("Prosperous Innkeeper", 200, Rarity.UNCOMMON, mage.cards.p.ProsperousInnkeeper.class));
|
||||||
|
cards.add(new SetCardInfo("Shortcut Seeker", 73, Rarity.COMMON, mage.cards.s.ShortcutSeeker.class));
|
||||||
cards.add(new SetCardInfo("Swamp", 270, Rarity.LAND, mage.cards.basiclands.Swamp.class, NON_FULL_USE_VARIOUS));
|
cards.add(new SetCardInfo("Swamp", 270, Rarity.LAND, mage.cards.basiclands.Swamp.class, NON_FULL_USE_VARIOUS));
|
||||||
cards.add(new SetCardInfo("Tasha's Hideous Laughter", 78, Rarity.RARE, mage.cards.t.TashasHideousLaughter.class));
|
cards.add(new SetCardInfo("Tasha's Hideous Laughter", 78, Rarity.RARE, mage.cards.t.TashasHideousLaughter.class));
|
||||||
cards.add(new SetCardInfo("Tiamat", 235, Rarity.MYTHIC, mage.cards.t.Tiamat.class));
|
cards.add(new SetCardInfo("Tiamat", 235, Rarity.MYTHIC, mage.cards.t.Tiamat.class));
|
||||||
|
|
|
@ -0,0 +1,387 @@
|
||||||
|
package org.mage.test.cards.dungeons;
|
||||||
|
|
||||||
|
import mage.abilities.common.SimpleActivatedAbility;
|
||||||
|
import mage.abilities.costs.mana.GenericManaCost;
|
||||||
|
import mage.abilities.effects.keyword.VentureIntoTheDungeonEffect;
|
||||||
|
import mage.abilities.keyword.DoubleStrikeAbility;
|
||||||
|
import mage.constants.PhaseStep;
|
||||||
|
import mage.constants.Zone;
|
||||||
|
import mage.counters.CounterType;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
|
import org.junit.Assert;
|
||||||
|
import org.junit.Test;
|
||||||
|
import org.mage.test.player.TestPlayer;
|
||||||
|
import org.mage.test.serverside.base.CardTestPlayerBase;
|
||||||
|
|
||||||
|
import java.util.stream.Stream;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class DungeonTest extends CardTestPlayerBase {
|
||||||
|
|
||||||
|
private static final String TOMB_OF_ANNIHILATION = "Tomb of Annihilation";
|
||||||
|
private static final String LOST_MINE_OF_PHANDELVER = "Lost Mine of Phandelver";
|
||||||
|
private static final String DUNGEON_OF_THE_MAD_MAGE = "Dungeon of the Mad Mage";
|
||||||
|
private static final String FLAMESPEAKER_ADEPT = "Flamespeaker Adept";
|
||||||
|
private static final String GLOOM_STALKER = "Gloom Stalker";
|
||||||
|
private static final String DUNGEON_CRAWLER = "Dungeon Crawler";
|
||||||
|
private static final String SILVERCOAT_LION = "Silvercoat Lion";
|
||||||
|
|
||||||
|
private void makeTester() {
|
||||||
|
makeTester(playerA);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void makeTester(TestPlayer player) {
|
||||||
|
addCustomCardWithAbility(
|
||||||
|
"tester", player,
|
||||||
|
new SimpleActivatedAbility(new VentureIntoTheDungeonEffect(), new GenericManaCost(0))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
private Stream<Dungeon> makeStream(TestPlayer player) {
|
||||||
|
return currentGame
|
||||||
|
.getState()
|
||||||
|
.getCommand()
|
||||||
|
.stream()
|
||||||
|
.filter(Dungeon.class::isInstance)
|
||||||
|
.map(Dungeon.class::cast)
|
||||||
|
.filter(dungeon -> dungeon.isControlledBy(player.getId()));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Dungeon getCurrentDungeon(TestPlayer player) {
|
||||||
|
Assert.assertTrue(
|
||||||
|
"Players should not control more than one dungeon",
|
||||||
|
makeStream(player).count() < 2
|
||||||
|
);
|
||||||
|
return makeStream(player).findFirst().orElse(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertDungeonRoom(String dungeonName, String roomName) {
|
||||||
|
assertDungeonRoom(playerA, dungeonName, roomName);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void assertDungeonRoom(TestPlayer player, String dungeonName, String roomName) {
|
||||||
|
Dungeon dungeon = getCurrentDungeon(player);
|
||||||
|
if (dungeonName == null) {
|
||||||
|
Assert.assertNull("There should be no dungeon", dungeon);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Assert.assertNotNull("Dungeon should not be null", dungeon);
|
||||||
|
Assert.assertEquals("Dungeon should be " + dungeonName, dungeonName, dungeon.getName());
|
||||||
|
Assert.assertEquals(
|
||||||
|
"Current room is " + roomName,
|
||||||
|
roomName, dungeon.getCurrentRoom().getName()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_room1() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, FLAMESPEAKER_ADEPT);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPowerToughness(playerA, FLAMESPEAKER_ADEPT, 4, 3);
|
||||||
|
assertPermanentCount(playerA, "Goblin", 0);
|
||||||
|
assertDungeonRoom(LOST_MINE_OF_PHANDELVER, "Cave Entrance");
|
||||||
|
assertLife(playerA, 20);
|
||||||
|
assertLife(playerB, 20);
|
||||||
|
assertHandCount(playerA, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_room2() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, FLAMESPEAKER_ADEPT);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPowerToughness(playerA, FLAMESPEAKER_ADEPT, 4, 3);
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertDungeonRoom(LOST_MINE_OF_PHANDELVER, "Goblin Lair");
|
||||||
|
assertLife(playerA, 20);
|
||||||
|
assertLife(playerB, 20);
|
||||||
|
assertHandCount(playerA, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_room3() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, FLAMESPEAKER_ADEPT);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPowerToughness(playerA, FLAMESPEAKER_ADEPT, 4, 3);
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertDungeonRoom(LOST_MINE_OF_PHANDELVER, "Dark Pool");
|
||||||
|
assertLife(playerA, 20 + 1);
|
||||||
|
assertLife(playerB, 20 - 1);
|
||||||
|
assertHandCount(playerA, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_room4() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, FLAMESPEAKER_ADEPT);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPowerToughness(playerA, FLAMESPEAKER_ADEPT, 4, 3);
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
assertLife(playerA, 20 + 1);
|
||||||
|
assertLife(playerB, 20 - 1);
|
||||||
|
assertHandCount(playerA, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_multipleTurns() {
|
||||||
|
makeTester();
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
activateAbility(2, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
activateAbility(2, PhaseStep.POSTCOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
|
||||||
|
setStopAt(2, PhaseStep.END_TURN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
assertLife(playerA, 20 + 1);
|
||||||
|
assertLife(playerB, 20 - 1);
|
||||||
|
assertHandCount(playerA, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_rollback() {
|
||||||
|
makeTester();
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
activateAbility(2, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
activateAbility(2, PhaseStep.POSTCOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
|
||||||
|
rollbackTurns(2, PhaseStep.END_TURN, playerA, 0);
|
||||||
|
setStopAt(2, PhaseStep.END_TURN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertDungeonRoom(LOST_MINE_OF_PHANDELVER, "Goblin Lair");
|
||||||
|
assertLife(playerA, 20);
|
||||||
|
assertLife(playerB, 20);
|
||||||
|
assertHandCount(playerA, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__LostMineOfPhandelver_rollbackDifferentChoice() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, SILVERCOAT_LION);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
activateAbility(2, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
activateAbility(2, PhaseStep.POSTCOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
|
||||||
|
rollbackTurns(2, PhaseStep.END_TURN, playerA, 0);
|
||||||
|
|
||||||
|
rollbackAfterActionsStart();
|
||||||
|
activateAbility(2, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Storeroom
|
||||||
|
addTarget(playerA, SILVERCOAT_LION);
|
||||||
|
rollbackAfterActionsEnd();
|
||||||
|
|
||||||
|
setStopAt(2, PhaseStep.END_TURN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPowerToughness(playerA, SILVERCOAT_LION, 3, 3);
|
||||||
|
assertCounterCount(playerA, SILVERCOAT_LION, CounterType.P1P1, 1);
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertDungeonRoom(LOST_MINE_OF_PHANDELVER, "Storeroom");
|
||||||
|
assertLife(playerA, 20);
|
||||||
|
assertLife(playerB, 20);
|
||||||
|
assertHandCount(playerA, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__Dungeon_multiplePlayers() {
|
||||||
|
makeTester(playerA);
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, FLAMESPEAKER_ADEPT);
|
||||||
|
makeTester(playerB);
|
||||||
|
addCard(Zone.BATTLEFIELD, playerB, FLAMESPEAKER_ADEPT);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "{0}:");
|
||||||
|
setChoice(playerB, DUNGEON_OF_THE_MAD_MAGE);
|
||||||
|
waitStackResolved(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "{0}:");
|
||||||
|
waitStackResolved(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "{0}:");
|
||||||
|
setChoice(playerB, "Yes"); // Goblin Bazaar
|
||||||
|
waitStackResolved(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.POSTCOMBAT_MAIN, playerB, "{0}:");
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.END_TURN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertPowerToughness(playerA, FLAMESPEAKER_ADEPT, 4, 3);
|
||||||
|
assertPowerToughness(playerB, FLAMESPEAKER_ADEPT, 6, 3);
|
||||||
|
assertPermanentCount(playerA, "Goblin", 1);
|
||||||
|
assertPermanentCount(playerB, "Treasure", 1);
|
||||||
|
assertDungeonRoom(playerA, LOST_MINE_OF_PHANDELVER, "Dark Pool");
|
||||||
|
assertDungeonRoom(playerB, DUNGEON_OF_THE_MAD_MAGE, "Lost Level");
|
||||||
|
assertLife(playerA, 20 + 1);
|
||||||
|
assertLife(playerB, 20 - 1 + 1);
|
||||||
|
assertHandCount(playerA, 0);
|
||||||
|
assertHandCount(playerB, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__CompletedDungeonCondition_true() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, GLOOM_STALKER);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertAbility(playerA, GLOOM_STALKER, DoubleStrikeAbility.getInstance(), true);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__CompletedDungeonCondition_false() {
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, GLOOM_STALKER);
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertAbility(playerA, GLOOM_STALKER, DoubleStrikeAbility.getInstance(), false);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__CompletedDungeonCondition_falseThenTrue() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.BATTLEFIELD, playerA, GLOOM_STALKER);
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.UPKEEP);
|
||||||
|
execute();
|
||||||
|
assertAbility(playerA, GLOOM_STALKER, DoubleStrikeAbility.getInstance(), false);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertAbility(playerA, GLOOM_STALKER, DoubleStrikeAbility.getInstance(), true);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Test
|
||||||
|
public void test__CompletedDungeonTriggeredAbility() {
|
||||||
|
makeTester();
|
||||||
|
addCard(Zone.GRAVEYARD, playerA, DUNGEON_CRAWLER);
|
||||||
|
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, LOST_MINE_OF_PHANDELVER);
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // Goblin Lair
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "No"); // Dark Pool
|
||||||
|
waitStackResolved(1, PhaseStep.PRECOMBAT_MAIN);
|
||||||
|
activateAbility(1, PhaseStep.PRECOMBAT_MAIN, playerA, "{0}:");
|
||||||
|
setChoice(playerA, "Yes"); // return Dungeon Crawler
|
||||||
|
|
||||||
|
setStopAt(1, PhaseStep.POSTCOMBAT_MAIN);
|
||||||
|
execute();
|
||||||
|
assertAllCommandsUsed();
|
||||||
|
|
||||||
|
assertHandCount(playerA, DUNGEON_CRAWLER, 1);
|
||||||
|
assertDungeonRoom(null, null);
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,9 +1,5 @@
|
||||||
package mage.abilities;
|
package mage.abilities;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
|
||||||
import java.util.UUID;
|
|
||||||
import mage.MageIdentifier;
|
import mage.MageIdentifier;
|
||||||
import mage.MageObject;
|
import mage.MageObject;
|
||||||
import mage.abilities.costs.*;
|
import mage.abilities.costs.*;
|
||||||
|
@ -21,6 +17,7 @@ import mage.cards.Card;
|
||||||
import mage.cards.SplitCard;
|
import mage.cards.SplitCard;
|
||||||
import mage.constants.*;
|
import mage.constants.*;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
import mage.game.command.Emblem;
|
import mage.game.command.Emblem;
|
||||||
import mage.game.command.Plane;
|
import mage.game.command.Plane;
|
||||||
import mage.game.events.GameEvent;
|
import mage.game.events.GameEvent;
|
||||||
|
@ -37,6 +34,11 @@ import mage.util.ThreadLocalStringBuilder;
|
||||||
import mage.watchers.Watcher;
|
import mage.watchers.Watcher;
|
||||||
import org.apache.log4j.Logger;
|
import org.apache.log4j.Logger;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.Iterator;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @author BetaSteward_at_googlemail.com
|
* @author BetaSteward_at_googlemail.com
|
||||||
*/
|
*/
|
||||||
|
@ -963,7 +965,7 @@ public abstract class AbilityImpl implements Ability {
|
||||||
}
|
}
|
||||||
MageObject object = game.getObject(this.getSourceId());
|
MageObject object = game.getObject(this.getSourceId());
|
||||||
// emblem/planes are always actual
|
// emblem/planes are always actual
|
||||||
if (object instanceof Emblem || object instanceof Plane) {
|
if (object instanceof Emblem || object instanceof Dungeon || object instanceof Plane) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -13,9 +13,7 @@ import mage.abilities.mana.ManaOptions;
|
||||||
import mage.cards.Card;
|
import mage.cards.Card;
|
||||||
import mage.constants.*;
|
import mage.constants.*;
|
||||||
import mage.game.Game;
|
import mage.game.Game;
|
||||||
import mage.game.command.Commander;
|
import mage.game.command.CommandObject;
|
||||||
import mage.game.command.Emblem;
|
|
||||||
import mage.game.command.Plane;
|
|
||||||
import mage.game.permanent.Permanent;
|
import mage.game.permanent.Permanent;
|
||||||
import mage.util.CardUtil;
|
import mage.util.CardUtil;
|
||||||
|
|
||||||
|
@ -212,12 +210,8 @@ public abstract class ActivatedAbilityImpl extends AbilityImpl implements Activa
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
MageObject mageObject = game.getObject(this.sourceId);
|
MageObject mageObject = game.getObject(this.sourceId);
|
||||||
if (mageObject instanceof Emblem) {
|
if (mageObject instanceof CommandObject) {
|
||||||
return ((Emblem) mageObject).isControlledBy(playerId);
|
return ((CommandObject) mageObject).isControlledBy(playerId);
|
||||||
} else if (mageObject instanceof Plane) {
|
|
||||||
return ((Plane) mageObject).isControlledBy(playerId);
|
|
||||||
} else if (mageObject instanceof Commander) {
|
|
||||||
return ((Commander) mageObject).isControlledBy(playerId);
|
|
||||||
} else if (game.getState().getZone(this.sourceId) != Zone.BATTLEFIELD) {
|
} else if (game.getState().getZone(this.sourceId) != Zone.BATTLEFIELD) {
|
||||||
return ((Card) mageObject).isOwnedBy(playerId);
|
return ((Card) mageObject).isOwnedBy(playerId);
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,49 @@
|
||||||
|
package mage.abilities.common;
|
||||||
|
|
||||||
|
import mage.abilities.TriggeredAbilityImpl;
|
||||||
|
import mage.abilities.effects.Effect;
|
||||||
|
import mage.constants.Zone;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.events.GameEvent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class CompletedDungeonTriggeredAbility extends TriggeredAbilityImpl {
|
||||||
|
|
||||||
|
public CompletedDungeonTriggeredAbility(Effect effect) {
|
||||||
|
this(effect, false);
|
||||||
|
}
|
||||||
|
|
||||||
|
public CompletedDungeonTriggeredAbility(Effect effect, boolean optional) {
|
||||||
|
this(Zone.BATTLEFIELD, effect, optional);
|
||||||
|
}
|
||||||
|
|
||||||
|
public CompletedDungeonTriggeredAbility(Zone zone, Effect effect, boolean optional) {
|
||||||
|
super(zone, effect, optional);
|
||||||
|
}
|
||||||
|
|
||||||
|
private CompletedDungeonTriggeredAbility(final CompletedDungeonTriggeredAbility ability) {
|
||||||
|
super(ability);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean checkEventType(GameEvent event, Game game) {
|
||||||
|
return event.getType() == GameEvent.EventType.DUNGEON_COMPLETED;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean checkTrigger(GameEvent event, Game game) {
|
||||||
|
return isControlledBy(event.getPlayerId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public CompletedDungeonTriggeredAbility copy() {
|
||||||
|
return new CompletedDungeonTriggeredAbility(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getRule() {
|
||||||
|
return "Whenever you complete a dungeon, " + super.getRule();
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,30 @@
|
||||||
|
package mage.abilities.condition.common;
|
||||||
|
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.condition.Condition;
|
||||||
|
import mage.abilities.hint.ConditionHint;
|
||||||
|
import mage.abilities.hint.Hint;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.watchers.common.CompletedDungeonWatcher;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public enum CompletedDungeonCondition implements Condition {
|
||||||
|
instance;
|
||||||
|
private static final Hint hint = new ConditionHint(instance, "You've completed a dungeon");
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
return CompletedDungeonWatcher.checkPlayer(source.getControllerId(), game);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return "you've completed a dungeon";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Hint getHint() {
|
||||||
|
return hint;
|
||||||
|
}
|
||||||
|
}
|
|
@ -42,7 +42,9 @@ public class CantAttackTargetEffect extends RestrictionEffect {
|
||||||
}
|
}
|
||||||
String text = "target " + mode.getTargets().get(0).getTargetName() + " can't attack";
|
String text = "target " + mode.getTargets().get(0).getTargetName() + " can't attack";
|
||||||
if (this.duration == Duration.EndOfTurn) {
|
if (this.duration == Duration.EndOfTurn) {
|
||||||
text += " this turn";
|
return text + " this turn";
|
||||||
|
} else if (this.duration == Duration.UntilYourNextTurn) {
|
||||||
|
return text + " until your next turn";
|
||||||
}
|
}
|
||||||
return text;
|
return text;
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,32 @@
|
||||||
|
package mage.abilities.effects.keyword;
|
||||||
|
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.effects.OneShotEffect;
|
||||||
|
import mage.constants.Outcome;
|
||||||
|
import mage.game.Game;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class VentureIntoTheDungeonEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
public VentureIntoTheDungeonEffect() {
|
||||||
|
super(Outcome.Benefit);
|
||||||
|
staticText = "venture into the dungeon. <i>(Enter the first room or advance to the next room.)</i>";
|
||||||
|
}
|
||||||
|
|
||||||
|
private VentureIntoTheDungeonEffect(final VentureIntoTheDungeonEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public VentureIntoTheDungeonEffect copy() {
|
||||||
|
return new VentureIntoTheDungeonEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
game.ventureIntoDungeon(source.getControllerId());
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
|
@ -14,6 +14,7 @@ public enum CardType {
|
||||||
ARTIFACT("Artifact", true, true),
|
ARTIFACT("Artifact", true, true),
|
||||||
CONSPIRACY("Conspiracy", false, false),
|
CONSPIRACY("Conspiracy", false, false),
|
||||||
CREATURE("Creature", true, true),
|
CREATURE("Creature", true, true),
|
||||||
|
DUNGEON("Dungeon", false, false),
|
||||||
ENCHANTMENT("Enchantment", true, true),
|
ENCHANTMENT("Enchantment", true, true),
|
||||||
INSTANT("Instant", false, true),
|
INSTANT("Instant", false, true),
|
||||||
LAND("Land", true, true),
|
LAND("Land", true, true),
|
||||||
|
|
|
@ -53,6 +53,7 @@ public enum MageObjectType {
|
||||||
TOKEN("Token", true, true),
|
TOKEN("Token", true, true),
|
||||||
SPELL("Spell", false, true),
|
SPELL("Spell", false, true),
|
||||||
PERMANENT("Permanent", true, true),
|
PERMANENT("Permanent", true, true),
|
||||||
|
DUNGEON("Dungeon", false, false),
|
||||||
EMBLEM("Emblem", false, false),
|
EMBLEM("Emblem", false, false),
|
||||||
COMMANDER("Commander", false, false),
|
COMMANDER("Commander", false, false),
|
||||||
DESIGNATION("Designation", false, false),
|
DESIGNATION("Designation", false, false),
|
||||||
|
|
|
@ -409,6 +409,7 @@ public enum SubType {
|
||||||
DOMRI("Domri", SubTypeSet.PlaneswalkerType),
|
DOMRI("Domri", SubTypeSet.PlaneswalkerType),
|
||||||
DOOKU("Dooku", SubTypeSet.PlaneswalkerType, true), // Star Wars
|
DOOKU("Dooku", SubTypeSet.PlaneswalkerType, true), // Star Wars
|
||||||
DOVIN("Dovin", SubTypeSet.PlaneswalkerType),
|
DOVIN("Dovin", SubTypeSet.PlaneswalkerType),
|
||||||
|
ELLYWICK("Ellywick", SubTypeSet.PlaneswalkerType),
|
||||||
ELSPETH("Elspeth", SubTypeSet.PlaneswalkerType),
|
ELSPETH("Elspeth", SubTypeSet.PlaneswalkerType),
|
||||||
ESTRID("Estrid", SubTypeSet.PlaneswalkerType),
|
ESTRID("Estrid", SubTypeSet.PlaneswalkerType),
|
||||||
FREYALISE("Freyalise", SubTypeSet.PlaneswalkerType),
|
FREYALISE("Freyalise", SubTypeSet.PlaneswalkerType),
|
||||||
|
|
|
@ -19,10 +19,7 @@ import mage.choices.Choice;
|
||||||
import mage.constants.*;
|
import mage.constants.*;
|
||||||
import mage.counters.Counters;
|
import mage.counters.Counters;
|
||||||
import mage.game.combat.Combat;
|
import mage.game.combat.Combat;
|
||||||
import mage.game.command.CommandObject;
|
import mage.game.command.*;
|
||||||
import mage.game.command.Commander;
|
|
||||||
import mage.game.command.Emblem;
|
|
||||||
import mage.game.command.Plane;
|
|
||||||
import mage.game.events.GameEvent;
|
import mage.game.events.GameEvent;
|
||||||
import mage.game.events.Listener;
|
import mage.game.events.Listener;
|
||||||
import mage.game.events.PlayerQueryEvent;
|
import mage.game.events.PlayerQueryEvent;
|
||||||
|
@ -80,6 +77,10 @@ public interface Game extends MageItem, Serializable {
|
||||||
|
|
||||||
MageObject getEmblem(UUID objectId);
|
MageObject getEmblem(UUID objectId);
|
||||||
|
|
||||||
|
Dungeon getDungeon(UUID objectId);
|
||||||
|
|
||||||
|
Dungeon getPlayerDungeon(UUID objectId);
|
||||||
|
|
||||||
UUID getControllerId(UUID objectId);
|
UUID getControllerId(UUID objectId);
|
||||||
|
|
||||||
UUID getOwnerId(UUID objectId);
|
UUID getOwnerId(UUID objectId);
|
||||||
|
@ -394,6 +395,10 @@ public interface Game extends MageItem, Serializable {
|
||||||
|
|
||||||
void addCommander(Commander commander);
|
void addCommander(Commander commander);
|
||||||
|
|
||||||
|
Dungeon addDungeon(Dungeon dungeon, UUID playerId);
|
||||||
|
|
||||||
|
void ventureIntoDungeon(UUID playerId);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Adds a permanent to the battlefield
|
* Adds a permanent to the battlefield
|
||||||
*
|
*
|
||||||
|
|
|
@ -38,10 +38,7 @@ import mage.filter.common.FilterCreaturePermanent;
|
||||||
import mage.filter.predicate.mageobject.NamePredicate;
|
import mage.filter.predicate.mageobject.NamePredicate;
|
||||||
import mage.filter.predicate.permanent.ControllerIdPredicate;
|
import mage.filter.predicate.permanent.ControllerIdPredicate;
|
||||||
import mage.game.combat.Combat;
|
import mage.game.combat.Combat;
|
||||||
import mage.game.command.CommandObject;
|
import mage.game.command.*;
|
||||||
import mage.game.command.Commander;
|
|
||||||
import mage.game.command.Emblem;
|
|
||||||
import mage.game.command.Plane;
|
|
||||||
import mage.game.events.*;
|
import mage.game.events.*;
|
||||||
import mage.game.events.TableEvent.EventType;
|
import mage.game.events.TableEvent.EventType;
|
||||||
import mage.game.mulligan.Mulligan;
|
import mage.game.mulligan.Mulligan;
|
||||||
|
@ -355,7 +352,7 @@ public abstract class GameImpl implements Game, Serializable {
|
||||||
if (item.getId().equals(objectId)) {
|
if (item.getId().equals(objectId)) {
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
if (item.getSourceId().equals(objectId) && item instanceof Spell) {
|
if (item instanceof Spell && item.getSourceId().equals(objectId)) {
|
||||||
return item;
|
return item;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -431,6 +428,59 @@ public abstract class GameImpl implements Game, Serializable {
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Dungeon getDungeon(UUID objectId) {
|
||||||
|
return state
|
||||||
|
.getCommand()
|
||||||
|
.stream()
|
||||||
|
.filter(commandObject -> commandObject.getId().equals(objectId))
|
||||||
|
.filter(Dungeon.class::isInstance)
|
||||||
|
.map(Dungeon.class::cast)
|
||||||
|
.findFirst()
|
||||||
|
.orElse(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Dungeon getPlayerDungeon(UUID playerId) {
|
||||||
|
return state
|
||||||
|
.getCommand()
|
||||||
|
.stream()
|
||||||
|
.filter(commandObject -> commandObject.isControlledBy(playerId))
|
||||||
|
.filter(Dungeon.class::isInstance)
|
||||||
|
.map(Dungeon.class::cast)
|
||||||
|
.findFirst()
|
||||||
|
.orElse(null);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void removeDungeon(Dungeon dungeon) {
|
||||||
|
if (dungeon == null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
Player player = getPlayer(dungeon.getControllerId());
|
||||||
|
if (player != null) {
|
||||||
|
informPlayers(player.getLogName() + " has completed " + dungeon.getLogName());
|
||||||
|
}
|
||||||
|
state.getCommand().remove(dungeon);
|
||||||
|
fireEvent(GameEvent.getEvent(
|
||||||
|
GameEvent.EventType.DUNGEON_COMPLETED, dungeon.getId(), null,
|
||||||
|
dungeon.getControllerId(), dungeon.getName(), 0
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
private Dungeon getOrCreateDungeon(UUID playerId) {
|
||||||
|
Dungeon dungeon = this.getPlayerDungeon(playerId);
|
||||||
|
if (dungeon != null && dungeon.hasNextRoom()) {
|
||||||
|
return dungeon;
|
||||||
|
}
|
||||||
|
removeDungeon(dungeon);
|
||||||
|
return this.addDungeon(Dungeon.selectDungeon(playerId, this), playerId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void ventureIntoDungeon(UUID playerId) {
|
||||||
|
this.getOrCreateDungeon(playerId).moveToNextRoom(playerId, this);
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public UUID getOwnerId(UUID objectId) {
|
public UUID getOwnerId(UUID objectId) {
|
||||||
return getOwnerId(getObject(objectId));
|
return getOwnerId(getObject(objectId));
|
||||||
|
@ -1658,6 +1708,13 @@ public abstract class GameImpl implements Game, Serializable {
|
||||||
state.addCommandObject(commander);
|
state.addCommandObject(commander);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Dungeon addDungeon(Dungeon dungeon, UUID playerId) {
|
||||||
|
dungeon.setControllerId(playerId);
|
||||||
|
state.addCommandObject(dungeon);
|
||||||
|
return dungeon;
|
||||||
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void addPermanent(Permanent permanent, int createOrder) {
|
public void addPermanent(Permanent permanent, int createOrder) {
|
||||||
if (createOrder == 0) {
|
if (createOrder == 0) {
|
||||||
|
@ -1902,6 +1959,34 @@ public abstract class GameImpl implements Game, Serializable {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// If a Dungeon is on its last room and is not the source of any triggered abilities, it is removed
|
||||||
|
Set<Dungeon> dungeonsToRemove = new HashSet<>();
|
||||||
|
for (CommandObject commandObject : state.getCommand()) {
|
||||||
|
if (!(commandObject instanceof Dungeon)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
Dungeon dungeon = (Dungeon) commandObject;
|
||||||
|
boolean removeDungeon = !dungeon.hasNextRoom()
|
||||||
|
&& this.getStack()
|
||||||
|
.stream()
|
||||||
|
.filter(DungeonRoom::isRoomTrigger)
|
||||||
|
.map(StackObject::getSourceId)
|
||||||
|
.noneMatch(dungeon.getId()::equals)
|
||||||
|
&& this.state
|
||||||
|
.getTriggered(dungeon.getControllerId())
|
||||||
|
.stream()
|
||||||
|
.filter(DungeonRoom::isRoomTrigger)
|
||||||
|
.map(Ability::getSourceId)
|
||||||
|
.noneMatch(dungeon.getId()::equals);
|
||||||
|
if (removeDungeon) {
|
||||||
|
dungeonsToRemove.add(dungeon);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (Dungeon dungeon : dungeonsToRemove) {
|
||||||
|
this.removeDungeon(dungeon);
|
||||||
|
somethingHappened = true;
|
||||||
|
}
|
||||||
|
|
||||||
// If a commander is in a graveyard or in exile and that card was put into that zone
|
// If a commander is in a graveyard or in exile and that card was put into that zone
|
||||||
// since the last time state-based actions were checked, its owner may put it into the command zone.
|
// since the last time state-based actions were checked, its owner may put it into the command zone.
|
||||||
// signature spells goes to command zone all the time
|
// signature spells goes to command zone all the time
|
||||||
|
|
|
@ -1,26 +1,20 @@
|
||||||
|
|
||||||
|
|
||||||
package mage.game.command;
|
package mage.game.command;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
|
||||||
* @author Viserion
|
* @author Viserion
|
||||||
*/
|
*/
|
||||||
public class Command extends ArrayList<CommandObject> {
|
public class Command extends ArrayList<CommandObject> {
|
||||||
|
|
||||||
public Command () {}
|
public Command() {
|
||||||
|
|
||||||
public Command(final Command command) {
|
|
||||||
addAll(command);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/*public void checkTriggers(GameEvent event, Game game) {
|
private Command(final Command command) {
|
||||||
for (CommandObject commandObject: this) {
|
for (CommandObject commandObject : command) {
|
||||||
commandObject.checkTriggers(event, game);
|
add(commandObject.copy());
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}*/
|
|
||||||
|
|
||||||
public Command copy() {
|
public Command copy() {
|
||||||
return new Command(this);
|
return new Command(this);
|
||||||
|
|
371
Mage/src/main/java/mage/game/command/Dungeon.java
Normal file
371
Mage/src/main/java/mage/game/command/Dungeon.java
Normal file
|
@ -0,0 +1,371 @@
|
||||||
|
package mage.game.command;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.MageObject;
|
||||||
|
import mage.ObjectColor;
|
||||||
|
import mage.abilities.Abilities;
|
||||||
|
import mage.abilities.AbilitiesImpl;
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.costs.mana.ManaCost;
|
||||||
|
import mage.abilities.costs.mana.ManaCosts;
|
||||||
|
import mage.abilities.costs.mana.ManaCostsImpl;
|
||||||
|
import mage.abilities.effects.ContinuousEffect;
|
||||||
|
import mage.abilities.effects.Effect;
|
||||||
|
import mage.abilities.text.TextPart;
|
||||||
|
import mage.cards.FrameStyle;
|
||||||
|
import mage.choices.Choice;
|
||||||
|
import mage.choices.ChoiceImpl;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.Outcome;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.constants.SuperType;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.command.dungeons.DungeonOfTheMadMage;
|
||||||
|
import mage.game.command.dungeons.LostMineOfPhandelver;
|
||||||
|
import mage.game.command.dungeons.TombOfAnnihilation;
|
||||||
|
import mage.game.events.GameEvent;
|
||||||
|
import mage.game.events.ZoneChangeEvent;
|
||||||
|
import mage.players.Player;
|
||||||
|
import mage.util.GameLog;
|
||||||
|
import mage.util.SubTypes;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class Dungeon implements CommandObject {
|
||||||
|
|
||||||
|
private static final Set<String> dungeonNames = new HashSet<>();
|
||||||
|
|
||||||
|
static {
|
||||||
|
dungeonNames.add("Tomb of Annihilation");
|
||||||
|
dungeonNames.add("Lost Mine of Phandelver");
|
||||||
|
dungeonNames.add("Dungeon of the Mad Mage");
|
||||||
|
}
|
||||||
|
|
||||||
|
private static final ArrayList<CardType> emptySet = new ArrayList<>(Arrays.asList(CardType.DUNGEON));
|
||||||
|
private static final ObjectColor emptyColor = new ObjectColor();
|
||||||
|
private static final ManaCosts<ManaCost> emptyCost = new ManaCostsImpl<>();
|
||||||
|
|
||||||
|
private final String name;
|
||||||
|
private UUID id;
|
||||||
|
private UUID controllerId;
|
||||||
|
private boolean copy;
|
||||||
|
private MageObject copyFrom; // copied card INFO (used to call original adjusters)
|
||||||
|
private FrameStyle frameStyle;
|
||||||
|
private final Abilities<Ability> abilites = new AbilitiesImpl<>();
|
||||||
|
private final String expansionSetCodeForImage;
|
||||||
|
private final List<DungeonRoom> dungeonRooms = new ArrayList<>();
|
||||||
|
private DungeonRoom currentRoom = null;
|
||||||
|
|
||||||
|
public Dungeon(String name, String expansionSetCodeForImage) {
|
||||||
|
this.id = UUID.randomUUID();
|
||||||
|
this.name = name;
|
||||||
|
this.expansionSetCodeForImage = expansionSetCodeForImage;
|
||||||
|
}
|
||||||
|
|
||||||
|
public Dungeon(final Dungeon dungeon) {
|
||||||
|
this.id = dungeon.id;
|
||||||
|
this.name = dungeon.name;
|
||||||
|
this.frameStyle = dungeon.frameStyle;
|
||||||
|
this.controllerId = dungeon.controllerId;
|
||||||
|
this.copy = dungeon.copy;
|
||||||
|
this.copyFrom = (dungeon.copyFrom != null ? dungeon.copyFrom : null);
|
||||||
|
this.expansionSetCodeForImage = dungeon.expansionSetCodeForImage;
|
||||||
|
this.copyRooms(dungeon);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void copyRooms(Dungeon dungeon) {
|
||||||
|
Map<String, DungeonRoom> copyMap = new HashMap<>();
|
||||||
|
for (DungeonRoom dungeonRoom : dungeon.dungeonRooms) {
|
||||||
|
DungeonRoom copiedRoom = copyMap.computeIfAbsent(dungeonRoom.getName(), (s) -> dungeonRoom.copy());
|
||||||
|
for (DungeonRoom nextRoom : dungeonRoom.getNextRooms()) {
|
||||||
|
copiedRoom.addNextRoom(copyMap.computeIfAbsent(nextRoom.getName(), (s) -> nextRoom.copy()));
|
||||||
|
}
|
||||||
|
this.addRoom(copiedRoom);
|
||||||
|
}
|
||||||
|
this.currentRoom = copyMap.computeIfAbsent(dungeon.currentRoom.getName(), (s) -> dungeon.currentRoom.copy());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addRoom(DungeonRoom room) {
|
||||||
|
this.dungeonRooms.add(room);
|
||||||
|
room.getRoomTriggeredAbility().setSourceId(id);
|
||||||
|
this.abilites.add(room.getRoomTriggeredAbility());
|
||||||
|
}
|
||||||
|
|
||||||
|
public void moveToNextRoom(UUID playerId, Game game) {
|
||||||
|
if (currentRoom == null) {
|
||||||
|
currentRoom = dungeonRooms.get(0);
|
||||||
|
} else {
|
||||||
|
currentRoom = currentRoom.chooseNextRoom(playerId, game);
|
||||||
|
}
|
||||||
|
Player player = game.getPlayer(getControllerId());
|
||||||
|
if (player != null) {
|
||||||
|
game.informPlayers(player.getLogName() + " has entered " + currentRoom.getName());
|
||||||
|
}
|
||||||
|
game.fireEvent(GameEvent.getEvent(
|
||||||
|
GameEvent.EventType.ROOM_ENTERED, currentRoom.getId(), null, playerId
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
public DungeonRoom getCurrentRoom() {
|
||||||
|
return currentRoom;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasNextRoom() {
|
||||||
|
return currentRoom != null && currentRoom.hasNextRoom();
|
||||||
|
}
|
||||||
|
|
||||||
|
public List<String> getRules() {
|
||||||
|
List<String> rules = new ArrayList<>();
|
||||||
|
rules.add("<i>(" + (
|
||||||
|
currentRoom != null ?
|
||||||
|
"Currently in " + currentRoom.getName() :
|
||||||
|
"Not currently in a room"
|
||||||
|
) + ")</i>");
|
||||||
|
dungeonRooms.stream().map(DungeonRoom::toString).forEach(rules::add);
|
||||||
|
return rules;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Dungeon selectDungeon(UUID playerId, Game game) {
|
||||||
|
Player player = game.getPlayer(playerId);
|
||||||
|
Choice choice = new ChoiceImpl(true);
|
||||||
|
choice.setMessage("Choose a dungeon to venture into");
|
||||||
|
choice.setChoices(dungeonNames);
|
||||||
|
player.choose(Outcome.Neutral, choice, game);
|
||||||
|
switch (choice.getChoice()) {
|
||||||
|
case "Tomb of Annihilation":
|
||||||
|
return new TombOfAnnihilation();
|
||||||
|
case "Lost Mine of Phandelver":
|
||||||
|
return new LostMineOfPhandelver();
|
||||||
|
case "Dungeon of the Mad Mage":
|
||||||
|
return new DungeonOfTheMadMage();
|
||||||
|
default:
|
||||||
|
throw new UnsupportedOperationException("A dungeon should have been chosen");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public FrameStyle getFrameStyle() {
|
||||||
|
return frameStyle;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void assignNewId() {
|
||||||
|
this.id = UUID.randomUUID();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MageObject getSourceObject() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UUID getSourceId() {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UUID getControllerId() {
|
||||||
|
return this.controllerId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setControllerId(UUID controllerId) {
|
||||||
|
this.controllerId = controllerId;
|
||||||
|
this.abilites.setControllerId(controllerId);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setCopy(boolean isCopy, MageObject copyFrom) {
|
||||||
|
this.copy = isCopy;
|
||||||
|
this.copyFrom = (copyFrom != null ? copyFrom.copy() : null);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isCopy() {
|
||||||
|
return this.copy;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MageObject getCopyFrom() {
|
||||||
|
return this.copyFrom;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getIdName() {
|
||||||
|
return getName() + " [" + getId().toString().substring(0, 3) + ']';
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getLogName() {
|
||||||
|
return GameLog.getColoredObjectIdName(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getImageName() {
|
||||||
|
return this.name;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setName(String name) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ArrayList<CardType> getCardType() {
|
||||||
|
return emptySet;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public SubTypes getSubtype() {
|
||||||
|
return new SubTypes();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public SubTypes getSubtype(Game game) {
|
||||||
|
return new SubTypes();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasSubtype(SubType subtype, Game game) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EnumSet<SuperType> getSuperType() {
|
||||||
|
return EnumSet.noneOf(SuperType.class);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Abilities<Ability> getAbilities() {
|
||||||
|
return abilites;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean hasAbility(Ability ability, Game game) {
|
||||||
|
return getAbilities().contains(ability);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ObjectColor getColor() {
|
||||||
|
return emptyColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ObjectColor getColor(Game game) {
|
||||||
|
return emptyColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ObjectColor getFrameColor(Game game) {
|
||||||
|
return emptyColor;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public ManaCosts<ManaCost> getManaCost() {
|
||||||
|
return emptyCost;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getManaValue() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MageInt getPower() {
|
||||||
|
return MageInt.EmptyMageInt;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MageInt getToughness() {
|
||||||
|
return MageInt.EmptyMageInt;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getStartingLoyalty() {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setStartingLoyalty(int startingLoyalty) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void adjustCosts(Ability ability, Game game) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void adjustTargets(Ability ability, Game game) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public UUID getId() {
|
||||||
|
return this.id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Dungeon copy() {
|
||||||
|
return new Dungeon(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getExpansionSetCodeForImage() {
|
||||||
|
return expansionSetCodeForImage;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int getZoneChangeCounter(Game game) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void updateZoneChangeCounter(Game game, ZoneChangeEvent event) {
|
||||||
|
throw new UnsupportedOperationException("Unsupported operation");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setZoneChangeCounter(int value, Game game) {
|
||||||
|
throw new UnsupportedOperationException("Unsupported operation");
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean isAllCreatureTypes(Game game) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setIsAllCreatureTypes(boolean value) {
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void setIsAllCreatureTypes(Game game, boolean value) {
|
||||||
|
}
|
||||||
|
|
||||||
|
public void discardEffects() {
|
||||||
|
for (Ability ability : abilites) {
|
||||||
|
for (Effect effect : ability.getEffects()) {
|
||||||
|
if (effect instanceof ContinuousEffect) {
|
||||||
|
((ContinuousEffect) effect).discard();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public List<TextPart> getTextParts() {
|
||||||
|
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public TextPart addTextPart(TextPart textPart) {
|
||||||
|
throw new UnsupportedOperationException("Not supported yet."); //To change body of generated methods, choose Tools | Templates.
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void removePTCDA() {
|
||||||
|
}
|
||||||
|
}
|
167
Mage/src/main/java/mage/game/command/DungeonRoom.java
Normal file
167
Mage/src/main/java/mage/game/command/DungeonRoom.java
Normal file
|
@ -0,0 +1,167 @@
|
||||||
|
package mage.game.command;
|
||||||
|
|
||||||
|
import mage.abilities.TriggeredAbility;
|
||||||
|
import mage.abilities.TriggeredAbilityImpl;
|
||||||
|
import mage.abilities.effects.Effect;
|
||||||
|
import mage.constants.Outcome;
|
||||||
|
import mage.constants.Zone;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.events.GameEvent;
|
||||||
|
import mage.game.stack.StackAbility;
|
||||||
|
import mage.game.stack.StackObject;
|
||||||
|
import mage.players.Player;
|
||||||
|
import mage.target.Target;
|
||||||
|
import mage.util.CardUtil;
|
||||||
|
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
import java.util.UUID;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class DungeonRoom {
|
||||||
|
|
||||||
|
private final UUID id;
|
||||||
|
private final String name;
|
||||||
|
private final List<DungeonRoom> nextRooms = new ArrayList<>();
|
||||||
|
private final RoomTriggeredAbility roomTriggeredAbility;
|
||||||
|
|
||||||
|
public DungeonRoom(String name, Effect... effects) {
|
||||||
|
this.id = UUID.randomUUID();
|
||||||
|
this.name = name;
|
||||||
|
roomTriggeredAbility = new RoomTriggeredAbility(this, effects);
|
||||||
|
}
|
||||||
|
|
||||||
|
private DungeonRoom(final DungeonRoom room) {
|
||||||
|
this.id = room.id;
|
||||||
|
this.name = room.name;
|
||||||
|
this.roomTriggeredAbility = new RoomTriggeredAbility(this, room.roomTriggeredAbility);
|
||||||
|
}
|
||||||
|
|
||||||
|
public DungeonRoom copy() {
|
||||||
|
return new DungeonRoom(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addTarget(Target target) {
|
||||||
|
roomTriggeredAbility.addTarget(target);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void addNextRoom(DungeonRoom room) {
|
||||||
|
nextRooms.add(room);
|
||||||
|
}
|
||||||
|
|
||||||
|
public RoomTriggeredAbility getRoomTriggeredAbility() {
|
||||||
|
return roomTriggeredAbility;
|
||||||
|
}
|
||||||
|
|
||||||
|
public UUID getId() {
|
||||||
|
return id;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String toString() {
|
||||||
|
return roomTriggeredAbility.getText();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getName() {
|
||||||
|
return name;
|
||||||
|
}
|
||||||
|
|
||||||
|
public boolean hasNextRoom() {
|
||||||
|
return !nextRooms.isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
List<DungeonRoom> getNextRooms() {
|
||||||
|
return nextRooms;
|
||||||
|
}
|
||||||
|
|
||||||
|
public DungeonRoom chooseNextRoom(UUID playerId, Game game) {
|
||||||
|
switch (nextRooms.size()) {
|
||||||
|
case 0:
|
||||||
|
return null;
|
||||||
|
case 1:
|
||||||
|
return nextRooms.get(0);
|
||||||
|
case 2:
|
||||||
|
DungeonRoom room1 = nextRooms.get(0);
|
||||||
|
DungeonRoom room2 = nextRooms.get(1);
|
||||||
|
Player player = game.getPlayer(playerId);
|
||||||
|
if (player == null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return player.chooseUse(
|
||||||
|
Outcome.Neutral, "Choose which room to go to",
|
||||||
|
null, room1.name, room2.name, null, game
|
||||||
|
) ? room1 : room2;
|
||||||
|
default:
|
||||||
|
throw new UnsupportedOperationException("there shouldn't be more than two rooms to go to");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String generateDestinationText() {
|
||||||
|
if (nextRooms.isEmpty()) {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
return " <i>(Leads to "
|
||||||
|
+ nextRooms
|
||||||
|
.stream()
|
||||||
|
.map(DungeonRoom::getName)
|
||||||
|
.reduce((a, b) -> a + " or " + b)
|
||||||
|
.orElse("")
|
||||||
|
+ ")</i>";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isRoomTrigger(StackObject stackObject) {
|
||||||
|
return stackObject instanceof StackAbility
|
||||||
|
&& stackObject.getStackAbility() instanceof RoomTriggeredAbility;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isRoomTrigger(TriggeredAbility ability) {
|
||||||
|
return ability instanceof RoomTriggeredAbility;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RoomTriggeredAbility extends TriggeredAbilityImpl {
|
||||||
|
|
||||||
|
private final DungeonRoom room;
|
||||||
|
|
||||||
|
RoomTriggeredAbility(DungeonRoom room, Effect... effects) {
|
||||||
|
super(Zone.COMMAND, null, false);
|
||||||
|
this.room = room;
|
||||||
|
for (Effect effect : effects) {
|
||||||
|
this.addEffect(effect);
|
||||||
|
}
|
||||||
|
this.setRuleVisible(false);
|
||||||
|
}
|
||||||
|
|
||||||
|
RoomTriggeredAbility(DungeonRoom room, final RoomTriggeredAbility ability) {
|
||||||
|
super(ability);
|
||||||
|
this.room = room;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean checkEventType(GameEvent event, Game game) {
|
||||||
|
return event.getType() == GameEvent.EventType.ROOM_ENTERED;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean checkTrigger(GameEvent event, Game game) {
|
||||||
|
return event.getTargetId().equals(room.getId());
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RoomTriggeredAbility copy() {
|
||||||
|
return new RoomTriggeredAbility(this.room, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
public String getText() {
|
||||||
|
return room.getName() + " — "
|
||||||
|
+ CardUtil.getTextWithFirstCharUpperCase(super.getRule())
|
||||||
|
+ room.generateDestinationText();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getRule() {
|
||||||
|
return "When you enter this room, " + super.getRule() + " <i>(" + room.getName() + ")</i>";
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,189 @@
|
||||||
|
package mage.game.command.dungeons;
|
||||||
|
|
||||||
|
import mage.ApprovingObject;
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.effects.OneShotEffect;
|
||||||
|
import mage.abilities.effects.common.CreateTokenEffect;
|
||||||
|
import mage.abilities.effects.common.GainLifeEffect;
|
||||||
|
import mage.abilities.effects.common.combat.CantAttackTargetEffect;
|
||||||
|
import mage.abilities.effects.keyword.ScryEffect;
|
||||||
|
import mage.cards.Card;
|
||||||
|
import mage.cards.Cards;
|
||||||
|
import mage.cards.CardsImpl;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.constants.Outcome;
|
||||||
|
import mage.constants.Zone;
|
||||||
|
import mage.filter.StaticFilters;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
|
import mage.game.command.DungeonRoom;
|
||||||
|
import mage.game.permanent.token.SkeletonToken2;
|
||||||
|
import mage.game.permanent.token.TreasureToken;
|
||||||
|
import mage.players.Player;
|
||||||
|
import mage.target.common.TargetCardInHand;
|
||||||
|
import mage.target.common.TargetCreaturePermanent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class DungeonOfTheMadMage extends Dungeon {
|
||||||
|
|
||||||
|
public DungeonOfTheMadMage() {
|
||||||
|
super("Dungeon of the Mad Mage", "AFR");
|
||||||
|
// (1) Yawning Portal — You gain 1 life. (→ 2)
|
||||||
|
DungeonRoom yawningPortal = new DungeonRoom("Yawning Portal", new GainLifeEffect(1));
|
||||||
|
|
||||||
|
// (2) Dungeon Level — Scry 1. (→ 3a or 3b)
|
||||||
|
DungeonRoom dungeonLevel = new DungeonRoom(
|
||||||
|
"Dungeon Level", new ScryEffect(1, false)
|
||||||
|
);
|
||||||
|
|
||||||
|
// (3a) Goblin Bazaar — Create a Treasure token. (→ 4)
|
||||||
|
DungeonRoom goblinBazaar = new DungeonRoom("Goblin Bazaar", new CreateTokenEffect(new TreasureToken()));
|
||||||
|
|
||||||
|
// (3b) Twisted Caverns — Target creature can't attack until your next turn. (→ 4)
|
||||||
|
DungeonRoom twistedCaverns = new DungeonRoom(
|
||||||
|
"Twisted Caverns", new CantAttackTargetEffect(Duration.UntilYourNextTurn)
|
||||||
|
);
|
||||||
|
twistedCaverns.addTarget(new TargetCreaturePermanent());
|
||||||
|
|
||||||
|
// (4) Lost Level — Scry 2. (→ 5a or 5b)
|
||||||
|
DungeonRoom lostLevel = new DungeonRoom("Lost Level", new ScryEffect(2, false));
|
||||||
|
|
||||||
|
// (5a) Runestone Caverns — Exile the top two cards of your library. You may play them. (→ 6)
|
||||||
|
DungeonRoom runestoneCaverns = new DungeonRoom("Runestone Caverns", new RunestoneCavernsEffect());
|
||||||
|
|
||||||
|
// (5b) Muiral's Graveyard — Create two 1/1 black Skeleton creature tokens. (→ 6)
|
||||||
|
DungeonRoom muiralsGraveyard = new DungeonRoom(
|
||||||
|
"Muiral's Graveyard", new CreateTokenEffect(new SkeletonToken2(), 2)
|
||||||
|
);
|
||||||
|
|
||||||
|
// (6) Deep Mines — Scry 3. (→ 7)
|
||||||
|
DungeonRoom deepMines = new DungeonRoom("Deep Mines", new ScryEffect(3, false));
|
||||||
|
|
||||||
|
// (7) Mad Wizard's Lair — Draw three cards and reveal them. You may cast one of them without paying its mana cost.
|
||||||
|
DungeonRoom madWizardsLair = new DungeonRoom("Mad Wizard's Lair", new MadWizardsLairEffect());
|
||||||
|
|
||||||
|
yawningPortal.addNextRoom(dungeonLevel);
|
||||||
|
dungeonLevel.addNextRoom(goblinBazaar);
|
||||||
|
dungeonLevel.addNextRoom(twistedCaverns);
|
||||||
|
goblinBazaar.addNextRoom(lostLevel);
|
||||||
|
twistedCaverns.addNextRoom(lostLevel);
|
||||||
|
lostLevel.addNextRoom(runestoneCaverns);
|
||||||
|
lostLevel.addNextRoom(muiralsGraveyard);
|
||||||
|
runestoneCaverns.addNextRoom(deepMines);
|
||||||
|
muiralsGraveyard.addNextRoom(deepMines);
|
||||||
|
deepMines.addNextRoom(madWizardsLair);
|
||||||
|
|
||||||
|
this.addRoom(yawningPortal);
|
||||||
|
this.addRoom(dungeonLevel);
|
||||||
|
this.addRoom(goblinBazaar);
|
||||||
|
this.addRoom(twistedCaverns);
|
||||||
|
this.addRoom(lostLevel);
|
||||||
|
this.addRoom(runestoneCaverns);
|
||||||
|
this.addRoom(muiralsGraveyard);
|
||||||
|
this.addRoom(deepMines);
|
||||||
|
this.addRoom(madWizardsLair);
|
||||||
|
}
|
||||||
|
|
||||||
|
private DungeonOfTheMadMage(final DungeonOfTheMadMage dungeon) {
|
||||||
|
super(dungeon);
|
||||||
|
}
|
||||||
|
|
||||||
|
public DungeonOfTheMadMage copy() {
|
||||||
|
return new DungeonOfTheMadMage(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class RunestoneCavernsEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
RunestoneCavernsEffect() {
|
||||||
|
super(Outcome.Benefit);
|
||||||
|
staticText = "exile the top two cards of your library. You may play them";
|
||||||
|
}
|
||||||
|
|
||||||
|
private RunestoneCavernsEffect(final RunestoneCavernsEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public RunestoneCavernsEffect copy() {
|
||||||
|
return new RunestoneCavernsEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Player player = game.getPlayer(source.getControllerId());
|
||||||
|
if (player == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Cards cards = new CardsImpl(player.getLibrary().getTopCards(game, 2));
|
||||||
|
if (cards.isEmpty()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
player.moveCards(cards, Zone.EXILED, source, game);
|
||||||
|
while (!cards.isEmpty()) {
|
||||||
|
for (Card card : cards.getCards(game)) {
|
||||||
|
if (!player.chooseUse(Outcome.PlayForFree, "Play " + card.getName() + "?", source, game)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), Boolean.TRUE);
|
||||||
|
player.cast(
|
||||||
|
player.chooseAbilityForCast(card, game, false),
|
||||||
|
game, false, new ApprovingObject(source, game)
|
||||||
|
);
|
||||||
|
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), null);
|
||||||
|
}
|
||||||
|
cards.retainZone(Zone.EXILED, game);
|
||||||
|
if (cards.isEmpty() || !player.chooseUse(
|
||||||
|
Outcome.PlayForFree, "Continue playing the exiled cards?", source, game
|
||||||
|
)) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class MadWizardsLairEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
MadWizardsLairEffect() {
|
||||||
|
super(Outcome.Benefit);
|
||||||
|
staticText = "draw three cards and reveal them. You may cast one of them without paying its mana cost";
|
||||||
|
}
|
||||||
|
|
||||||
|
private MadWizardsLairEffect(final MadWizardsLairEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public MadWizardsLairEffect copy() {
|
||||||
|
return new MadWizardsLairEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Player player = game.getPlayer(source.getControllerId());
|
||||||
|
if (player == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Cards cards = new CardsImpl(player.getLibrary().getTopCards(game, 3));
|
||||||
|
if (player.drawCards(3, source, game) != cards.size()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
player.revealCards(source, cards, game);
|
||||||
|
TargetCardInHand target = new TargetCardInHand(0, 1, StaticFilters.FILTER_CARD_NON_LAND);
|
||||||
|
player.choose(Outcome.PlayForFree, cards, target, game);
|
||||||
|
Card card = player.getHand().get(target.getFirstTarget(), game);
|
||||||
|
if (card == null) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), Boolean.TRUE);
|
||||||
|
player.cast(
|
||||||
|
player.chooseAbilityForCast(card, game, true),
|
||||||
|
game, true, new ApprovingObject(source, game)
|
||||||
|
);
|
||||||
|
game.getState().setValue("PlayFromNotOwnHandZone" + card.getId(), null);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,86 @@
|
||||||
|
package mage.game.command.dungeons;
|
||||||
|
|
||||||
|
import mage.abilities.effects.common.CreateTokenEffect;
|
||||||
|
import mage.abilities.effects.common.DrawCardSourceControllerEffect;
|
||||||
|
import mage.abilities.effects.common.GainLifeEffect;
|
||||||
|
import mage.abilities.effects.common.LoseLifeOpponentsEffect;
|
||||||
|
import mage.abilities.effects.common.continuous.BoostTargetEffect;
|
||||||
|
import mage.abilities.effects.common.counter.AddCountersTargetEffect;
|
||||||
|
import mage.abilities.effects.keyword.ScryEffect;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.counters.CounterType;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
|
import mage.game.command.DungeonRoom;
|
||||||
|
import mage.game.permanent.token.GoblinToken;
|
||||||
|
import mage.game.permanent.token.TreasureToken;
|
||||||
|
import mage.target.common.TargetCreaturePermanent;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class LostMineOfPhandelver extends Dungeon {
|
||||||
|
|
||||||
|
public LostMineOfPhandelver() {
|
||||||
|
super("Lost Mine of Phandelver", "AFR");
|
||||||
|
// (1) Cave Entrance — Scry 1. (→ 2a or 2b)
|
||||||
|
DungeonRoom caveEntrance = new DungeonRoom(
|
||||||
|
"Cave Entrance", new ScryEffect(1, false)
|
||||||
|
);
|
||||||
|
|
||||||
|
// (2a) Goblin Lair — Create a 1/1 red Goblin creature token. (→ 3a or 3b)
|
||||||
|
DungeonRoom goblinLair = new DungeonRoom("Goblin Lair", new CreateTokenEffect(new GoblinToken()));
|
||||||
|
|
||||||
|
// (2b) Mine Tunnels — Create a Treasure token. (→ 3b or 3c)
|
||||||
|
DungeonRoom mineTunnels = new DungeonRoom("Mine Tunnels", new CreateTokenEffect(new TreasureToken()));
|
||||||
|
|
||||||
|
// (3a) Storeroom — Put a +1/+1 counter on target creature. (→ 4)
|
||||||
|
DungeonRoom storeroom = new DungeonRoom(
|
||||||
|
"Storeroom", new AddCountersTargetEffect(CounterType.P1P1.createInstance())
|
||||||
|
);
|
||||||
|
storeroom.addTarget(new TargetCreaturePermanent());
|
||||||
|
|
||||||
|
// (3b) Dark Pool — Each opponent loses 1 life and you gain 1 life. (→ 4)
|
||||||
|
DungeonRoom darkPool = new DungeonRoom(
|
||||||
|
"Dark Pool", new LoseLifeOpponentsEffect(1),
|
||||||
|
new GainLifeEffect(1).concatBy("and")
|
||||||
|
);
|
||||||
|
|
||||||
|
// (3c) Fungi Cavern — Target creature gets -4/-0 until your next turn. (→ 4)
|
||||||
|
DungeonRoom fungiCavern = new DungeonRoom(
|
||||||
|
"Fungi Cavern", new BoostTargetEffect(-4, 0, Duration.UntilYourNextTurn)
|
||||||
|
);
|
||||||
|
fungiCavern.addTarget(new TargetCreaturePermanent());
|
||||||
|
|
||||||
|
// (4) Temple of Dumathoin — Draw a card.
|
||||||
|
DungeonRoom templeOfDumathoin = new DungeonRoom(
|
||||||
|
"Temple of Dumathoin", new DrawCardSourceControllerEffect(1)
|
||||||
|
);
|
||||||
|
|
||||||
|
caveEntrance.addNextRoom(goblinLair);
|
||||||
|
caveEntrance.addNextRoom(mineTunnels);
|
||||||
|
goblinLair.addNextRoom(storeroom);
|
||||||
|
goblinLair.addNextRoom(darkPool);
|
||||||
|
mineTunnels.addNextRoom(darkPool);
|
||||||
|
mineTunnels.addNextRoom(fungiCavern);
|
||||||
|
storeroom.addNextRoom(templeOfDumathoin);
|
||||||
|
darkPool.addNextRoom(templeOfDumathoin);
|
||||||
|
fungiCavern.addNextRoom(templeOfDumathoin);
|
||||||
|
|
||||||
|
this.addRoom(caveEntrance);
|
||||||
|
this.addRoom(goblinLair);
|
||||||
|
this.addRoom(mineTunnels);
|
||||||
|
this.addRoom(storeroom);
|
||||||
|
this.addRoom(darkPool);
|
||||||
|
this.addRoom(fungiCavern);
|
||||||
|
this.addRoom(templeOfDumathoin);
|
||||||
|
}
|
||||||
|
|
||||||
|
private LostMineOfPhandelver(final LostMineOfPhandelver dungeon) {
|
||||||
|
super(dungeon);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public LostMineOfPhandelver copy() {
|
||||||
|
return new LostMineOfPhandelver(this);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,266 @@
|
||||||
|
package mage.game.command.dungeons;
|
||||||
|
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.dynamicvalue.common.CardTypeAssignment;
|
||||||
|
import mage.abilities.effects.OneShotEffect;
|
||||||
|
import mage.abilities.effects.common.CreateTokenEffect;
|
||||||
|
import mage.abilities.effects.common.LoseLifeAllPlayersEffect;
|
||||||
|
import mage.cards.Card;
|
||||||
|
import mage.cards.Cards;
|
||||||
|
import mage.cards.CardsImpl;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.Outcome;
|
||||||
|
import mage.filter.StaticFilters;
|
||||||
|
import mage.filter.common.FilterControlledPermanent;
|
||||||
|
import mage.filter.predicate.Predicates;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.command.Dungeon;
|
||||||
|
import mage.game.command.DungeonRoom;
|
||||||
|
import mage.game.permanent.Permanent;
|
||||||
|
import mage.game.permanent.token.TheAtropalToken;
|
||||||
|
import mage.players.Player;
|
||||||
|
import mage.target.TargetPermanent;
|
||||||
|
import mage.target.common.TargetControlledPermanent;
|
||||||
|
import mage.target.common.TargetDiscard;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class TombOfAnnihilation extends Dungeon {
|
||||||
|
|
||||||
|
static final FilterControlledPermanent filter
|
||||||
|
= new FilterControlledPermanent("an artifact, a creature, or a land");
|
||||||
|
|
||||||
|
static {
|
||||||
|
filter.add(Predicates.or(
|
||||||
|
CardType.ARTIFACT.getPredicate(),
|
||||||
|
CardType.CREATURE.getPredicate(),
|
||||||
|
CardType.LAND.getPredicate()
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
|
public TombOfAnnihilation() {
|
||||||
|
super("Tomb of Annihilation", "AFR");
|
||||||
|
// (1) Trapped Entry — Each player loses 1 life. (→ 2a or 2b)
|
||||||
|
DungeonRoom trappedEntry = new DungeonRoom("Trapped Entry", new LoseLifeAllPlayersEffect(1));
|
||||||
|
|
||||||
|
// (2a) Veils of Fear — Each player loses 2 life unless they discard a card. (→ 3)
|
||||||
|
DungeonRoom veilsOfFear = new DungeonRoom("Veils of Fear", new VeilsOfFearEffect());
|
||||||
|
|
||||||
|
// (2b) Oubliette — Discard a card and sacrifice an artifact, a creature, and a land. (→ 4)
|
||||||
|
DungeonRoom oubliette = new DungeonRoom("Oubliette", new OublietteEffect());
|
||||||
|
|
||||||
|
// (3) Sandfall Cell — Each player loses 2 life unless they sacrifice an artifact, a creature, or a land. (→ 4)
|
||||||
|
DungeonRoom sandfallCell = new DungeonRoom("Sandfall Cell", new SandfallCellEffect());
|
||||||
|
|
||||||
|
// (4) Cradle of the Death God — Create The Atropal, a legendary 4/4 black God Horror creature token with deathtouch.
|
||||||
|
DungeonRoom cradleOfTheDeathGod = new DungeonRoom("Cradle of the Death God", new CreateTokenEffect(new TheAtropalToken()));
|
||||||
|
|
||||||
|
trappedEntry.addNextRoom(veilsOfFear);
|
||||||
|
trappedEntry.addNextRoom(oubliette);
|
||||||
|
veilsOfFear.addNextRoom(sandfallCell);
|
||||||
|
oubliette.addNextRoom(sandfallCell);
|
||||||
|
sandfallCell.addNextRoom(cradleOfTheDeathGod);
|
||||||
|
|
||||||
|
this.addRoom(trappedEntry);
|
||||||
|
this.addRoom(veilsOfFear);
|
||||||
|
this.addRoom(oubliette);
|
||||||
|
this.addRoom(sandfallCell);
|
||||||
|
this.addRoom(cradleOfTheDeathGod);
|
||||||
|
}
|
||||||
|
|
||||||
|
private TombOfAnnihilation(final TombOfAnnihilation dungeon) {
|
||||||
|
super(dungeon);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TombOfAnnihilation copy() {
|
||||||
|
return new TombOfAnnihilation(this);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class VeilsOfFearEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
VeilsOfFearEffect() {
|
||||||
|
super(Outcome.Neutral);
|
||||||
|
staticText = "each player loses 2 life unless they discard a card";
|
||||||
|
}
|
||||||
|
|
||||||
|
private VeilsOfFearEffect(final VeilsOfFearEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public VeilsOfFearEffect copy() {
|
||||||
|
return new VeilsOfFearEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Map<UUID, Card> map = new HashMap<>();
|
||||||
|
for (UUID playerId : game.getState().getPlayersInRange(source.getControllerId(), game)) {
|
||||||
|
Player player = game.getPlayer(playerId);
|
||||||
|
if (player == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
TargetDiscard target = new TargetDiscard(0, 1, StaticFilters.FILTER_CARD, playerId);
|
||||||
|
player.choose(Outcome.PreventDamage, target, source.getSourceId(), game);
|
||||||
|
map.put(playerId, game.getCard(target.getFirstTarget()));
|
||||||
|
}
|
||||||
|
for (Map.Entry<UUID, Card> entry : map.entrySet()) {
|
||||||
|
Player player = game.getPlayer(entry.getKey());
|
||||||
|
if (player == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (entry.getValue() != null) {
|
||||||
|
player.discard(entry.getValue(), false, source, game);
|
||||||
|
} else {
|
||||||
|
player.loseLife(2, game, source, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class OublietteEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
OublietteEffect() {
|
||||||
|
super(Outcome.Sacrifice);
|
||||||
|
staticText = "discard a card and sacrifice an artifact, a creature, and a land";
|
||||||
|
}
|
||||||
|
|
||||||
|
private OublietteEffect(final OublietteEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public OublietteEffect copy() {
|
||||||
|
return new OublietteEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Player player = game.getPlayer(source.getControllerId());
|
||||||
|
if (player == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
player.discard(1, false, false, source, game);
|
||||||
|
int saccable = OublietteTarget.checkTargetCount(source, game);
|
||||||
|
if (saccable < 1) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
OublietteTarget target = new OublietteTarget(Math.min(saccable, 3));
|
||||||
|
player.choose(Outcome.Sacrifice, target, source.getSourceId(), game);
|
||||||
|
for (UUID targetId : target.getTargets()) {
|
||||||
|
Permanent permanent = game.getPermanent(targetId);
|
||||||
|
if (permanent != null) {
|
||||||
|
permanent.sacrifice(source, game);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class OublietteTarget extends TargetControlledPermanent {
|
||||||
|
|
||||||
|
private static final CardTypeAssignment cardTypeAssigner = new CardTypeAssignment(
|
||||||
|
CardType.ARTIFACT,
|
||||||
|
CardType.CREATURE,
|
||||||
|
CardType.LAND
|
||||||
|
);
|
||||||
|
private static final FilterControlledPermanent filter = TombOfAnnihilation.filter.copy();
|
||||||
|
|
||||||
|
static {
|
||||||
|
filter.setMessage("an artifact, a creature, and a land");
|
||||||
|
}
|
||||||
|
|
||||||
|
OublietteTarget(int numTargets) {
|
||||||
|
super(numTargets, numTargets, filter, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
private OublietteTarget(final OublietteTarget target) {
|
||||||
|
super(target);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public OublietteTarget copy() {
|
||||||
|
return new OublietteTarget(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean canTarget(UUID playerId, UUID id, Ability ability, Game game) {
|
||||||
|
if (!super.canTarget(playerId, id, ability, game)) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
Permanent permanent = game.getPermanent(id);
|
||||||
|
if (permanent == null) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (this.getTargets().isEmpty()) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
Cards cards = new CardsImpl(this.getTargets());
|
||||||
|
cards.add(permanent);
|
||||||
|
return cardTypeAssigner.getRoleCount(cards, game) >= cards.size();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Set<UUID> possibleTargets(UUID sourceId, UUID sourceControllerId, Game game) {
|
||||||
|
Set<UUID> possibleTargets = super.possibleTargets(sourceId, sourceControllerId, game);
|
||||||
|
possibleTargets.removeIf(uuid -> !this.canTarget(sourceControllerId, uuid, null, game));
|
||||||
|
return possibleTargets;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int checkTargetCount(Ability source, Game game) {
|
||||||
|
List<Permanent> permanents = game
|
||||||
|
.getBattlefield()
|
||||||
|
.getActivePermanents(filter, source.getControllerId(), source.getSourceId(), game);
|
||||||
|
return cardTypeAssigner.getRoleCount(new CardsImpl(permanents), game);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class SandfallCellEffect extends OneShotEffect {
|
||||||
|
|
||||||
|
SandfallCellEffect() {
|
||||||
|
super(Outcome.Neutral);
|
||||||
|
staticText = "each player loses 2 life unless they sacrifice an artifact, a creature, or a land";
|
||||||
|
}
|
||||||
|
|
||||||
|
private SandfallCellEffect(final SandfallCellEffect effect) {
|
||||||
|
super(effect);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public SandfallCellEffect copy() {
|
||||||
|
return new SandfallCellEffect(this);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean apply(Game game, Ability source) {
|
||||||
|
Map<UUID, Permanent> map = new HashMap<>();
|
||||||
|
for (UUID playerId : game.getState().getPlayersInRange(source.getControllerId(), game)) {
|
||||||
|
Player player = game.getPlayer(playerId);
|
||||||
|
if (player == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
TargetPermanent target = new TargetPermanent(0, 1, TombOfAnnihilation.filter, true);
|
||||||
|
player.choose(Outcome.PreventDamage, target, source.getSourceId(), game);
|
||||||
|
map.put(playerId, game.getPermanent(target.getFirstTarget()));
|
||||||
|
}
|
||||||
|
for (Map.Entry<UUID, Permanent> entry : map.entrySet()) {
|
||||||
|
Player player = game.getPlayer(entry.getKey());
|
||||||
|
if (player == null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (entry.getValue() != null) {
|
||||||
|
entry.getValue().sacrifice(source, game);
|
||||||
|
} else {
|
||||||
|
player.loseLife(2, game, source, false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -0,0 +1,81 @@
|
||||||
|
package mage.game.command.emblems;
|
||||||
|
|
||||||
|
import mage.abilities.Ability;
|
||||||
|
import mage.abilities.common.SimpleStaticAbility;
|
||||||
|
import mage.abilities.dynamicvalue.DynamicValue;
|
||||||
|
import mage.abilities.effects.Effect;
|
||||||
|
import mage.abilities.effects.common.continuous.BoostControlledEffect;
|
||||||
|
import mage.abilities.effects.common.continuous.GainAbilityControlledEffect;
|
||||||
|
import mage.abilities.hint.Hint;
|
||||||
|
import mage.abilities.keyword.HasteAbility;
|
||||||
|
import mage.abilities.keyword.TrampleAbility;
|
||||||
|
import mage.constants.Duration;
|
||||||
|
import mage.filter.StaticFilters;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.command.Emblem;
|
||||||
|
import mage.watchers.common.CompletedDungeonWatcher;
|
||||||
|
|
||||||
|
import java.util.Set;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class EllywickTumblestrumEmblem extends Emblem {
|
||||||
|
|
||||||
|
// −7: You get an emblem with "Creatures you control have trample and haste and get +2/+2 for each differently named dungeon you've completed."
|
||||||
|
public EllywickTumblestrumEmblem() {
|
||||||
|
this.setName("Emblem Ellywick");
|
||||||
|
this.setExpansionSetCodeForImage("AFR");
|
||||||
|
Ability ability = new SimpleStaticAbility(new GainAbilityControlledEffect(
|
||||||
|
TrampleAbility.getInstance(), Duration.EndOfGame,
|
||||||
|
StaticFilters.FILTER_PERMANENT_CREATURES
|
||||||
|
));
|
||||||
|
ability.addEffect(new GainAbilityControlledEffect(
|
||||||
|
HasteAbility.getInstance(), Duration.EndOfGame,
|
||||||
|
StaticFilters.FILTER_PERMANENT_CREATURES
|
||||||
|
).setText("and haste"));
|
||||||
|
ability.addEffect(new BoostControlledEffect(
|
||||||
|
EllywickTumblestrumEmblemValue.instance,
|
||||||
|
EllywickTumblestrumEmblemValue.instance,
|
||||||
|
Duration.EndOfGame
|
||||||
|
).setText("and get +2/+2 for each differently named dungeon you've completed"));
|
||||||
|
this.getAbilities().add(ability.addHint(EllywickTumblestrumEmblemHint.instance));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum EllywickTumblestrumEmblemValue implements DynamicValue {
|
||||||
|
instance;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public int calculate(Game game, Ability sourceAbility, Effect effect) {
|
||||||
|
return 2 * CompletedDungeonWatcher.getCompletedNames(sourceAbility.getControllerId(), game).size();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EllywickTumblestrumEmblemValue copy() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getMessage() {
|
||||||
|
return "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
enum EllywickTumblestrumEmblemHint implements Hint {
|
||||||
|
instance;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public String getText(Game game, Ability ability) {
|
||||||
|
Set<String> names = CompletedDungeonWatcher.getCompletedNames(ability.getControllerId(), game);
|
||||||
|
if (names.isEmpty()) {
|
||||||
|
return "No dungeons completed";
|
||||||
|
}
|
||||||
|
return "Completed dungeons: " + String.join(", ", names);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public EllywickTumblestrumEmblemHint copy() {
|
||||||
|
return this;
|
||||||
|
}
|
||||||
|
}
|
|
@ -448,6 +448,13 @@ public class GameEvent implements Serializable {
|
||||||
flag not used for this event
|
flag not used for this event
|
||||||
*/
|
*/
|
||||||
VOTE, VOTED,
|
VOTE, VOTED,
|
||||||
|
/* dungeons
|
||||||
|
targetId id of the room
|
||||||
|
sourceId sourceId of the ability causing player to venture
|
||||||
|
playerId player in the dungeon
|
||||||
|
*/
|
||||||
|
ROOM_ENTERED,
|
||||||
|
DUNGEON_COMPLETED,
|
||||||
//custom events
|
//custom events
|
||||||
CUSTOM_EVENT
|
CUSTOM_EVENT
|
||||||
}
|
}
|
||||||
|
|
|
@ -12,7 +12,7 @@ import mage.constants.Zone;
|
||||||
public final class SkeletonToken extends TokenImpl {
|
public final class SkeletonToken extends TokenImpl {
|
||||||
|
|
||||||
public SkeletonToken() {
|
public SkeletonToken() {
|
||||||
super("Skeleton", "1/1 black Skeleton creature with \"{B}: Regenerate this creature\"");
|
super("Skeleton", "1/1 black Skeleton creature token with \"{B}: Regenerate this creature\"");
|
||||||
cardType.add(CardType.CREATURE);
|
cardType.add(CardType.CREATURE);
|
||||||
this.subtype.add(SubType.SKELETON);
|
this.subtype.add(SubType.SKELETON);
|
||||||
color.setBlack(true);
|
color.setBlack(true);
|
||||||
|
|
|
@ -0,0 +1,28 @@
|
||||||
|
package mage.game.permanent.token;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class SkeletonToken2 extends TokenImpl {
|
||||||
|
|
||||||
|
public SkeletonToken2() {
|
||||||
|
super("Skeleton", "1/1 black Skeleton creature token");
|
||||||
|
cardType.add(CardType.CREATURE);
|
||||||
|
this.subtype.add(SubType.SKELETON);
|
||||||
|
color.setBlack(true);
|
||||||
|
power = new MageInt(1);
|
||||||
|
toughness = new MageInt(1);
|
||||||
|
}
|
||||||
|
|
||||||
|
public SkeletonToken2(final SkeletonToken2 token) {
|
||||||
|
super(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public SkeletonToken2 copy() {
|
||||||
|
return new SkeletonToken2(this);
|
||||||
|
}
|
||||||
|
}
|
|
@ -0,0 +1,33 @@
|
||||||
|
package mage.game.permanent.token;
|
||||||
|
|
||||||
|
import mage.MageInt;
|
||||||
|
import mage.abilities.keyword.DeathtouchAbility;
|
||||||
|
import mage.constants.CardType;
|
||||||
|
import mage.constants.SubType;
|
||||||
|
import mage.constants.SuperType;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public final class TheAtropalToken extends TokenImpl {
|
||||||
|
|
||||||
|
public TheAtropalToken() {
|
||||||
|
super("The Atropal", "The Atropal, a legendary 4/4 black God Horror creature token with deathtouch");
|
||||||
|
supertype.add(SuperType.LEGENDARY);
|
||||||
|
cardType.add(CardType.CREATURE);
|
||||||
|
color.setBlack(true);
|
||||||
|
subtype.add(SubType.GOD);
|
||||||
|
subtype.add(SubType.HORROR);
|
||||||
|
power = new MageInt(4);
|
||||||
|
toughness = new MageInt(4);
|
||||||
|
addAbility(DeathtouchAbility.getInstance());
|
||||||
|
}
|
||||||
|
|
||||||
|
public TheAtropalToken(final TheAtropalToken token) {
|
||||||
|
super(token);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TheAtropalToken copy() {
|
||||||
|
return new TheAtropalToken(this);
|
||||||
|
}
|
||||||
|
}
|
|
@ -1044,5 +1044,4 @@ public interface Player extends MageItem, Copyable<Player> {
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
FilterMana getPhyrexianColors();
|
FilterMana getPhyrexianColors();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -4726,5 +4726,4 @@ public abstract class PlayerImpl implements Player, Serializable {
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return getName() + " (" + super.getClass().getSimpleName() + ")";
|
return getName() + " (" + super.getClass().getSimpleName() + ")";
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,44 @@
|
||||||
|
package mage.watchers.common;
|
||||||
|
|
||||||
|
import mage.constants.WatcherScope;
|
||||||
|
import mage.game.Game;
|
||||||
|
import mage.game.events.GameEvent;
|
||||||
|
import mage.watchers.Watcher;
|
||||||
|
|
||||||
|
import java.util.*;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @author TheElk801
|
||||||
|
*/
|
||||||
|
public class CompletedDungeonWatcher extends Watcher {
|
||||||
|
|
||||||
|
private final Map<UUID, Set<String>> playerMap = new HashMap<>();
|
||||||
|
private static final Set<String> emptySet = new HashSet<>();
|
||||||
|
|
||||||
|
public CompletedDungeonWatcher() {
|
||||||
|
super(WatcherScope.GAME);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void watch(GameEvent event, Game game) {
|
||||||
|
switch (event.getType()) {
|
||||||
|
case DUNGEON_COMPLETED:
|
||||||
|
playerMap.computeIfAbsent(event.getPlayerId(), u -> new HashSet<>()).add(event.getData());
|
||||||
|
return;
|
||||||
|
case BEGINNING_PHASE_PRE:
|
||||||
|
if (game.getTurnNum() == 1) {
|
||||||
|
playerMap.clear();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean checkPlayer(UUID playerId, Game game) {
|
||||||
|
CompletedDungeonWatcher watcher = game.getState().getWatcher(CompletedDungeonWatcher.class);
|
||||||
|
return watcher != null && !watcher.playerMap.getOrDefault(playerId, emptySet).isEmpty();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Set<String> getCompletedNames(UUID playerId, Game game) {
|
||||||
|
CompletedDungeonWatcher watcher = game.getState().getWatcher(CompletedDungeonWatcher.class);
|
||||||
|
return watcher != null ? watcher.playerMap.getOrDefault(playerId, emptySet) : emptySet;
|
||||||
|
}
|
||||||
|
}
|
|
@ -41320,12 +41320,14 @@ Temple of the False God|Commander 2021|326|U||Land|||{T}: Add {C}{C}. Activate o
|
||||||
Temple of Triumph|Commander 2021|327|R||Land|||Temple of Triumph enters the battlefield tapped.$When Temple of Triumph enters the battlefield, scry 1.${T}: Add {R} or {W}.|
|
Temple of Triumph|Commander 2021|327|R||Land|||Temple of Triumph enters the battlefield tapped.$When Temple of Triumph enters the battlefield, scry 1.${T}: Add {R} or {W}.|
|
||||||
Tranquil Thicket|Commander 2021|408|C||Land|||Tranquil Thicket enters the battlefield tapped.${T}: Add {G}.$Cycling {G}|
|
Tranquil Thicket|Commander 2021|408|C||Land|||Tranquil Thicket enters the battlefield tapped.${T}: Add {G}.$Cycling {G}|
|
||||||
Yavimaya Coast|Commander 2021|409|R||Land|||{T}: Add {C}.${T}: Add {G} or {U}. Yavimaya Coast deals 1 damage to you.|
|
Yavimaya Coast|Commander 2021|409|R||Land|||{T}: Add {C}.${T}: Add {G} or {U}. Yavimaya Coast deals 1 damage to you.|
|
||||||
|
Cloister Gargoyle|Adventures in the Forgotten Realms|7|U|{2}{W}|Artifact Creature - Gargoyle|0|4|When Cloister Gargoyle enters the battlefield, venture into the dungeon.$As long as you've completed a dungeon, Cloister Gargoyle gets +3/+0 and has flying.|
|
||||||
Flumph|Adventures in the Forgotten Realms|15|R|{1}{W}|Creature - Jellyfish|0|4|Defender, flying$Whenever Flumph is dealt damage, you and target opponent each draw a card.|
|
Flumph|Adventures in the Forgotten Realms|15|R|{1}{W}|Creature - Jellyfish|0|4|Defender, flying$Whenever Flumph is dealt damage, you and target opponent each draw a card.|
|
||||||
Gloom Stalker|Adventures in the Forgotten Realms|16|C|{2}{W}|Creature - Dwarf Ranger|2|3|As long as you've completed a dungeon, Gloom Stalker has double strike.|
|
Gloom Stalker|Adventures in the Forgotten Realms|16|C|{2}{W}|Creature - Dwarf Ranger|2|3|As long as you've completed a dungeon, Gloom Stalker has double strike.|
|
||||||
Nadaar, Selfless Paladin|Adventures in the Forgotten Realms|27|R|{2}{W}|Legendary Creature - Dragon Knight|3|3|Vigilance$Whenever Nadaar, Selfless Paladin enters the battlefield or attacks, venture into the dungeon.$Other creatures you control get +1/+1 as long as you've completed a dungeon.|
|
Nadaar, Selfless Paladin|Adventures in the Forgotten Realms|27|R|{2}{W}|Legendary Creature - Dragon Knight|3|3|Vigilance$Whenever Nadaar, Selfless Paladin enters the battlefield or attacks, venture into the dungeon.$Other creatures you control get +1/+1 as long as you've completed a dungeon.|
|
||||||
Portable Hole|Adventures in the Forgotten Realms|33|U|{W}|Artifact|||When Portable Hole enters the battlefield, exile target nonland permanent an opponent controls with mana value 2 or less until Portable Hole leaves the battlefield.|
|
Portable Hole|Adventures in the Forgotten Realms|33|U|{W}|Artifact|||When Portable Hole enters the battlefield, exile target nonland permanent an opponent controls with mana value 2 or less until Portable Hole leaves the battlefield.|
|
||||||
Shortcut Seeker|Adventures in the Forgotten Realms|73|C|{3}{U}|Creature - Human Rogue|2|5|Whenever Shortcut Seeker deals combat damage to a player, venture into the dungeon.|
|
Shortcut Seeker|Adventures in the Forgotten Realms|73|C|{3}{U}|Creature - Human Rogue|2|5|Whenever Shortcut Seeker deals combat damage to a player, venture into the dungeon.|
|
||||||
Tasha's Hideous Laughter|Adventures in the Forgotten Realms|78|R|{1}{U}{U}|Sorcery|||Each opponent exiles cards from the top of their library until that player has exiled cards with total mana value 20 or more.|
|
Tasha's Hideous Laughter|Adventures in the Forgotten Realms|78|R|{1}{U}{U}|Sorcery|||Each opponent exiles cards from the top of their library until that player has exiled cards with total mana value 20 or more.|
|
||||||
|
Dungeon Crawler|Adventures in the Forgotten Realms|99|U|{B}|Creature - Zombie|2|1|Dungeon Crawler enters the battlefield tapped.$Whenever you complete a dungeon, you may return Dungeon Crawler from your graveyard to your hand.|
|
||||||
Lolth, Spider Queen|Adventures in the Forgotten Realms|112|M|{3}{B}{B}|Legendary Planeswalker - Lolth|4|Whenever a creature you control dies, put a loyalty counter on Lolth, Spider Queen.$0: You draw a card and you lose 1 life.$−3: Create two 2/1 black Spider creature tokens with menace and reach.$−8: You get an emblem with "Whenever an opponent is dealt combat damage by one or more creatures you control, if that player lost less than 8 life this turn, they lose life equal to the difference."|
|
Lolth, Spider Queen|Adventures in the Forgotten Realms|112|M|{3}{B}{B}|Legendary Planeswalker - Lolth|4|Whenever a creature you control dies, put a loyalty counter on Lolth, Spider Queen.$0: You draw a card and you lose 1 life.$−3: Create two 2/1 black Spider creature tokens with menace and reach.$−8: You get an emblem with "Whenever an opponent is dealt combat damage by one or more creatures you control, if that player lost less than 8 life this turn, they lose life equal to the difference."|
|
||||||
Power Word Kill|Adventures in the Forgotten Realms|114|U|{1}{B}|Instant|||Destroy target non-Angel, non-Demon, non-Devil, non-Dragon creature.|
|
Power Word Kill|Adventures in the Forgotten Realms|114|U|{1}{B}|Instant|||Destroy target non-Angel, non-Demon, non-Devil, non-Dragon creature.|
|
||||||
Vorpal Sword|Adventures in the Forgotten Realms|124|R|{B}|Artifact - Equipment|||Equipped creature gets +2/+0 and has deathtouch.${5}{B}{B}{B}: Until end of turn, Vorpal Sword gains "Whenever equipped creature deals combat damage to a player, that player loses the game."$Equip {B}{B}|
|
Vorpal Sword|Adventures in the Forgotten Realms|124|R|{B}|Artifact - Equipment|||Equipped creature gets +2/+0 and has deathtouch.${5}{B}{B}{B}: Until end of turn, Vorpal Sword gains "Whenever equipped creature deals combat damage to a player, that player loses the game."$Equip {B}{B}|
|
||||||
|
|
Loading…
Reference in a new issue