- Volrath's Shapeshifter: switched to a less aggressive update schedule (no update unless necessary), which fixes interaction with composite triggers that consist of several related parts (e.g. Undying, Persist).

- Volrath's Shapeshifter: QoL update: do not show the same text and discard ability twice if Volrath's Shapeshifter is attempting to copy the text of another Volrath's Shapeshifter that is on top of the graveyard.

git-svn-id: http://svn.slightlymagic.net/forge/trunk@35779 269b9781-a132-4a9b-9d4e-f004f1b56b58
This commit is contained in:
Agetian 2017-09-26 13:55:28 +00:00
parent 258c46868a
commit f12c49f712
2 changed files with 73 additions and 56 deletions

View File

@ -203,6 +203,8 @@ public class GameAction {
} }
if (c.getStates().contains(CardStateName.OriginalText)) { if (c.getStates().contains(CardStateName.OriginalText)) {
c.clearStates(CardStateName.OriginalText, false); c.clearStates(CardStateName.OriginalText, false);
c.removeSVar("GainingTextFrom");
c.removeSVar("GainingTextFromTimestamp");
} }
c.updateStateForView(); c.updateStateForView();
} else if (c.getStates().contains(CardStateName.OriginalText)) { } else if (c.getStates().contains(CardStateName.OriginalText)) {
@ -210,6 +212,8 @@ public class GameAction {
CardFactory.copyState(c, CardStateName.OriginalText, c, CardStateName.Original, false); CardFactory.copyState(c, CardStateName.OriginalText, c, CardStateName.Original, false);
c.setState(CardStateName.Original, false); c.setState(CardStateName.Original, false);
c.clearStates(CardStateName.OriginalText, false); c.clearStates(CardStateName.OriginalText, false);
c.removeSVar("GainingTextFrom");
c.removeSVar("GainingTextFromTimestamp");
c.updateStateForView(); c.updateStateForView();
} }

View File

@ -17,21 +17,10 @@
*/ */
package forge.game.staticability; package forge.game.staticability;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Map;
import java.util.Set;
import forge.card.CardStateName;
import forge.game.card.*;
import forge.util.TextUtil;
import org.apache.commons.lang3.StringUtils;
import com.google.common.collect.Iterables; import com.google.common.collect.Iterables;
import com.google.common.collect.Lists; import com.google.common.collect.Lists;
import forge.GameCommand; import forge.GameCommand;
import forge.card.CardStateName;
import forge.card.MagicColor; import forge.card.MagicColor;
import forge.game.Game; import forge.game.Game;
import forge.game.GlobalRuleChange; import forge.game.GlobalRuleChange;
@ -40,6 +29,7 @@ import forge.game.StaticEffects;
import forge.game.ability.AbilityFactory; import forge.game.ability.AbilityFactory;
import forge.game.ability.AbilityUtils; import forge.game.ability.AbilityUtils;
import forge.game.ability.ApiType; import forge.game.ability.ApiType;
import forge.game.card.*;
import forge.game.cost.Cost; import forge.game.cost.Cost;
import forge.game.player.Player; import forge.game.player.Player;
import forge.game.replacement.ReplacementEffect; import forge.game.replacement.ReplacementEffect;
@ -50,6 +40,10 @@ import forge.game.spellability.SpellAbility;
import forge.game.trigger.Trigger; import forge.game.trigger.Trigger;
import forge.game.trigger.TriggerHandler; import forge.game.trigger.TriggerHandler;
import forge.game.zone.ZoneType; import forge.game.zone.ZoneType;
import forge.util.TextUtil;
import org.apache.commons.lang3.StringUtils;
import java.util.*;
/** /**
* The Class StaticAbility_Continuous. * The Class StaticAbility_Continuous.
@ -477,55 +471,74 @@ public final class StaticAbilityContinuous {
// Gain text from another card // Gain text from another card
if (layer == StaticAbilityLayer.TEXT) { if (layer == StaticAbilityLayer.TEXT) {
// Restore the original text in case it was remembered before // Make no changes in case the target for the ability is still the same as before
if (affectedCard.getStates().contains(CardStateName.OriginalText)) { boolean noChange = false;
affectedCard.clearTriggersNew(); if (gainTextSource != null && affectedCard.hasSVar("GainingTextFrom") && affectedCard.hasSVar("GainingTextFromTimestamp")
List<SpellAbility> saToRemove = Lists.newArrayList(); && gainTextSource.getName() == affectedCard.getSVar("GainingTextFrom")
for (SpellAbility saTemp : affectedCard.getSpellAbilities()) { && gainTextSource.getTimestamp() == Long.parseLong(affectedCard.getSVar("GainingTextFromTimestamp"))) {
if (saTemp.isTemporary()) { noChange = true;
saToRemove.add(saTemp);
}
}
for (SpellAbility saRem : saToRemove) {
affectedCard.removeSpellAbility(saRem);
}
CardFactory.copyState(affectedCard, CardStateName.OriginalText, affectedCard, CardStateName.Original, false);
} }
if (gainTextSource != null) { if (!noChange) {
if (!affectedCard.getStates().contains(CardStateName.OriginalText)) { // Restore the original text in case it was remembered before
// Remember the original text first in case it hasn't been done yet if (affectedCard.getStates().contains(CardStateName.OriginalText)) {
CardFactory.copyState(affectedCard, CardStateName.Original, affectedCard, CardStateName.OriginalText, false); affectedCard.clearTriggersNew();
} List<SpellAbility> saToRemove = Lists.newArrayList();
for (SpellAbility saTemp : affectedCard.getSpellAbilities()) {
CardFactory.copyState(gainTextSource, CardStateName.Original, affectedCard, CardStateName.Original, false); if (saTemp.isTemporary()) {
saToRemove.add(saTemp);
// Do not clone the set code and rarity from the target card }
affectedCard.getState(CardStateName.Original).setSetCode(affectedCard.getState(CardStateName.OriginalText).getSetCode());
affectedCard.getState(CardStateName.Original).setRarity(affectedCard.getState(CardStateName.OriginalText).getRarity());
// Enable this in case Volrath's original image is to be used
affectedCard.getState(CardStateName.Original).setImageKey(affectedCard.getState(CardStateName.OriginalText).getImageKey());
// Activated abilities (statics and repleffects are apparently copied vis copyState?)
for (SpellAbility sa : gainTextSource.getSpellAbilities()) {
if (sa instanceof AbilityActivated) {
SpellAbility newSA = ((AbilityActivated) sa).getCopy();
newSA.setOriginalHost(gainTextSource);
newSA.setIntrinsic(false);
newSA.setTemporary(true);
newSA.setHostCard(affectedCard);
affectedCard.addSpellAbility(newSA);
} }
} for (SpellAbility saRem : saToRemove) {
// Triggered abilities affectedCard.removeSpellAbility(saRem);
for (Trigger t: gainTextSource.getTriggers()) { }
affectedCard.addTrigger(t.getCopyForHostCard(affectedCard)); CardFactory.copyState(affectedCard, CardStateName.OriginalText, affectedCard, CardStateName.Original, false);
} }
// Volrath's Shapeshifter shapeshifting ability needs to be added onto the new text // TODO: find a better way to ascertain that the card will essentially try to copy its exact duplicate
if (params.containsKey("GainedTextHasThisStaticAbility")) { // (e.g. Volrath's Shapeshifter copying the text of another pristine Volrath's Shapeshifter), since the
affectedCard.getCurrentState().addStaticAbility(stAb); // check by name may fail in case one of the cards is modified in some way while the other is not
// (probably not very relevant for Volrath's Shapeshifter itself since it copies text on cards in GY).
if (gainTextSource != null && !gainTextSource.getCurrentState().getName().equals(affectedCard.getCurrentState().getName())) {
if (!affectedCard.getStates().contains(CardStateName.OriginalText)) {
// Remember the original text first in case it hasn't been done yet
CardFactory.copyState(affectedCard, CardStateName.Original, affectedCard, CardStateName.OriginalText, false);
}
CardFactory.copyState(gainTextSource, CardStateName.Original, affectedCard, CardStateName.Original, false);
// Do not clone the set code and rarity from the target card
affectedCard.getState(CardStateName.Original).setSetCode(affectedCard.getState(CardStateName.OriginalText).getSetCode());
affectedCard.getState(CardStateName.Original).setRarity(affectedCard.getState(CardStateName.OriginalText).getRarity());
// Enable this in case Volrath's original image is to be used
affectedCard.getState(CardStateName.Original).setImageKey(affectedCard.getState(CardStateName.OriginalText).getImageKey());
// Activated abilities (statics and repleffects are apparently copied via copyState?)
for (SpellAbility sa : gainTextSource.getSpellAbilities()) {
if (sa instanceof AbilityActivated) {
SpellAbility newSA = ((AbilityActivated) sa).getCopy();
newSA.setOriginalHost(gainTextSource);
newSA.setIntrinsic(false);
newSA.setTemporary(true);
newSA.setHostCard(affectedCard);
affectedCard.addSpellAbility(newSA);
}
}
// Triggered abilities
for (Trigger t : gainTextSource.getTriggers()) {
affectedCard.addTrigger(t.getCopyForHostCard(affectedCard));
}
// Volrath's Shapeshifter shapeshifting ability needs to be added onto the new text
if (params.containsKey("GainedTextHasThisStaticAbility")) {
affectedCard.getCurrentState().addStaticAbility(stAb);
}
// Remember the name and the timestamp of the card we're gaining text from, so we don't modify
// the card too aggressively when unnecessary
affectedCard.setSVar("GainingTextFrom", String.valueOf(gainTextSource.getName()));
affectedCard.setSVar("GainingTextFromTimestamp", String.valueOf(gainTextSource.getTimestamp()));
} }
} }
} }