support for templates

This commit is contained in:
Misko Hevery 2010-03-18 14:43:49 -07:00
parent 7634a3ed52
commit df607da0d1
4 changed files with 82 additions and 36 deletions

Binary file not shown.

View file

@ -1,5 +1,6 @@
function Scope(initialState, name) { function Scope(initialState, name) {
this.widgets = []; this.widgets = [];
this.evals = [];
this.watchListeners = {}; this.watchListeners = {};
this.name = name; this.name = name;
initialState = initialState || {}; initialState = initialState || {};
@ -11,6 +12,7 @@ function Scope(initialState, name) {
this.state['$root'] = this.state; this.state['$root'] = this.state;
} }
this.set('$watch', bind(this, this.addWatchListener)); this.set('$watch', bind(this, this.addWatchListener));
this.set('$eval', bind(this, this.addEval));
}; };
Scope.expressionCache = {}; Scope.expressionCache = {};
@ -48,17 +50,23 @@ Scope.prototype = {
updateView: function() { updateView: function() {
var self = this; var self = this;
this.fireWatchers(); this.fireWatchers();
_.each(this.widgets, function(widget){ foreach(this.widgets, function(widget){
self.evalWidget(widget, "", {}, function(){ self.evalWidget(widget, "", {}, function(){
this.updateView(self); this.updateView(self);
}); });
}); });
foreach(this.evals, bind(this, this.apply));
}, },
addWidget: function(controller) { addWidget: function(controller) {
if (controller) this.widgets.push(controller); if (controller) this.widgets.push(controller);
}, },
addEval: function(fn, listener) {
// todo: this should take a function/string and a listener
this.evals.push(fn);
},
isProperty: function(exp) { isProperty: function(exp) {
for ( var i = 0; i < exp.length; i++) { for ( var i = 0; i < exp.length; i++) {
var ch = exp.charAt(i); var ch = exp.charAt(i);
@ -190,8 +198,7 @@ Scope.prototype = {
}, },
fireWatchers: function() { fireWatchers: function() {
var self = this; var self = this, fired = false;
var fired = false;
foreach(this.watchListeners, function(watcher) { foreach(this.watchListeners, function(watcher) {
var value = self.eval(watcher.expression); var value = self.eval(watcher.expression);
if (value !== watcher.lastValue) { if (value !== watcher.lastValue) {
@ -206,6 +213,6 @@ Scope.prototype = {
}, },
apply: function(fn) { apply: function(fn) {
fn.apply(this.state, slice(arguments, 0, arguments.length)); fn.apply(this.state, slice.call(arguments, 1, arguments.length));
} }
}; };

View file

@ -58,12 +58,12 @@ angular.directive("bind-attr", function(expression, element){
angular.directive("repeat", function(expression, element){ angular.directive("repeat", function(expression, element){
var anchor = document.createComment(expression); var anchor = document.createComment(expression);
jQuery(element).replace(anchor); jQuery(element).replace(anchor);
var template = this.templetize(element); var template = this.compile(element);
var lhs = "item"; var lhs = "item";
var rhs = "items"; var rhs = "items";
var children = [];
return function(){ return function(){
this.$watch(rhs, function(items){ var children = [];
this.$eval(rhs, function(items){
foreach(children, function(child){ foreach(children, function(child){
child.element.remove(); child.element.remove();
}); });
@ -102,7 +102,7 @@ angular.directive("action", function(expression, element){
//ng-eval //ng-eval
angular.directive("eval", function(expression, element){ angular.directive("eval", function(expression, element){
return function(){ return function(){
this.$onUpdate( expression); this.$eval(expression);
}; };
}); });
//ng-watch //ng-watch

View file

@ -1,3 +1,9 @@
/**
* Template provides directions an how to bind to a given element.
* It contains a list of init functions which need to be called to
* bind to a new instance of elements. It also provides a list
* of child paths which contain child templates
*/
function Template() { function Template() {
this.paths = []; this.paths = [];
this.children = []; this.children = [];
@ -26,6 +32,11 @@ Template.prototype = {
} }
}, },
setExclusiveInit: function(init) {
this.inits = [init];
this.addInit = noop;
},
addChild: function(index, template) { addChild: function(index, template) {
this.paths.push(index); this.paths.push(index);
@ -33,39 +44,22 @@ Template.prototype = {
} }
}; };
///////////////////////////////////
// Compiler
//////////////////////////////////
function Compiler(directives){ function Compiler(directives){
this.directives = directives; this.directives = directives;
} }
DIRECTIVE = /^ng-(.*)$/; DIRECTIVE = /^ng-(.*)$/;
/**
* return {
* element:
* init: function(element){...}
* }
*
* internal data structure: {
* paths: [4, 5, 6],
* directive: name,
* init: function(expression, element){}
* }
*
* template : {
* inits: [fn(), fn()}
* paths: [1, 5],
* templates: [
* inits: []
* paths: []
* templates:
* ]
* }
*/
Compiler.prototype = { Compiler.prototype = {
compile: function(element) { compile: function(element) {
var template = this.templetize(element); var template = this.templetize(element) || new Template();
return function(){ return function(element){
var scope = new Scope(); var scope = new Scope();
scope.element = element;
return { return {
scope: scope, scope: scope,
element:element, element:element,
@ -79,17 +73,24 @@ Compiler.prototype = {
childTemplate, recurse = true; childTemplate, recurse = true;
// Process attributes/directives // Process attributes/directives
for (i = 0, items = element.attributes, length = items.length; for (i = 0, items = element.attributes || [], length = items.length;
i < length; i++) { i < length; i++) {
item = items[i]; item = items[i];
var match = item.name.match(DIRECTIVE); var match = item.name.match(DIRECTIVE);
if (match) { if (match) {
directive = this.directives[match[1]]; directive = this.directives[match[1]];
if (directive) { if (directive) {
init = directive.call({}, item.value, element); init = directive.call(this, item.value, element);
template = template || new Template(); template = template || new Template();
template.addInit(init); if (directive.exclusive) {
template.setExclusiveInit(init);
i = length; // quit iterations
} else {
template.addInit(init);
}
recurse = recurse && init; recurse = recurse && init;
} else {
error("Directive '" + match[0] + "' is not recognized.");
} }
} }
} }
@ -136,7 +137,7 @@ describe('compiler', function(){
}; };
compiler = new Compiler(directives); compiler = new Compiler(directives);
compile = function(html){ compile = function(html){
var e = element(html); var e = element("<div>" + html + "</div>");
var view = compiler.compile(e)(e); var view = compiler.compile(e)(e);
view.init(); view.init();
return view.scope; return view.scope;
@ -183,4 +184,42 @@ describe('compiler', function(){
var scope = compile('<span ng-hello="misko" ng-stop="true"><span ng-hello="adam"/></span>'); var scope = compile('<span ng-hello="misko" ng-stop="true"><span ng-hello="adam"/></span>');
expect(log).toEqual("hello misko"); expect(log).toEqual("hello misko");
}); });
it('should allow creation of templates', function(){
directives.duplicate = function(expr, element){
var template,
marker = document.createComment("marker"),
parentNode = element.parentNode;
parentNode.insertBefore(marker, element);
parentNode.removeChild(element);
element.removeAttribute("ng-duplicate");
template = this.compile(element);
return function(marker) {
var parentNode = marker.parentNode;
this.$eval(function() {
parentNode.insertBefore(
template(element.cloneNode(true)).element,
marker.nextSibling);
});
};
};
var scope = compile('before<span ng-duplicate="expr">x</span>after');
expect($(scope.element).html()).toEqual('before<!--marker-->after');
scope.updateView();
expect($(scope.element).html()).toEqual('before<!--marker--><span>x</span>after');
scope.updateView();
expect($(scope.element).html()).toEqual('before<!--marker--><span>x</span><span>x</span>after');
});
it('should allow for exculsive tags which suppress others', function(){
directives.exclusive = function(){
return function() {
log += ('exclusive');
};
};
directives.exclusive.exclusive = true;
compile('<span ng-hello="misko", ng-exclusive/>');
expect(log).toEqual('exclusive');
});
}); });