/* * Copyright 2007 (C) Thomas Parker <thpr@users.sourceforge.net> * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA */ package plugin.lsttokens.auto; import java.util.ArrayList; import java.util.Collection; import java.util.List; import java.util.Set; import java.util.StringTokenizer; import java.util.TreeSet; import pcgen.cdom.base.CDOMObject; import pcgen.cdom.base.CDOMReference; import pcgen.cdom.base.ChooseDriver; import pcgen.cdom.base.ChooseSelectionActor; import pcgen.cdom.base.Constants; import pcgen.cdom.content.ConditionalSelectionActor; import pcgen.cdom.enumeration.ListKey; import pcgen.cdom.helper.ShieldProfProvider; import pcgen.core.Equipment; import pcgen.core.PlayerCharacter; import pcgen.core.ShieldProf; import pcgen.core.prereq.Prerequisite; import pcgen.persistence.PersistenceLayerException; import pcgen.persistence.lst.prereq.PreParserFactory; import pcgen.rules.context.Changes; import pcgen.rules.context.LoadContext; import pcgen.rules.persistence.TokenUtilities; import pcgen.rules.persistence.token.AbstractNonEmptyToken; import pcgen.rules.persistence.token.CDOMSecondaryToken; import pcgen.rules.persistence.token.ParseResult; import pcgen.util.Logging; public class ShieldProfToken extends AbstractNonEmptyToken<CDOMObject> implements CDOMSecondaryToken<CDOMObject>, ChooseSelectionActor<ShieldProf> { private static final Class<ShieldProf> SHIELDPROF_CLASS = ShieldProf.class; private static final Class<Equipment> EQUIPMENT_CLASS = Equipment.class; @Override public String getParentToken() { return "AUTO"; } @Override public String getTokenName() { return "SHIELDPROF"; } private String getFullName() { return getParentToken() + Constants.COLON + getTokenName(); } @Override protected ParseResult parseNonEmptyToken(LoadContext context, CDOMObject obj, String value) { String shieldProf; Prerequisite prereq = null; // Do not initialize, null is significant! boolean isPre = false; if (value.indexOf("[") == -1) { // Supported version of PRExxx using |. Needs to be at the front of the // Parsing code because many objects expect the pre to have been determined // Ahead of time. Until deprecated code is removed, it will have to stay // like this. shieldProf = value; StringTokenizer tok = new StringTokenizer(shieldProf, Constants.PIPE); while (tok.hasMoreTokens()) { String token = tok.nextToken(); if (PreParserFactory.isPreReqString(token)) { if (isPre) { String errorText = "Invalid " + getTokenName() + ": " + value + " PRExxx must be at the END of the Token"; Logging.errorPrint(errorText); return new ParseResult.Fail(errorText, context); } prereq = getPrerequisite(token); if (prereq == null) { return new ParseResult.Fail("Error generating Prerequisite " + prereq + " in " + getFullName(), context); } int preStart = value.indexOf(token) - 1; shieldProf = value.substring(0, preStart); isPre = true; } } } else { return new ParseResult.Fail( "Use of [] for Prerequisites has been removed. " + "Please use | based standard", context); } ParseResult pr = checkForIllegalSeparator('|', shieldProf); if (!pr.passed()) { return pr; } boolean foundAny = false; boolean foundOther = false; StringTokenizer tok = new StringTokenizer(shieldProf, Constants.PIPE); List<CDOMReference<ShieldProf>> shieldProfs = new ArrayList<>(); List<CDOMReference<Equipment>> equipTypes = new ArrayList<>(); while (tok.hasMoreTokens()) { String aProf = tok.nextToken(); if (Constants.LST_PERCENT_LIST.equals(aProf)) { foundOther = true; ChooseSelectionActor<ShieldProf> cra; if (prereq == null) { cra = this; } else { ConditionalSelectionActor<ShieldProf> cca = new ConditionalSelectionActor<>(this); cca.addPrerequisite(prereq); cra = cca; } context.getObjectContext().addToList(obj, ListKey.NEW_CHOOSE_ACTOR, cra); } else if (Constants.LST_ALL.equalsIgnoreCase(aProf)) { foundAny = true; shieldProfs.add(context.getReferenceContext() .getCDOMAllReference(SHIELDPROF_CLASS)); } else if (aProf.startsWith("SHIELDTYPE.") || aProf.startsWith("SHIELDTYPE=")) { foundOther = true; CDOMReference<Equipment> ref = TokenUtilities.getTypeReference( context, EQUIPMENT_CLASS, "SHIELD." + aProf.substring(11)); if (ref == null) { return ParseResult.INTERNAL_ERROR; } equipTypes.add(ref); } else { foundOther = true; shieldProfs.add(context.getReferenceContext().getCDOMReference(SHIELDPROF_CLASS, aProf)); } } if (foundAny && foundOther) { return new ParseResult.Fail("Non-sensical " + getFullName() + ": Contains ANY and a specific reference: " + value, context); } if (!shieldProfs.isEmpty() || !equipTypes.isEmpty()) { ShieldProfProvider pp = new ShieldProfProvider(shieldProfs, equipTypes); if (prereq != null) { pp.addPrerequisite(prereq); } context.getObjectContext().addToList(obj, ListKey.AUTO_SHIELDPROF, pp); } return ParseResult.SUCCESS; } @Override public String[] unparse(LoadContext context, CDOMObject obj) { Changes<ShieldProfProvider> changes = context.getObjectContext().getListChanges(obj, ListKey.AUTO_SHIELDPROF); Changes<ChooseSelectionActor<?>> listChanges = context.getObjectContext() .getListChanges(obj, ListKey.NEW_CHOOSE_ACTOR); Collection<ShieldProfProvider> added = changes.getAdded(); Set<String> set = new TreeSet<>(); Collection<ChooseSelectionActor<?>> listAdded = listChanges.getAdded(); boolean foundAny = false; boolean foundOther = false; if (listAdded != null && !listAdded.isEmpty()) { for (ChooseSelectionActor<?> cra : listAdded) { if (cra.getSource().equals(getTokenName())) { try { set.add(cra.getLstFormat()); foundOther = true; } catch (PersistenceLayerException e) { context.addWriteMessage("Error writing Prerequisite: " + e); return null; } } } } if (added != null) { for (ShieldProfProvider spp : added) { StringBuilder sb = new StringBuilder(); sb.append(spp.getLstFormat()); if (spp.hasPrerequisites()) { sb.append('|'); sb.append(getPrerequisiteString(context, spp .getPrerequisiteList())); } String ab = sb.toString(); boolean isUnconditionalAll = Constants.LST_ALL.equals(ab); foundAny |= isUnconditionalAll; foundOther |= !isUnconditionalAll; set.add(ab); } } if (foundAny && foundOther) { context.addWriteMessage("Non-sensical " + getFullName() + ": Contains ANY and a specific reference: " + set); return null; } if (set.isEmpty()) { //okay return null; } return set.toArray(new String[set.size()]); } @Override public Class<CDOMObject> getTokenClass() { return CDOMObject.class; } @Override public void applyChoice(ChooseDriver obj, ShieldProf sp, PlayerCharacter pc) { pc.addShieldProf(obj, sp); } @Override public void removeChoice(ChooseDriver obj, ShieldProf sp, PlayerCharacter pc) { pc.removeShieldProf(obj, sp); } @Override public Class<ShieldProf> getChoiceClass() { return SHIELDPROF_CLASS; } @Override public String getSource() { return getTokenName(); } @Override public String getLstFormat() { return Constants.LST_PERCENT_LIST; } }