/
com/planet_ink/coffee_mud/Abilities/Common/
com/planet_ink/coffee_mud/Abilities/Diseases/
com/planet_ink/coffee_mud/Abilities/Druid/
com/planet_ink/coffee_mud/Abilities/Fighter/
com/planet_ink/coffee_mud/Abilities/Languages/
com/planet_ink/coffee_mud/Abilities/Misc/
com/planet_ink/coffee_mud/Abilities/Prayers/
com/planet_ink/coffee_mud/Abilities/Properties/
com/planet_ink/coffee_mud/Abilities/Skills/
com/planet_ink/coffee_mud/Abilities/Songs/
com/planet_ink/coffee_mud/Abilities/Specializations/
com/planet_ink/coffee_mud/Abilities/Spells/
com/planet_ink/coffee_mud/Abilities/Thief/
com/planet_ink/coffee_mud/Abilities/Traps/
com/planet_ink/coffee_mud/Behaviors/
com/planet_ink/coffee_mud/CharClasses/
com/planet_ink/coffee_mud/CharClasses/interfaces/
com/planet_ink/coffee_mud/Commands/
com/planet_ink/coffee_mud/Commands/interfaces/
com/planet_ink/coffee_mud/Common/
com/planet_ink/coffee_mud/Common/interfaces/
com/planet_ink/coffee_mud/Exits/interfaces/
com/planet_ink/coffee_mud/Items/Armor/
com/planet_ink/coffee_mud/Items/Basic/
com/planet_ink/coffee_mud/Items/BasicTech/
com/planet_ink/coffee_mud/Items/CompTech/
com/planet_ink/coffee_mud/Items/MiscMagic/
com/planet_ink/coffee_mud/Items/Weapons/
com/planet_ink/coffee_mud/Items/interfaces/
com/planet_ink/coffee_mud/Libraries/
com/planet_ink/coffee_mud/Libraries/interfaces/
com/planet_ink/coffee_mud/Locales/
com/planet_ink/coffee_mud/MOBS/
com/planet_ink/coffee_mud/Races/
com/planet_ink/coffee_mud/Races/interfaces/
com/planet_ink/coffee_mud/WebMacros/
com/planet_ink/coffee_mud/WebMacros/interfaces/
com/planet_ink/coffee_mud/core/
com/planet_ink/coffee_mud/core/collections/
com/planet_ink/coffee_mud/core/interfaces/
com/planet_ink/coffee_mud/core/intermud/
com/planet_ink/coffee_mud/core/intermud/i3/
com/planet_ink/coffee_web/server/
com/planet_ink/siplet/applet/
lib/
resources/factions/
resources/fakedb/
resources/progs/autoplayer/
resources/quests/holidays/
web/
web/admin.templates/
web/admin/grinder/
web/admin/images/
web/clan.templates/
web/pub.templates/
web/pub/images/mxp/
web/pub/sounds/
web/pub/textedit/
package com.planet_ink.siplet.support;

import java.util.*;

/*
   Copyright 2008-2019 Bo Zimmerman

   Licensed under the Apache License, Version 2.0 (the "License");
   you may not use this file except in compliance with the License.
   You may obtain a copy of the License at

	   http://www.apache.org/licenses/LICENSE-2.0

   Unless required by applicable law or agreed to in writing, software
   distributed under the License is distributed on an "AS IS" BASIS,
   WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
   See the License for the specific language governing permissions and
   limitations under the License.
 */
public class MXPElement implements Cloneable
{
	public static final int	BIT_OPEN				= 1;
	public static final int	BIT_COMMAND				= 2;
	public static final int	BIT_NEEDTEXT			= 4;
	public static final int	BIT_SPECIAL				= 8;
	public static final int	BIT_HTML				= 16;
	public static final int	BIT_NOTSUPPORTED		= 32;
	public static final int	BIT_EATTEXT				= 64;
	public static final int	BIT_DISABLED			= 128;

	private String			name					= "";
	private String			definition				= "";
	private String			attributes				= "";
	private String			flag					= "";
	private String			unsupportedParms		= "";
	private int				bitmap					= 0;
	private Vector<String>	parsedAttributes		= null;
	private List<String>	userParms				= new Vector<String>();
	private boolean			basicElement			= true;

	private Hashtable<String,String>		attributeValues			= null;
	private Hashtable<String,String>		alternativeAttributes	= null;

	private int				bufInsert				= -1;

	public MXPElement(final String newName, final String theDefinition, final String theAttributes, final String theFlag, final int theBitmap)
	{
		super();
		name = newName;
		definition = theDefinition;
		attributes = theAttributes;
		flag = theFlag;
		bitmap = theBitmap;
		if ((!isCommand()) && (theDefinition.toUpperCase().indexOf("&TEXT;") >= 0))
			bitmap = bitmap | BIT_NEEDTEXT;
	}

	public MXPElement(final String newName, final String theDefinition, final String theAttributes, final String theFlag, final int theBitmap, final String unsupported)
	{
		super();
		name = newName;
		definition = theDefinition;
		attributes = theAttributes;
		flag = theFlag;
		bitmap = theBitmap;
		if ((!isCommand()) && (theDefinition.toUpperCase().indexOf("&TEXT;") >= 0))
			bitmap = bitmap | BIT_NEEDTEXT;
		unsupportedParms = unsupported;
	}

	@SuppressWarnings({ "unchecked", "rawtypes" })
	public MXPElement copyOf()
	{
		try
		{
			final MXPElement E = (MXPElement) this.clone();
			if (E.parsedAttributes != null)
				E.parsedAttributes = (Vector) E.parsedAttributes.clone();
			if (E.attributeValues != null)
				E.attributeValues = (Hashtable) E.attributeValues.clone();
			if (E.alternativeAttributes != null)
				E.alternativeAttributes = (Hashtable) E.alternativeAttributes.clone();
			if (E.userParms != null)
				E.userParms = new Vector<String>(E.userParms);
			return E;
		}
		catch (final Exception e)
		{
		}
		return this;
	}

	public String name()
	{
		return name;
	}

	public void setName(final String newName)
	{
		name = newName;
	}

	public boolean isCommand()
	{
		return Util.bset(bitmap, BIT_COMMAND);
	}

	public boolean isOpen()
	{
		return Util.bset(bitmap, BIT_OPEN);
	}

	public boolean isHTML()
	{
		return Util.bset(bitmap, BIT_HTML);
	}

	public boolean isSpecialProcessor()
	{
		return Util.bset(bitmap, BIT_SPECIAL);
	}

	public boolean isDisabled()
	{
		return Util.bset(bitmap, BIT_DISABLED);
	}

	public boolean isTextEater()
	{
		return Util.bset(bitmap, BIT_EATTEXT);
	}

	public String getDefinition()
	{
		return definition;
	}

	public void setDefinition(final String defi)
	{
		definition = defi;
	}

	public String getAttributes()
	{
		return attributes;
	}

	public boolean needsText()
	{
		return Util.bset(bitmap, BIT_NEEDTEXT);
	}

	public void setNotBasicElement()
	{
		basicElement = false;
	}

	public boolean isBasicElement()
	{
		return basicElement;
	}

	public boolean isGenerallySupported()
	{
		return !Util.bset(bitmap, BIT_NOTSUPPORTED);
	}

	public void setBitmap(final int newBitmap)
	{
		bitmap = newBitmap;
	}

	public int getBitmap()
	{
		return bitmap;
	}

	public Vector<String> getUnsupportedParms()
	{
		if ((unsupportedParms == null) || (unsupportedParms.trim().length() == 0))
			return new Vector<String>();
		return Util.parseSpaces(unsupportedParms, true);
	}

	public void setAttributes(final String newAttributes)
	{
		attributes = newAttributes;
		parsedAttributes = null;
		attributeValues = null;
		alternativeAttributes = null;
	}

	public String getAttributeValue(String tag)
	{
		getParsedAttributes();
		tag = tag.toUpperCase().trim();
		if (attributeValues.containsKey(tag))
			return attributeValues.get(tag).toString();
		return null;
	}

	public void setAttributeValue(final String tag, final String value)
	{
		getParsedAttributes();
		attributeValues.remove(tag);
		if (value != null)
			attributeValues.put(tag, value);
	}

	public synchronized Vector<String> getParsedAttributes()
	{
		if (parsedAttributes != null)
			return parsedAttributes;
		parsedAttributes = new Vector<String>();
		attributeValues = new Hashtable<String,String>();
		alternativeAttributes = new Hashtable<String,String>();
		final StringBuffer buf = new StringBuffer(attributes.trim());
		StringBuffer bit = new StringBuffer("");
		char quotes = '\0';
		int i = -1;
		char lastC = ' ';
		boolean firstEqual = false;
		while ((++i) < buf.length())
		{
			switch (buf.charAt(i))
			{
			case '=':
				if ((!firstEqual) && (bit.length() > 0))
				{
					final String tag = bit.toString().toUpperCase().trim();
					bit = new StringBuffer("");
					parsedAttributes.addElement(tag);
					attributeValues.put(tag, bit.toString());
				}
				else
					bit.append(buf.charAt(i));
				firstEqual = true;
				break;
			case '\n':
			case '\r':
			case ' ':
			case '\t':
				if (quotes == '\0')
				{
					if ((!firstEqual) && (bit.length() > 0))
						parsedAttributes.addElement(bit.toString().toUpperCase().trim());
					bit = new StringBuffer("");
					firstEqual = false;
				}
				else
					bit.append(buf.charAt(i));
				break;
			case '"':
			case '\'':
				if (lastC == '\\')
					bit.append(buf.charAt(i));
				else
				if ((lastC == '=') || (quotes != '\0') || ((quotes == '\0') && ((lastC == ' ') || (lastC == '\t'))))
				{
					if ((quotes != '\0') && (quotes == buf.charAt(i)))
					{
						quotes = '\0';
						if ((!firstEqual) && (bit.length() > 0))
							parsedAttributes.addElement(bit.toString().toUpperCase().trim());
						bit = new StringBuffer("");
						firstEqual = false;
					}
					else
					{
						if (quotes != '\0')
							bit.append(buf.charAt(i));
						else
							quotes = buf.charAt(i);
					}
				}
				else
					bit.append(buf.charAt(i));
				break;
			default:
				bit.append(buf.charAt(i));
				break;
			}
			lastC = buf.charAt(i);
		}
		if ((!firstEqual) && (bit.length() > 0))
			parsedAttributes.addElement(bit.toString().toUpperCase().trim());
		for (int p = parsedAttributes.size() - 1; p >= 0; p--)
		{
			final String PA = parsedAttributes.elementAt(p);
			final String VAL = attributeValues.get(PA);
			if ((VAL != null) && (parsedAttributes.contains(VAL.toString())))
			{
				parsedAttributes.removeElementAt(p);
				attributeValues.remove(PA);
				alternativeAttributes.put(PA, VAL.toString());
			}
		}
		return parsedAttributes;
	}

	public String getFlag()
	{
		return flag;
	}

	public List<String> getUserParms()
	{
		return userParms;
	}

	public void saveSettings(final int insertPoint, final Vector<String> theUserParms)
	{
		bufInsert = insertPoint;
		userParms = theUserParms;
	}

	public int getBufInsert()
	{
		return bufInsert;
	}

	public void deleteAttribute(final String name)
	{
		getParsedAttributes();
		attributeValues.remove(name.toUpperCase().trim());
	}

	public Vector<String> getCloseTags(final String desc)
	{
		final StringBuffer buf = new StringBuffer(desc);
		final Vector<String> tags = new Vector<String>();
		StringBuffer bit = null;
		char quotes = '\0';
		int i = -1;
		char lastC = ' ';
		while ((++i) < buf.length())
		{
			switch (buf.charAt(i))
			{
			case '<':
				if (quotes != '\0')
					bit = null;
				else
				if (bit != null)
				{
					if (MXP.tagDebug)
					{
						System.out.println("/TAG/CLOSER2S=" + Util.toStringList(tags));
						System.out.flush();
					}
					return tags;
				}
				else
					bit = new StringBuffer("");
				break;
			case '>':
				if ((quotes == '\0') && (bit != null) && (bit.toString().trim().length() > 0))
					tags.add(bit.toString().toUpperCase().trim());
				bit = null;
				break;
			case ' ':
			case '\t':
				if ((quotes == '\0') && (bit != null) && (bit.toString().trim().length() > 0))
					tags.add(bit.toString().toUpperCase().trim());
				bit = null;
				break;
			case '"':
			case '\'':
				if (lastC == '\\')
					bit = null;
				else
				if ((quotes != '\0') && (quotes == buf.charAt(i)))
					quotes = '\0';
				else
				if (quotes == '\0')
					quotes = buf.charAt(i);
				bit = null;
				break;
			default:
				if ((bit != null) && (Character.isLetterOrDigit(buf.charAt(i))))
					bit.append(buf.charAt(i));
				else
					bit = null;
				break;
			}
			lastC = buf.charAt(i);
		}
		if (MXP.tagDebug)
		{
			System.out.println("/TAG/CLOSERS=" + Util.toStringList(tags));
			System.out.flush();
		}
		return tags;
	}

	public String getFoldedDefinition(final String text)
	{
		final Vector<String> aV = getParsedAttributes();
		attributeValues.remove("TEXT");
		attributeValues.put("TEXT", text);
		if ((userParms != null) && (userParms.size() > 0))
		{
			int position = -1;
			String avParm = null;
			String userParm = null;
			for (int u = 0; u < userParms.size(); u++)
			{
				userParm = userParms.get(u).toUpperCase().trim();
				int xx = userParm.indexOf('=');
				if ((xx > 0) && (alternativeAttributes.containsKey(userParm.substring(0, xx).trim())))
				{
					final String newKey = alternativeAttributes.get(userParm.substring(0, xx).trim());
					final String uu = userParms.get(u);
					xx = uu.indexOf('=');
					userParms.set(u, newKey + uu.substring(xx));
					userParm = userParms.get(u).toUpperCase().trim();
				}
				boolean found = false;
				if (userParm != null)
				{
					for (int a = 0; a < aV.size(); a++)
					{
						avParm = aV.elementAt(a);
						if ((userParm.startsWith(avParm + "=")) || (avParm.equals(userParm)))
						{
							found = true;
							if (a > position)
								position = a;
							attributeValues.remove(avParm);
							if (avParm != null)
								attributeValues.put(avParm, (userParm.equals(avParm)) ? "" : userParms.get(u).trim().substring(avParm.length() + 1));
							break;
						}
					}
				}
				if ((!found) && (position < (aV.size() - 1)))
				{
					position++;
					avParm = aV.elementAt(position);
					attributeValues.remove(avParm);
					attributeValues.put(avParm, userParms.get(u).trim());
				}
			}
		}
		return definition;
	}
}