/* * 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.ArrayList; import java.util.Collection; import java.util.List; import java.util.StringTokenizer; import pcgen.base.lang.StringUtil; import pcgen.cdom.base.CDOMObject; import pcgen.cdom.base.Constants; import pcgen.cdom.enumeration.ListKey; import pcgen.core.Description; import pcgen.core.prereq.Prerequisite; import pcgen.io.EntityEncoder; import pcgen.rules.context.LoadContext; import pcgen.rules.context.PatternChanges; import pcgen.rules.persistence.token.AbstractTokenWithSeparator; import pcgen.rules.persistence.token.CDOMPrimaryToken; import pcgen.rules.persistence.token.ParseResult; /** * Handles DESC token processing * * @author djones4 */ public class DescLst extends AbstractTokenWithSeparator<CDOMObject> implements CDOMPrimaryToken<CDOMObject> { /** * @see pcgen.persistence.lst.LstToken#getTokenName() */ @Override public String getTokenName() { return "DESC"; //$NON-NLS-1$ } @Override protected char separator() { return '|'; } @Override protected ParseResult parseTokenWithSeparator(LoadContext context, CDOMObject obj, String aDesc) { if (Constants.LST_DOT_CLEAR.equals(aDesc)) { context.getObjectContext().removeList(obj, ListKey.DESCRIPTION); return ParseResult.SUCCESS; } if (aDesc.startsWith(Constants.LST_DOT_CLEAR_DOT)) { context.getObjectContext().removePatternFromList(obj, ListKey.DESCRIPTION, aDesc.substring(7)); return ParseResult.SUCCESS; } StringTokenizer tok = new StringTokenizer(aDesc, Constants.PIPE); String descString = tok.nextToken(); if (looksLikeAPrerequisite(descString)) { return new ParseResult.Fail(getTokenName() + " encountered only a PRExxx: " + aDesc, context); } String ds = EntityEncoder.decode(descString); if (!StringUtil.hasBalancedParens(ds)) { return new ParseResult.Fail(getTokenName() + " encountered imbalanced Parenthesis: " + aDesc, context); } ParseResult pr = checkForInvalidXMLChars(ds); if (!pr.passed()) { return pr; } Description desc = new Description(ds); if (tok.hasMoreTokens()) { String token = tok.nextToken(); while (true) { if (Constants.LST_DOT_CLEAR.equals(token)) { return new ParseResult.Fail(getTokenName() + " tag confused by '.CLEAR' as a " + "middle token: " + aDesc, context); } else if (looksLikeAPrerequisite(token)) { break; } else { desc.addVariable(token); } if (!tok.hasMoreTokens()) { // No prereqs, so we're done context.getObjectContext().addToList(obj, ListKey.DESCRIPTION, desc); return ParseResult.SUCCESS; } token = tok.nextToken(); } while (true) { Prerequisite prereq = getPrerequisite(token); if (prereq == null) { return new ParseResult.Fail( " (Did you put Abilities after the " + "PRExxx tags in " + getTokenName() + ":?)", context); } desc.addPrerequisite(prereq); if (!tok.hasMoreTokens()) { break; } token = tok.nextToken(); } } context.getObjectContext().addToList(obj, ListKey.DESCRIPTION, desc); return ParseResult.SUCCESS; } @Override public String[] unparse(LoadContext context, CDOMObject obj) { PatternChanges<Description> changes = context.getObjectContext().getListPatternChanges(obj, ListKey.DESCRIPTION); if (changes == null || changes.isEmpty()) { return null; } List<String> list = new ArrayList<>(); Collection<String> removedItems = changes.getRemoved(); if (changes.includesGlobalClear()) { if (removedItems != null && !removedItems.isEmpty()) { context.addWriteMessage("Non-sensical relationship in " + getTokenName() + ": global .CLEAR and local .CLEAR. performed"); return null; } list.add(Constants.LST_DOT_CLEAR); } if (removedItems != null && !removedItems.isEmpty()) { for (String d : removedItems) { list.add(Constants.LST_DOT_CLEAR_DOT + d); } } Collection<Description> addedItems = changes.getAdded(); if (addedItems != null) { for (Description d : addedItems) { list.add(d.getPCCText()); } } if (list.isEmpty()) { return null; } return list.toArray(new String[list.size()]); } @Override public Class<CDOMObject> getTokenClass() { return CDOMObject.class; } }