mirror of
https://github.com/magefree/mage.git
synced 2026-01-09 20:32:06 -08:00
Tokens reworked:
- removed outdated code; - updated logic to choose a set code for a tokens in different use cases (related to #10150); - added many tests for client and server token's data (related to #10139); - prepare for tokens database (related #6955);
This commit is contained in:
parent
ff15edbce8
commit
d17df585c5
13 changed files with 409 additions and 89 deletions
|
|
@ -18,4 +18,8 @@ public interface CommandObject extends MageObject, Controllable {
|
|||
|
||||
@Override
|
||||
CommandObject copy();
|
||||
|
||||
String getExpansionSetCodeForImage();
|
||||
|
||||
void setExpansionSetCodeForImage(String expansionSetCodeForImage);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -282,6 +282,16 @@ public class Commander implements CommandObject {
|
|||
return sourceObject.getImageName();
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getExpansionSetCodeForImage() {
|
||||
return sourceObject.getExpansionSetCode();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setExpansionSetCodeForImage(String expansionSetCodeForImage) {
|
||||
throw new IllegalStateException("Can't change a set code of the commander, source card already has it");
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getZoneChangeCounter(Game game) {
|
||||
return sourceObject.getZoneChangeCounter(game);
|
||||
|
|
|
|||
|
|
@ -327,10 +327,12 @@ public class Dungeon implements CommandObject {
|
|||
return new Dungeon(this);
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getExpansionSetCodeForImage() {
|
||||
return expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setExpansionSetCodeForImage(String expansionSetCodeForImage) {
|
||||
this.expansionSetCodeForImage = expansionSetCodeForImage;
|
||||
}
|
||||
|
|
|
|||
|
|
@ -243,14 +243,16 @@ public class Emblem implements CommandObject {
|
|||
return new Emblem(this);
|
||||
}
|
||||
|
||||
public void setExpansionSetCodeForImage(String expansionSetCodeForImage) {
|
||||
this.expansionSetCodeForImage = expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getExpansionSetCodeForImage() {
|
||||
return expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setExpansionSetCodeForImage(String expansionSetCodeForImage) {
|
||||
this.expansionSetCodeForImage = expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getZoneChangeCounter(Game game) {
|
||||
return 1; // Emblems can't move zones until now so return always 1
|
||||
|
|
|
|||
|
|
@ -242,14 +242,16 @@ public class Plane implements CommandObject {
|
|||
return new Plane(this);
|
||||
}
|
||||
|
||||
public void setExpansionSetCodeForImage(String expansionSetCodeForImage) {
|
||||
this.expansionSetCodeForImage = expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getExpansionSetCodeForImage() {
|
||||
return expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setExpansionSetCodeForImage(String expansionSetCodeForImage) {
|
||||
this.expansionSetCodeForImage = expansionSetCodeForImage;
|
||||
}
|
||||
|
||||
@Override
|
||||
public int getZoneChangeCounter(Game game) {
|
||||
return 1; // Emblems can't move zones until now so return always 1
|
||||
|
|
|
|||
|
|
@ -1703,8 +1703,8 @@ public abstract class PermanentImpl extends CardImpl implements Permanent {
|
|||
}
|
||||
|
||||
@Override
|
||||
public void setCardNumber(String cid) {
|
||||
this.cardNumber = cid;
|
||||
public void setCardNumber(String cardNumber) {
|
||||
this.cardNumber = cardNumber;
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
|
|||
|
|
@ -121,4 +121,24 @@ public class PermanentToken extends PermanentImpl {
|
|||
// token don't have game card, so return itself
|
||||
return this;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getCardNumber() {
|
||||
return token.getOriginalCardNumber();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setCardNumber(String cardNumber) {
|
||||
throw new IllegalArgumentException("Wrong code usage: you can't change a token's card number");
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getExpansionSetCode() {
|
||||
return token.getOriginalExpansionSetCode();
|
||||
}
|
||||
|
||||
@Override
|
||||
public void setExpansionSetCode(String expansionSetCode) {
|
||||
throw new IllegalArgumentException("Wrong code usage: you can't change a token's set code");
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -58,6 +58,4 @@ public interface Token extends MageObject {
|
|||
void setCopySourceCard(Card copySourceCard);
|
||||
|
||||
void setExpansionSetCodeForImage(String code);
|
||||
|
||||
boolean updateExpansionSetCode(String setCode);
|
||||
}
|
||||
|
|
|
|||
|
|
@ -13,6 +13,7 @@ import mage.constants.Outcome;
|
|||
import mage.constants.SubType;
|
||||
import mage.constants.Zone;
|
||||
import mage.game.Game;
|
||||
import mage.game.command.CommandObject;
|
||||
import mage.game.events.CreateTokenEvent;
|
||||
import mage.game.events.CreatedTokenEvent;
|
||||
import mage.game.events.ZoneChangeEvent;
|
||||
|
|
@ -34,8 +35,6 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
private int tokenType;
|
||||
private String originalCardNumber;
|
||||
private String originalExpansionSetCode;
|
||||
private String tokenDescriptor;
|
||||
private boolean expansionSetCodeChecked;
|
||||
private Card copySourceCard; // the card the Token is a copy from
|
||||
private static final int MAX_TOKENS_PER_GAME = 500;
|
||||
|
||||
|
|
@ -73,7 +72,6 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
this.lastAddedTokenIds.addAll(token.lastAddedTokenIds);
|
||||
this.originalCardNumber = token.originalCardNumber;
|
||||
this.originalExpansionSetCode = token.originalExpansionSetCode;
|
||||
this.expansionSetCodeChecked = token.expansionSetCodeChecked;
|
||||
this.copySourceCard = token.copySourceCard; // will never be changed
|
||||
this.availableImageSetCodes = token.availableImageSetCodes;
|
||||
}
|
||||
|
|
@ -81,20 +79,6 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
@Override
|
||||
public abstract Token copy();
|
||||
|
||||
private void setTokenDescriptor() {
|
||||
this.tokenDescriptor = tokenDescriptor();
|
||||
}
|
||||
|
||||
private String tokenDescriptor() {
|
||||
String strName = this.name.replaceAll("[^a-zA-Z0-9]", "");
|
||||
String strColor = this.color.toString().replaceAll("[^a-zA-Z0-9]", "");
|
||||
String strSubtype = this.subtype.toString().replaceAll("[^a-zA-Z0-9]", "");
|
||||
String strCardType = this.cardType.toString().replaceAll("[^a-zA-Z0-9]", "");
|
||||
String descriptor = strName + '.' + strColor + '.' + strSubtype + '.' + strCardType + '.' + this.power + '.' + this.toughness;
|
||||
descriptor = descriptor.toUpperCase(Locale.ENGLISH);
|
||||
return descriptor;
|
||||
}
|
||||
|
||||
@Override
|
||||
public String getDescription() {
|
||||
return description;
|
||||
|
|
@ -156,28 +140,41 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
return putOntoBattlefield(amount, game, source, controllerId, tapped, attacking, null);
|
||||
}
|
||||
|
||||
private String getSetCode(Game game, UUID sourceId) {
|
||||
// moved here from CreateTokenEffect because not all cards that create tokens use CreateTokenEffect
|
||||
// they use putOntoBattlefield directly
|
||||
// TODO: Check this setCode handling because it makes no sense if token put into play with e.g. "Feldon of the third Path"
|
||||
public static String generateSetCode(TokenImpl token, Game game, UUID sourceId) {
|
||||
// Choose a set code's by priority:
|
||||
// - use source's code
|
||||
// - use parent source's code (too complicated, so ignore it)
|
||||
// - use random set code
|
||||
// - use default set code
|
||||
// Token type must be set on set's code update
|
||||
|
||||
// source
|
||||
String setCode = null;
|
||||
if (this.getOriginalExpansionSetCode() != null && !this.getOriginalExpansionSetCode().isEmpty()) {
|
||||
setCode = this.getOriginalExpansionSetCode();
|
||||
} else {
|
||||
Card source = game.getCard(sourceId);
|
||||
if (source != null) {
|
||||
setCode = source.getExpansionSetCode();
|
||||
} else {
|
||||
MageObject object = game.getObject(sourceId);
|
||||
if (object instanceof PermanentToken) {
|
||||
setCode = ((PermanentToken) object).getExpansionSetCode();
|
||||
}
|
||||
}
|
||||
Card sourceCard = game.getCard(sourceId);
|
||||
if (sourceCard != null) {
|
||||
setCode = sourceCard.getExpansionSetCode();
|
||||
}
|
||||
MageObject sourceObject = game.getObject(sourceId);
|
||||
if (sourceObject instanceof CommandObject) {
|
||||
setCode = ((CommandObject) sourceObject).getExpansionSetCodeForImage();
|
||||
}
|
||||
|
||||
if (!expansionSetCodeChecked) {
|
||||
expansionSetCodeChecked = this.updateExpansionSetCode(setCode);
|
||||
// TODO: change to tokens database
|
||||
if (token.availableImageSetCodes.contains(setCode)) {
|
||||
return setCode;
|
||||
}
|
||||
|
||||
// random
|
||||
if (!token.availableImageSetCodes.isEmpty()) {
|
||||
return token.availableImageSetCodes.get(RandomUtil.nextInt(token.availableImageSetCodes.size()));
|
||||
}
|
||||
|
||||
// default
|
||||
// TODO: implement
|
||||
if (setCode == null) {
|
||||
setCode = "DEFAULT";
|
||||
}
|
||||
|
||||
return setCode;
|
||||
}
|
||||
|
||||
|
|
@ -242,14 +239,19 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
for (Map.Entry<Token, Integer> entry : event.getTokens().entrySet()) {
|
||||
Token token = entry.getKey();
|
||||
int amount = entry.getValue();
|
||||
String setCode = token instanceof TokenImpl ? ((TokenImpl) token).getSetCode(game, event.getSourceId()) : null;
|
||||
|
||||
// choose token's set code due source
|
||||
String setCode = TokenImpl.generateSetCode((TokenImpl) token, game, source == null ? null : source.getSourceId());
|
||||
token.setOriginalExpansionSetCode(setCode);
|
||||
token.setExpansionSetCodeForImage(setCode);
|
||||
|
||||
List<Permanent> needTokens = new ArrayList<>();
|
||||
List<Permanent> allowedTokens = new ArrayList<>();
|
||||
|
||||
// prepare tokens to enter
|
||||
for (int i = 0; i < amount; i++) {
|
||||
// use event.getPlayerId() as controller cause it can be replaced by replacement effect
|
||||
// TODO: add random setTokenType here?
|
||||
// use event.getPlayerId() as controller because it can be replaced by replacement effect
|
||||
PermanentToken newPermanent = new PermanentToken(token, event.getPlayerId(), setCode, game);
|
||||
game.getState().addCard(newPermanent);
|
||||
needTokens.add(newPermanent);
|
||||
|
|
@ -257,7 +259,7 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
newPermanent.setTapped(tapped);
|
||||
|
||||
ZoneChangeEvent emptyEvent = new ZoneChangeEvent(newPermanent, newPermanent.getControllerId(), Zone.OUTSIDE, Zone.BATTLEFIELD);
|
||||
// tokens zcc must simulate card's zcc too keep copied card/spell settings
|
||||
// tokens zcc must simulate card's zcc to keep copied card/spell settings
|
||||
// (example: etb's kicker ability of copied creature spell, see tests with Deathforge Shaman)
|
||||
newPermanent.updateZoneChangeCounter(game, emptyEvent);
|
||||
}
|
||||
|
|
@ -410,10 +412,10 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
|
||||
@Override
|
||||
public void setOriginalExpansionSetCode(String originalExpansionSetCode) {
|
||||
// TODO: delete
|
||||
// TODO: remove original set code at all... token image must be takes by card source or by latest set (on null source)
|
||||
// TODO: if set have same tokens then selects it by random
|
||||
this.originalExpansionSetCode = originalExpansionSetCode;
|
||||
setTokenDescriptor();
|
||||
}
|
||||
|
||||
@Override
|
||||
|
|
@ -430,28 +432,7 @@ public abstract class TokenImpl extends MageObjectImpl implements Token {
|
|||
|
||||
@Override
|
||||
public void setExpansionSetCodeForImage(String code) {
|
||||
if (!availableImageSetCodes.isEmpty()) {
|
||||
if (availableImageSetCodes.contains(code)) {
|
||||
setOriginalExpansionSetCode(code);
|
||||
} else // we should not set random set if appropriate set is already used
|
||||
{
|
||||
if (getOriginalExpansionSetCode() == null || getOriginalExpansionSetCode().isEmpty()
|
||||
|| !availableImageSetCodes.contains(getOriginalExpansionSetCode())) {
|
||||
setOriginalExpansionSetCode(availableImageSetCodes.get(RandomUtil.nextInt(availableImageSetCodes.size())));
|
||||
}
|
||||
}
|
||||
} else if (getOriginalExpansionSetCode() == null || getOriginalExpansionSetCode().isEmpty()) {
|
||||
setOriginalExpansionSetCode(code);
|
||||
}
|
||||
setTokenDescriptor();
|
||||
}
|
||||
|
||||
@Override
|
||||
public boolean updateExpansionSetCode(String setCode) {
|
||||
if (setCode == null || setCode.isEmpty()) {
|
||||
return false;
|
||||
}
|
||||
this.setExpansionSetCodeForImage(setCode);
|
||||
return true;
|
||||
// TODO: delete
|
||||
setOriginalExpansionSetCode(code);
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -561,7 +561,7 @@ public final class CardUtil {
|
|||
*/
|
||||
public static int parseCardNumberAsInt(String cardNumber) {
|
||||
|
||||
if (cardNumber.isEmpty()) {
|
||||
if (cardNumber == null || cardNumber.isEmpty()) {
|
||||
throw new IllegalArgumentException("Card number is empty.");
|
||||
}
|
||||
|
||||
|
|
|
|||
|
|
@ -36,12 +36,14 @@ public class CopyTokenFunction implements Function<Token, Card> {
|
|||
|
||||
MageObject sourceObj = source;
|
||||
if (source instanceof PermanentToken) {
|
||||
// create token from another token
|
||||
sourceObj = ((PermanentToken) source).getToken();
|
||||
// to show the source image, the original values have to be used
|
||||
target.setOriginalExpansionSetCode(((Token) sourceObj).getOriginalExpansionSetCode());
|
||||
target.setOriginalCardNumber(((Token) sourceObj).getOriginalCardNumber());
|
||||
target.setCopySourceCard(((PermanentToken) source).getToken().getCopySourceCard());
|
||||
} else if (source instanceof PermanentCard) {
|
||||
// create token from non-token permanent
|
||||
if (((PermanentCard) source).isMorphed() || ((PermanentCard) source).isManifested()) {
|
||||
MorphAbility.setPermanentToFaceDownCreature(target, game);
|
||||
return target;
|
||||
|
|
@ -57,6 +59,7 @@ public class CopyTokenFunction implements Function<Token, Card> {
|
|||
target.setCopySourceCard((Card) sourceObj);
|
||||
}
|
||||
} else {
|
||||
// create token from non-permanent object like card (example: Embalm ability)
|
||||
target.setOriginalExpansionSetCode(source.getExpansionSetCode());
|
||||
target.setOriginalCardNumber(source.getCardNumber());
|
||||
target.setCopySourceCard(source);
|
||||
|
|
|
|||
Loading…
Add table
Add a link
Reference in a new issue