Created
February 8, 2012 00:43
-
-
Save jiggliemon/1763600 to your computer and use it in GitHub Desktop.
A Simple Template constructor
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
define(['template/mixin'], function( TemplateMixin ) { | |
function Template (config) { | |
config = config || {}; | |
this._template = null; | |
this._context = {} | |
if( (typeof config.template === 'string') || (typeof config == 'string') ) { | |
this._template = config.template || config; | |
} | |
} | |
Template.prototype = TemplateMixin; | |
return Template; | |
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
define( function () { | |
function escape (string) { | |
return ('' + string) | |
.replace(/&/g, '&') | |
.replace(/</g, '<') | |
.replace(/>/g, '>') | |
.replace(/"/g, '"') | |
.replace(/'/g, ''') | |
.replace(/\//g, '/'); | |
}; | |
var TemplateMethods = { | |
_context: {} | |
,_template: null | |
,_templateTags: { | |
open: '<%' | |
,close: '%>' | |
} | |
,_templateOperators: { | |
interpolate: ['=([\\s\\S]+?)', function (match, code) { | |
return "'," + code.replace(/\\'/g, "'") + ",'"; | |
}] | |
,escape: ['-([\\s\\S]+?)', function (match, code) { | |
return "',escape(" + code.replace(/\\'/g, "'") + "),'"; | |
}] | |
} | |
,setContext: function (key, value) { | |
var context = this._context = this._context || {} | |
if(typeof key === 'object') { | |
for(var k in key) { | |
if(key.hasOwnProperty(k)) { | |
this.setContext(k, key[k]); | |
} | |
} | |
return; | |
} | |
context[key] = value; | |
return this; | |
} | |
,getContext: function (args) { | |
var args = (Array.isArray(args))? args : slice.call(arguments,0) | |
,context = {};; | |
if(arguments.length > 0) { | |
args.forEach(function (arg) { | |
context[arg] = this._context[arg]; | |
}); | |
} | |
return context; | |
} | |
,setTags: function ( tags) { | |
var key; | |
for (key in tags) { | |
if (tags.hasOwnProperty(key)) { | |
this._templateTags[key] = tags[key]; | |
} | |
} | |
return this; | |
} | |
,setTag: function( tag, str) { | |
this._templateTags[tag] = str; | |
} | |
,getTags: function () { | |
return this._templateTags; | |
} | |
,getTag: function (tag) { | |
return this._templateTags[tag]; | |
} | |
,setTemplate: function ( /* String */ str) { | |
if(typeof str === 'string'){ | |
this._template = str; | |
this.fireEvent && this.fireEvent('template:ready:latched',str); | |
} | |
} | |
,getTemplate: function () { | |
return this._template || '<b>No template loaded</b>'; | |
} | |
,parseOperators: function () { | |
var key, operator; | |
for (key in this._templateOperators) { | |
if (this._templateOperators.hasOwnProperty(key)) { | |
operator = this._templateOperators[key]; | |
if (typeof operator[0] === 'string') { | |
this.addOperator(key, operator[0], operator[1]); | |
} | |
} | |
} | |
} | |
,getOperators: function () { | |
if (!this._operatorsParsed) { | |
this.parseOperators(); | |
} | |
return this._templateOperators; | |
} | |
,addOperator: function ( /* String */ name, /* || String */ regexp, /* Function || String */ fn) { | |
// This will be part of a str.replace method | |
// So the arguments should match those that you would use | |
// for the .replace method on strings. | |
if (typeof regexp.exec !== 'function') { | |
regexp = new RegExp(this.getTag('open') + regexp + this.getTag('close'), 'g'); | |
} | |
this._templateOperators[name] = [regexp, fn]; | |
} | |
,compile: function ( /* Object */ data) { | |
data = data || this._context; | |
var open = this.getTag('open') | |
,close = this.getTag('close') | |
,operators = this.getOperators() | |
,key, body, head = 'var p=[],print=function(){p.push.apply(p,arguments);};' | |
,wrapper = ["with(__o){p.push('", "');}return p.join('');"] | |
,compiled = null | |
,template = this.getTemplate() | |
,inner = (!template) ? "<b>No template</b>" : template.replace(/[\r\t\n]/g, " "); | |
for (key in operators) { | |
if (operators.hasOwnProperty(key)) { | |
inner = inner.replace(operators[key][0], operators[key][1]); | |
} | |
} | |
// This method will evaluate in the template. | |
inner = inner.replace(new RegExp(open + '([\\s\\S]+?)' + close, 'g'), function (match, code) { | |
return "');" + code.replace(/\\'/g, "'").replace(/[\r\n\t]/g, ' ') + ";p.push('"; | |
}); | |
// Close off the template string. | |
inner = inner.split("\t").join("');").split("\r").join("\\'"); | |
try { | |
body = head + wrapper.join(inner); | |
compiled = new Function('__o', head + wrapper.join(inner)); | |
} catch (ex) { | |
console.error(ex); | |
throw new Error('Syntax error in template: function body :: ' + body); | |
} | |
return compiled(data); | |
} | |
}; | |
return TemplateMethods; | |
}); |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment