/* * Copyright 2008 (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; import java.util.Arrays; import java.util.Collection; import java.util.List; import java.util.Set; import java.util.StringTokenizer; import java.util.TreeSet; import pcgen.base.util.HashMapToList; import pcgen.cdom.base.CDOMObject; import pcgen.cdom.base.Category; import pcgen.cdom.base.Constants; import pcgen.cdom.base.Loadable; import pcgen.cdom.base.Ungranted; import pcgen.cdom.enumeration.ListKey; import pcgen.cdom.inst.PCClassLevel; import pcgen.cdom.reference.CDOMSingleRef; import pcgen.cdom.reference.CategorizedCDOMReference; import pcgen.cdom.reference.Qualifier; import pcgen.cdom.reference.ReferenceManufacturer; import pcgen.cdom.reference.ReferenceUtilities; import pcgen.core.Ability; import pcgen.core.Deity; import pcgen.core.Domain; import pcgen.core.Equipment; import pcgen.core.PCClass; import pcgen.core.PCTemplate; import pcgen.core.Race; import pcgen.core.Skill; import pcgen.core.WeaponProf; import pcgen.core.spell.Spell; import pcgen.rules.context.Changes; import pcgen.rules.context.LoadContext; import pcgen.rules.persistence.token.AbstractTokenWithSeparator; import pcgen.rules.persistence.token.CDOMPrimaryToken; import pcgen.rules.persistence.token.ParseResult; import pcgen.util.StringPClassUtil; /** * Deals with the QUALIFY token for Abilities */ public class QualifyToken extends AbstractTokenWithSeparator<CDOMObject> implements CDOMPrimaryToken<CDOMObject> { @Override public String getTokenName() { return "QUALIFY"; } public List<Class<? extends CDOMObject>> getLegalTypes() { return Arrays.asList(PCClassLevel.class, Ability.class, Deity.class, Domain.class, Equipment.class, PCClass.class, Race.class, Skill.class, Spell.class, PCTemplate.class, WeaponProf.class); } @Override protected ParseResult parseNonEmptyToken(LoadContext context, CDOMObject obj, String value) { if (obj instanceof Ungranted) { return new ParseResult.Fail("Cannot use " + getTokenName() + " on an Ungranted object type: " + obj.getClass().getSimpleName(), context); } if (!getLegalTypes().contains(obj.getClass())) { return new ParseResult.Fail("Cannot use QUALIFY on a " + obj.getClass(), context); } return super.parseNonEmptyToken(context, obj, value); } @Override protected char separator() { return '|'; } @Override protected ParseResult parseTokenWithSeparator(LoadContext context, CDOMObject obj, String value) { if (value.indexOf(Constants.PIPE) == -1) { return new ParseResult.Fail(getTokenName() + " requires at least two arguments, QualifyType and Key: " + value, context); } StringTokenizer st = new StringTokenizer(value, Constants.PIPE); String firstToken = st.nextToken(); ReferenceManufacturer<? extends Loadable> rm = context.getManufacturer(firstToken); if (rm == null) { return new ParseResult.Fail(getTokenName() + " unable to generate manufacturer for type: " + value, context); } while (st.hasMoreTokens()) { CDOMSingleRef<? extends Loadable> ref = rm.getReference(st .nextToken()); context.getObjectContext().addToList(obj, ListKey.QUALIFY, new Qualifier(rm .getReferenceClass(), ref)); } return ParseResult.SUCCESS; } @Override public String[] unparse(LoadContext context, CDOMObject obj) { Changes<Qualifier> changes = context.getObjectContext().getListChanges( obj, ListKey.QUALIFY); if (changes == null || changes.isEmpty()) { return null; } Collection<Qualifier> quals = changes.getAdded(); HashMapToList<String, CDOMSingleRef<?>> map = new HashMapToList<>(); for (Qualifier qual : quals) { Class<? extends Loadable> cl = qual.getQualifiedClass(); String s = StringPClassUtil.getStringFor(cl); CDOMSingleRef<?> ref = qual.getQualifiedReference(); String key = s; if (ref instanceof CategorizedCDOMReference) { Category<?> cat = ((CategorizedCDOMReference<?>) ref) .getCDOMCategory(); key += '=' + cat.getKeyName(); } map.addToListFor(key, ref); } Set<CDOMSingleRef<?>> set = new TreeSet<>( ReferenceUtilities.REFERENCE_SORTER); Set<String> returnSet = new TreeSet<>(); for (String key : map.getKeySet()) { set.clear(); set.addAll(map.getListFor(key)); StringBuilder sb = new StringBuilder(); sb.append(key).append(Constants.PIPE).append( ReferenceUtilities.joinLstFormat(set, Constants.PIPE)); returnSet.add(sb.toString()); } return returnSet.toArray(new String[returnSet.size()]); } @Override public Class<CDOMObject> getTokenClass() { return CDOMObject.class; } }