substitute-debug.js revision 637be785ab98741335d5cd6634fddc5f6ecdb41b
YUI.add('substitute', function(Y) {
/**
* String variable substitution and string formatting.
* If included, the substitute method is added to the YUI instance.
*
* @module substitute
*/
var L = Y.Lang, DUMP='dump', SPACE=' ', LBRACE='{', RBRACE='}',
/**
* The following methods are added to the YUI instance
* @class YUI~substitute
*/
/**
* Does variable substitution on a string. It scans through the string
* looking for expressions enclosed in { } braces. If an expression
* is found, it is used a key on the object. If there is a space in
* the key, the first word is used for the key and the rest is provided
* to an optional function to be used to programatically determine the
* value (the extra information might be used for this decision). If
* the value for the key in the object, or what is returned from the
* function has a string value, number value, or object value, it is
* substituted for the bracket expression and it repeats. If this
* value is an object, it uses the Object's toString() if this has
* been overridden, otherwise it does a shallow dump of the key/value
* pairs if Y.dump is available (if dump isn't available, toString()
* is used).
*
* This method is included in the 'substitute' module. It is not included
* in the YUI module.
*
* @method substitute
* @for YUI
* @param s {string} The string that will be modified.
* @param o An object containing the replacement values
* @param f {function} An optional function that can be used to
* process each match. It receives the key,
* value, and any extra metadata included with
* the key inside of the braces.
* @param ldelim {string} optional left delimiter for the replacement token (default: {)
* @param rdelim {string} optional right delimiter for the replacement token (default: })
* @return {string} the substituted string
*/
substitute = function (s, o, f, ldelim, rdelim) {
var i, j, k, key, v, meta, saved=[], token, dump;
ldelim = ldelim || LBRACE;
rdelim = rdelim || RBRACE;
for (;;) {
i = s.lastIndexOf(ldelim);
if (i < 0) {
break;
}
j = s.indexOf(rdelim, i);
if (i + 1 >= j) {
break;
}
//Extract key and meta info
token = s.substring(i + 1, j);
key = token;
meta = null;
k = key.indexOf(SPACE);
if (k > -1) {
meta = key.substring(k + 1);
key = key.substring(0, k);
}
// lookup the value
v = o[key];
// if a substitution function was provided, execute it
if (f) {
v = f(key, v, meta);
}
if (L.isObject(v)) {
if (!Y.dump) {
v = v.toString();
} else {
if (L.isArray(v)) {
v = Y.dump(v, parseInt(meta, 10));
} else {
meta = meta || "";
// look for the keyword 'dump', if found force obj dump
dump = meta.indexOf(DUMP);
if (dump > -1) {
meta = meta.substring(4);
}
// use the toString if it is not the Object toString
// and the 'dump' meta info was not found
if (v.toString===Object.prototype.toString||dump>-1) {
v = Y.dump(v, parseInt(meta, 10));
} else {
v = v.toString();
}
}
}
} else if (!L.isString(v) && !L.isNumber(v)) {
// This {block} has no replace string. Save it for later.
v = "~-" + saved.length + "-~";
saved[saved.length] = token;
// break;
}
s = s.substring(0, i) + v + s.substring(j + 1);
}
// restore saved {block}s
for (i=saved.length-1; i>=0; i=i-1) {
s = s.replace(new RegExp("~-" + i + "-~"), ldelim + saved[i] + rdelim, "g");
}
return s;
};
(function() {
// Static regex for matching {key} {key|input} patterns
var pattern = /\{\s*([^|{}]+?)\s*(?:\|([^\{}]*))?\s*\}/g,
// Recursion limit
MAX_RECURSION = 100,
// Replacer function factory
replacer = function (o, opts) {
return function (m, k, x) {
var v = o[k];
return v !== undefined ? v : (opts.preserve ? m : '');
};
},
// Wrapper to execute substitute until no more substitutions can be done
recurse = function (s, o, opts) {
var r = s, i = 0;
do {
s = r;
console.log('pass: ' + i);
console.log('s: ' + Y.dump(s));
console.log('o: ' + Y.dump(o));
r = substitute(s, o, opts);
} while (r !== s && ++i < MAX_RECURSION);
return r;
},
LDELIM = /\{/g,
RDELIM = /\}/g,
UNSPECIFIED = {},
substitute = function (s, o, opts) {
opts = opts || UNSPECIFIED;
if (false !== opts.recurse) {
opts.recurse = false;
return recurse.apply(this, arguments);
}
var ldelim = opts.ldelim, rdelim = opts.rdelim,
p = opts.pattern || pattern, r = opts.replacer || replacer;
if (ldelim || rdelim && !opts.pattern) {
pattern = pattern.replace(LDELIM, ldelim).replace(RDELIM, rdelim);
}
return s && s.replace ? s.replace(p, r(o, opts)) : s;
};
Y.substitute = substitute;
L.substitute = substitute;
})();
}, '@VERSION@' ,{optional:['dump']});