/* ********** Juicer ********** ${A Fast template engine} Project Home: http://juicer.name Author: Guokai Gtalk: badkaikai@gmail.com Blog: http://benben.cc Licence: MIT License Version: 0.4.0-dev */ (function() { var juicer = function() { var args = [].slice.call(arguments); args.push(juicer.options); if(arguments.length == 1) { return juicer.compile.apply(juicer, args); } if(arguments.length >= 2) { return juicer.to_html.apply(juicer, args); } }; var __escapehtml = { escapehash: { '<': '<', '>': '>', '&': '&', '"': '"', "'": ''', '/': '/' }, escapereplace: function(k) { return __escapehtml.escapehash[k]; }, escaping: function(str) { return typeof(str) !== 'string' ? str : str.replace(/[&<>"]/igm, this.escapereplace); }, detection: function(data) { return typeof(data) === 'undefined' ? '' : data; } }; var __throw = function(error) { if(console) { if(console.warn) { console.warn(error); return; } if(console.log) { console.log(error); return; } } throw(error); }; var __creator = function(o, proto) { o = o !== Object(o) ? {} : o; if(o.__proto__) { o.__proto__ = proto; return o; } var _Empty = function() {}; var n = new((_Empty).prototype = proto, _Empty); for(var i in o) { if(o.hasOwnProperty(i)) { n[i] = o[i]; } } return n; }; juicer.__cache = {}; juicer.version = '0.4.0-dev'; juicer.settings = { forstart: /{@each\s*([\w\.]*?)\s*as\s*(\w*?)\s*(,\s*\w*?)?}/igm, forend: /{@\/each}/igm, ifstart: /{@if\s*([^}]*?)}/igm, ifend: /{@\/if}/igm, elsestart: /{@else}/igm, elseifstart: /{@else if\s*([^}]*?)}/igm, interpolate: /\${([\s\S]+?)}/igm, noneencode: /\$\${([\s\S]+?)}/igm, inlinecomment: /{#[^}]*?}/igm, rangestart: /{@each\s*(\w*?)\s*in\s*range\((\d+?),(\d+?)\)}/igm }; juicer.options = { cache: true, strip: true, errorhandling: true, detection: true, _method: __creator({ __escapehtml: __escapehtml, __throw: __throw }, this) }; juicer.set = function(conf, value) { if(arguments.length === 2) { this.options[conf] = value; return; } if(conf === Object(conf)) { for(var i in conf) { if(conf.hasOwnProperty(i)) { this.options[i] = conf[i]; } } } }; juicer.register = function(fname, fn) { var _method = this.options._method; if(_method.hasOwnProperty(fname)) { return false; } return _method[fname] = fn; }; juicer.unregister = function(fname) { var _method = this.options._method; if(_method.hasOwnProperty(fname)) { return delete _method[fname]; } }; juicer.template = function(options) { var that = this; this.options = options; this.__interpolate = function(_name, _escape, options) { var _define = _name.split('|'), _fn = ''; if(_define.length > 1) { _name = _define.shift(); _fn = '_method.' + _define.shift(); } return '<%= ' + (_escape ? '_method.__escapehtml.escaping' : '') + '(' + (!options || options.detection !== false ? '_method.__escapehtml.detection' : '') + '(' + _fn + '(' + _name + ')' + ')' + ')' + ' %>'; }; this.__removeShell = function(tpl, options) { var _counter = 0; tpl = tpl //for expression .replace(juicer.settings.forstart, function($, _name, alias, key) { var alias = alias || 'value', key = key && key.substr(1); var _iterate = 'i' + _counter++; return '<% for(var ' + _iterate + '=0, l' + _iterate + '=' + _name + '.length;' + _iterate + '<l' + _iterate + ';' + _iterate + '++) {' + 'var ' + alias + '=' + _name + '[' + _iterate + '];' + (key ? ('var ' + key + '=' + _iterate + ';') : '') + ' %>'; }) .replace(juicer.settings.forend, '<% } %>') //if expression .replace(juicer.settings.ifstart, function($, condition) { return '<% if(' + condition + ') { %>'; }) .replace(juicer.settings.ifend, '<% } %>') //else expression .replace(juicer.settings.elsestart, function($) { return '<% } else { %>'; }) //else if expression .replace(juicer.settings.elseifstart, function($, condition) { return '<% } else if(' + condition + ') { %>'; }) //interpolate without escape .replace(juicer.settings.noneencode, function($, _name) { return that.__interpolate(_name, false, options); }) //interpolate with escape .replace(juicer.settings.interpolate, function($, _name) { return that.__interpolate(_name, true, options); }) //clean up comments .replace(juicer.settings.inlinecomment, '') //range expression .replace(juicer.settings.rangestart, function($, _name, start, end) { var _iterate = 'j' + _counter++; return '<% for(var ' + _iterate + '=0;' + _iterate + '<' + (end - start) + ';' + _iterate + '++) {' + 'var ' + _name + '=' + _iterate + ';' + ' %>'; }); //exception handling if(!options || options.errorhandling !== false) { tpl = '<% try { %>' + tpl; tpl += '<% } catch(e) {_method.__throw("Juicer Render Exception: "+e.message);} %>'; } return tpl; }; this.__toNative = function(tpl, options) { return this.__convert(tpl, !options || options.strip); }; this.__lexicalAnalyze = function(tpl) { var buffer = []; var prefix = ''; var indexOf = function(array, item) { if (Array.prototype.indexOf && array.indexOf === Array.prototype.indexOf) { return array.indexOf(item); } for(var i=0; i < array.length; i++) { if(array[i] === item) return i; } return -1; }; var variableAnalyze = function($, statement) { statement = statement.match(/\w+/igm)[0]; if(indexOf(buffer, statement) === -1) { buffer.push(statement); //fuck ie } }; tpl.replace(juicer.settings.forstart, variableAnalyze). replace(juicer.settings.interpolate, variableAnalyze). replace(juicer.settings.ifstart, variableAnalyze); for(var i = 0;i < buffer.length; i++) { prefix += 'var ' + buffer[i] + '=_.' + buffer[i] + ';'; } return '<% ' + prefix + ' %>'; }; this.__convert=function(tpl, strip) { var buffer = [].join(''); buffer += "'use strict';"; //use strict mode buffer += "var _=_||{};"; buffer += "var _out='';_out+='"; if(strip !== false) { buffer += tpl .replace(/\\/g, "\\\\") .replace(/[\r\t\n]/g, " ") .replace(/'(?=[^%]*%>)/g, "\t") .split("'").join("\\'") .split("\t").join("'") .replace(/<%=(.+?)%>/g, "';_out+=$1;_out+='") .split("<%").join("';") .split("%>").join("_out+='")+ "';return _out;"; return buffer; } buffer += tpl .replace(/\\/g, "\\\\") .replace(/[\r]/g, "\\r") .replace(/[\t]/g, "\\t") .replace(/[\n]/g, "\\n") .replace(/'(?=[^%]*%>)/g, "\t") .split("'").join("\\'") .split("\t").join("'") .replace(/<%=(.+?)%>/g, "';_out+=$1;_out+='") .split("<%").join("';") .split("%>").join("_out+='")+ "';return _out.replace(/[\\r\\n]\\s+[\\r\\n]/g, '\\r\\n');"; return buffer; }; this.parse = function(tpl, options) { var _that = this; if(!options || options.loose !== false) { tpl = this.__lexicalAnalyze(tpl) + tpl; } tpl = this.__removeShell(tpl, options); tpl = this.__toNative(tpl, options); this._render = new Function('_, _method', tpl); this.render = function(_, _method) { if(!_method || _method !== that.options._method) { _method = __creator(_method, that.options._method); } return _that._render.call(this, _, _method); }; return this; }; }; juicer.compile = function(tpl, options) { if(!options || options !== this.options) { options = __creator(options, this.options); } try { var engine = this.__cache[tpl] ? this.__cache[tpl] : new this.template(this.options).parse(tpl, options); if(!options || options.cache !== false) { this.__cache[tpl] = engine; } return engine; } catch(e) { __throw('Juicer Compile Exception: ' + e.message); return { render: function() {} //noop }; } }; juicer.to_html = function(tpl, data, options) { if(!options || options !== this.options) { options = __creator(options, this.options); } return this.compile(tpl, options).render(data, options._method); }; typeof(module) !== 'undefined' && module.exports ? module.exports = juicer : this.juicer = juicer; })();