// // This software is now distributed according to // the Lesser Gnu Public License. Please see // http://www.gnu.org/copyleft/lesser.txt for // the details. // -- Happy Computing! // package com.stevesoft.pat; import java.util.Hashtable; /** * ReplaceRule is a singly linked list of Objects which describe how to replace * the matched portion of a String. The only member method that you absolutely * need to define to use this class is apply(StringBuffer,RegRes) -- although * you may want define toString1() and clone1() (if you are unhappy with the * default methods) that are needed by the clone() or toString() methods on this * class. During the replacement process, each ReplaceRule tells the replacer * what to add to javajs.util.SB and uses the contents of the Regular expression * result to get the information it needs to do this. Here is an example * * @see com.stevesoft.pat.NullRule * @see com.stevesoft.pat.AmpersandRule * @see com.stevesoft.pat.BackRefRule * @see com.stevesoft.pat.LeftRule * @see com.stevesoft.pat.RightRule * @see com.stevesoft.pat.StringRule */ public abstract class ReplaceRule { /** points to the next ReplaceRule in the linked list. */ protected ReplaceRule next = null; /** * This function appends to the StringBufferLike the text you want to replaced * the portion of the String last matched. */ public abstract void apply(StringBufferLike sb, RegRes r); /** * A rule describing how to clone only the current ReplaceRule, and none of * the others in this linked list. It is called by clone() for each item in * the list. */ public Object clone1() { return new RuleHolder(this); } public final Object clone() { ReplaceRule x = (ReplaceRule) clone1(); ReplaceRule xsav = x; ReplaceRule y = this; while (y.next != null) { x.next = (ReplaceRule) y.next.clone1(); x.name = y.name; x = x.next; y = y.next; } return xsav; } static ReplaceRule add(ReplaceRule head, ReplaceRule adding) { if (head == null) { return head = adding; } head.addRule(adding); return head; } public ReplaceRule add(ReplaceRule adding) { return add(this, adding); } /** Add another ReplaceRule to the linked list. */ public void addRule(ReplaceRule r) { if (next == null) { next = r; } else { next.addRule(r); } } static Regex getvar = null; final static Regex getv() { // Thanks to Michael Jimenez for pointing out the need // to clone getvar rather than simply returning it. // Previously this was not thread safe. // if(getvar != null) return getvar; if (getvar != null) { return (Regex) getvar.clone(); } getvar = new Regex("(?:\\\\(\\d+)|" + // ref 1 "\\$(?:" + "(\\d+)|" + // ref 2 "(\\w+)|" + // ref 3 "([&'`])|" + // ref 4 "\\{(?:(\\d+)|" + // ref 5 "([^\n}\\\\]+))}" + // ref 6 ")|" + "\\\\([nrbtaef])|" + // ref 7 "\\\\c([\u0000-\uFFFF])|" + // ref 8 "\\\\x([A-Fa-f0-9]{2})|" + // ref 9 "\\\\([\u0000-\uFFFF])" + // ref 10 ")"); getvar.optimize(); return getvar; } static Hashtable defs = new Hashtable(); public static boolean isDefined(String s) { return defs.get(s) != null; } public static void define(String s, Regex r) { defs.put(s, r); } public static void define(String s, ReplaceRule r) { defs.put(s, r); r.name = s; } String name = getClass().getName(); public static void define(String s, Transformer t) { defs.put(s, t); } public static void undefine(String s) { defs.remove(s); } /** * This tells how to convert just the current element (and none of the other * items in the linked list) to a String. This method is called by toString() * for each item in the linked list. */ public String toString1() { return "${" + name + "}"; } /** Convert to a String. */ public final String toString() { javajs.util.SB sb = new javajs.util.SB(); sb.append(toString1()); ReplaceRule rr = this.next; while (rr != null) { sb.append(rr.toString1()); rr = rr.next; } return sb.toString(); } /** * Modified the behavior of a ReplaceRule by supplying an argument. If a * ReplaceRule named "foo" is defined and the pattern "s/x/${foo:5}/" is given * to Regex.perlCode, then the "foo" the definition of "foo" will be retrieved * and arg("5") will be called. If the result is non-null, that is the * ReplaceRule that will be used. If the result is null, then the pattern * works just as if it were "s/x/${foo}/". * * @see com.stevesoft.pat.Validator#arg(java.lang.String) */ public ReplaceRule arg(String s) { return null; } static int getHexDigit(char c) { if (c >= '0' && c <= '9') { return c - '0'; } if (c >= 'a' && c <= 'f') { return c - 'a' + 10; } return c - 'A' + 10; } }