/*
* 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.Arrays;
import java.util.Collection;
import java.util.List;
import java.util.Set;
import java.util.StringTokenizer;
import java.util.TreeSet;
import pcgen.base.lang.StringUtil;
import pcgen.base.util.TreeMapToList;
import pcgen.cdom.base.CDOMObject;
import pcgen.cdom.base.CDOMReference;
import pcgen.cdom.base.Categorized;
import pcgen.cdom.base.Category;
import pcgen.cdom.base.Constants;
import pcgen.cdom.base.Loadable;
import pcgen.cdom.enumeration.ListKey;
import pcgen.cdom.reference.CDOMSingleRef;
import pcgen.cdom.reference.CategorizedCDOMReference;
import pcgen.cdom.reference.ReferenceManufacturer;
import pcgen.core.Ability;
import pcgen.core.PCClass;
import pcgen.core.PObject;
import pcgen.core.Race;
import pcgen.core.Skill;
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.ComplexParseResult;
import pcgen.rules.persistence.token.ParseResult;
import pcgen.util.StringPClassUtil;
/**
* Deals with the SERVESAS token for Abilities
*/
public class ServesAsToken extends AbstractTokenWithSeparator<CDOMObject>
implements CDOMPrimaryToken<CDOMObject>
{
@Override
public String getTokenName()
{
return "SERVESAS";
}
public List<Class<? extends PObject>> getLegalTypes()
{
return Arrays.asList(
PCClass.class, Ability.class, Skill.class, Race.class
// Ability.class, Deity.class, Domain.class,Equipment.class,
// Race.class, Skill.class,Spell.class, PCTemplate.class,
// WeaponProf.class
);
}
@Override
protected ParseResult parseNonEmptyToken(LoadContext context, CDOMObject obj, String value)
{
if (!getLegalTypes().contains(obj.getClass()))
{
ComplexParseResult cpr = new ComplexParseResult();
cpr.addErrorMessage("Cannot use SERVESAS on a " + obj.getClass());
cpr.addErrorMessage(" bad use found in "
+ obj.getClass().getSimpleName() + " " + obj.getKeyName());
return cpr;
}
return super.parseNonEmptyToken(context, obj, value);
}
@Override
protected char separator()
{
return '|';
}
@Override
protected ParseResult parseTokenWithSeparator(LoadContext context,
CDOMObject obj, String value)
{
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);
}
if (!st.hasMoreTokens())
{
return new ParseResult.Fail(getTokenName()
+ " must include at least one target object", context);
}
if (!rm.getReferenceClass().equals(obj.getClass()))
{
return new ParseResult.Fail(getTokenName()
+ " expecting a POBJECT Type valid for "
+ obj.getClass().getSimpleName() + ", found: "
+ firstToken, context);
}
String servekey = StringPClassUtil.getStringFor(obj.getClass());
ListKey<CDOMReference> listkey = ListKey.getKeyFor(CDOMReference.class,
"SERVES_AS_" + servekey);
while (st.hasMoreTokens())
{
CDOMSingleRef<?> ref = rm.getReference(st.nextToken());
context.getObjectContext().addToList(obj, listkey, ref);
}
return ParseResult.SUCCESS;
}
@Override
public String[] unparse(LoadContext context, CDOMObject obj)
{
String key = StringPClassUtil.getStringFor(obj.getClass());
ListKey<CDOMReference> listkey = ListKey.getKeyFor(CDOMReference.class,
"SERVES_AS_" + key);
Changes<CDOMReference> changes = context.getObjectContext().getListChanges(obj,
listkey);
Collection<CDOMReference> removedItems = changes.getRemoved();
if (removedItems != null && !removedItems.isEmpty()
|| changes.includesGlobalClear())
{
context.addWriteMessage(getTokenName()
+ " does not support .CLEAR");
return null;
}
if (!changes.hasAddedItems())
{
// Zero indicates no Token (and no global clear, so nothing to do)
return null;
}
TreeMapToList<String, String> map = new TreeMapToList<String, String>();
for (CDOMReference<?> ref : changes.getAdded())
{
String mapKey = key;
if (Categorized.class.isAssignableFrom(obj.getClass()))
{
Category<?> cat = ((CategorizedCDOMReference<?>) ref)
.getCDOMCategory();
mapKey += '=' + cat.toString();
}
map.addToListFor(mapKey, ref.getLSTformat(false));
}
List<String> returnList = new ArrayList<String>();
for (String mapKey : map.getKeySet())
{
Set<String> set = new TreeSet<String>(map.getListFor(mapKey));
returnList.add(mapKey + '|'
+ StringUtil.joinToStringBuilder(set, Constants.PIPE));
}
return returnList.toArray(new String[returnList.size()]);
}
@Override
public Class<CDOMObject> getTokenClass()
{
return CDOMObject.class;
}
}