|  | (function(window) { | 
|  | var re = { | 
|  | not_string: /[^s]/, | 
|  | number: /[diefg]/, | 
|  | json: /[j]/, | 
|  | not_json: /[^j]/, | 
|  | text: /^[^\x25]+/, | 
|  | modulo: /^\x25{2}/, | 
|  | placeholder: /^\x25(?:([1-9]\d*)\$|\(([^\)]+)\))?(\+)?(0|'[^$])?(-)?(\d+)?(?:\.(\d+))?([b-gijosuxX])/, | 
|  | key: /^([a-z_][a-z_\d]*)/i, | 
|  | key_access: /^\.([a-z_][a-z_\d]*)/i, | 
|  | index_access: /^\[(\d+)\]/, | 
|  | sign: /^[\+\-]/ | 
|  | } | 
|  |  | 
|  | function sprintf() { | 
|  | var key = arguments[0], cache = sprintf.cache | 
|  | if (!(cache[key] && cache.hasOwnProperty(key))) { | 
|  | cache[key] = sprintf.parse(key) | 
|  | } | 
|  | return sprintf.format.call(null, cache[key], arguments) | 
|  | } | 
|  |  | 
|  | sprintf.format = function(parse_tree, argv) { | 
|  | var cursor = 1, tree_length = parse_tree.length, node_type = "", arg, output = [], i, k, match, pad, pad_character, pad_length, is_positive = true, sign = "" | 
|  | for (i = 0; i < tree_length; i++) { | 
|  | node_type = get_type(parse_tree[i]) | 
|  | if (node_type === "string") { | 
|  | output[output.length] = parse_tree[i] | 
|  | } | 
|  | else if (node_type === "array") { | 
|  | match = parse_tree[i] // convenience purposes only | 
|  | if (match[2]) { // keyword argument | 
|  | arg = argv[cursor] | 
|  | for (k = 0; k < match[2].length; k++) { | 
|  | if (!arg.hasOwnProperty(match[2][k])) { | 
|  | throw new Error(sprintf("[sprintf] property '%s' does not exist", match[2][k])) | 
|  | } | 
|  | arg = arg[match[2][k]] | 
|  | } | 
|  | } | 
|  | else if (match[1]) { // positional argument (explicit) | 
|  | arg = argv[match[1]] | 
|  | } | 
|  | else { // positional argument (implicit) | 
|  | arg = argv[cursor++] | 
|  | } | 
|  |  | 
|  | if (get_type(arg) == "function") { | 
|  | arg = arg() | 
|  | } | 
|  |  | 
|  | if (re.not_string.test(match[8]) && re.not_json.test(match[8]) && (get_type(arg) != "number" && isNaN(arg))) { | 
|  | throw new TypeError(sprintf("[sprintf] expecting number but found %s", get_type(arg))) | 
|  | } | 
|  |  | 
|  | if (re.number.test(match[8])) { | 
|  | is_positive = arg >= 0 | 
|  | } | 
|  |  | 
|  | switch (match[8]) { | 
|  | case "b": | 
|  | arg = arg.toString(2) | 
|  | break | 
|  | case "c": | 
|  | arg = String.fromCharCode(arg) | 
|  | break | 
|  | case "d": | 
|  | case "i": | 
|  | arg = parseInt(arg, 10) | 
|  | break | 
|  | case "j": | 
|  | arg = JSON.stringify(arg, null, match[6] ? parseInt(match[6]) : 0) | 
|  | break | 
|  | case "e": | 
|  | arg = match[7] ? arg.toExponential(match[7]) : arg.toExponential() | 
|  | break | 
|  | case "f": | 
|  | arg = match[7] ? parseFloat(arg).toFixed(match[7]) : parseFloat(arg) | 
|  | break | 
|  | case "g": | 
|  | arg = match[7] ? parseFloat(arg).toPrecision(match[7]) : parseFloat(arg) | 
|  | break | 
|  | case "o": | 
|  | arg = arg.toString(8) | 
|  | break | 
|  | case "s": | 
|  | arg = ((arg = String(arg)) && match[7] ? arg.substring(0, match[7]) : arg) | 
|  | break | 
|  | case "u": | 
|  | arg = arg >>> 0 | 
|  | break | 
|  | case "x": | 
|  | arg = arg.toString(16) | 
|  | break | 
|  | case "X": | 
|  | arg = arg.toString(16).toUpperCase() | 
|  | break | 
|  | } | 
|  | if (re.json.test(match[8])) { | 
|  | output[output.length] = arg | 
|  | } | 
|  | else { | 
|  | if (re.number.test(match[8]) && (!is_positive || match[3])) { | 
|  | sign = is_positive ? "+" : "-" | 
|  | arg = arg.toString().replace(re.sign, "") | 
|  | } | 
|  | else { | 
|  | sign = "" | 
|  | } | 
|  | pad_character = match[4] ? match[4] === "0" ? "0" : match[4].charAt(1) : " " | 
|  | pad_length = match[6] - (sign + arg).length | 
|  | pad = match[6] ? (pad_length > 0 ? str_repeat(pad_character, pad_length) : "") : "" | 
|  | output[output.length] = match[5] ? sign + arg + pad : (pad_character === "0" ? sign + pad + arg : pad + sign + arg) | 
|  | } | 
|  | } | 
|  | } | 
|  | return output.join("") | 
|  | } | 
|  |  | 
|  | sprintf.cache = {} | 
|  |  | 
|  | sprintf.parse = function(fmt) { | 
|  | var _fmt = fmt, match = [], parse_tree = [], arg_names = 0 | 
|  | while (_fmt) { | 
|  | if ((match = re.text.exec(_fmt)) !== null) { | 
|  | parse_tree[parse_tree.length] = match[0] | 
|  | } | 
|  | else if ((match = re.modulo.exec(_fmt)) !== null) { | 
|  | parse_tree[parse_tree.length] = "%" | 
|  | } | 
|  | else if ((match = re.placeholder.exec(_fmt)) !== null) { | 
|  | if (match[2]) { | 
|  | arg_names |= 1 | 
|  | var field_list = [], replacement_field = match[2], field_match = [] | 
|  | if ((field_match = re.key.exec(replacement_field)) !== null) { | 
|  | field_list[field_list.length] = field_match[1] | 
|  | while ((replacement_field = replacement_field.substring(field_match[0].length)) !== "") { | 
|  | if ((field_match = re.key_access.exec(replacement_field)) !== null) { | 
|  | field_list[field_list.length] = field_match[1] | 
|  | } | 
|  | else if ((field_match = re.index_access.exec(replacement_field)) !== null) { | 
|  | field_list[field_list.length] = field_match[1] | 
|  | } | 
|  | else { | 
|  | throw new SyntaxError("[sprintf] failed to parse named argument key") | 
|  | } | 
|  | } | 
|  | } | 
|  | else { | 
|  | throw new SyntaxError("[sprintf] failed to parse named argument key") | 
|  | } | 
|  | match[2] = field_list | 
|  | } | 
|  | else { | 
|  | arg_names |= 2 | 
|  | } | 
|  | if (arg_names === 3) { | 
|  | throw new Error("[sprintf] mixing positional and named placeholders is not (yet) supported") | 
|  | } | 
|  | parse_tree[parse_tree.length] = match | 
|  | } | 
|  | else { | 
|  | throw new SyntaxError("[sprintf] unexpected placeholder") | 
|  | } | 
|  | _fmt = _fmt.substring(match[0].length) | 
|  | } | 
|  | return parse_tree | 
|  | } | 
|  |  | 
|  | var vsprintf = function(fmt, argv, _argv) { | 
|  | _argv = (argv || []).slice(0) | 
|  | _argv.splice(0, 0, fmt) | 
|  | return sprintf.apply(null, _argv) | 
|  | } | 
|  |  | 
|  | /** | 
|  | * helpers | 
|  | */ | 
|  | function get_type(variable) { | 
|  | return Object.prototype.toString.call(variable).slice(8, -1).toLowerCase() | 
|  | } | 
|  |  | 
|  | function str_repeat(input, multiplier) { | 
|  | return Array(multiplier + 1).join(input) | 
|  | } | 
|  |  | 
|  | /** | 
|  | * export to either browser or node.js | 
|  | */ | 
|  | if (typeof exports !== "undefined") { | 
|  | exports.sprintf = sprintf | 
|  | exports.vsprintf = vsprintf | 
|  | } | 
|  | else { | 
|  | window.sprintf = sprintf | 
|  | window.vsprintf = vsprintf | 
|  |  | 
|  | if (typeof define === "function" && define.amd) { | 
|  | define(function() { | 
|  | return { | 
|  | sprintf: sprintf, | 
|  | vsprintf: vsprintf | 
|  | } | 
|  | }) | 
|  | } | 
|  | } | 
|  | })(typeof window === "undefined" ? this : window); |