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; } }