diff --git a/benchmarks/parsed-expressions-bp/main.html b/benchmarks/parsed-expressions-bp/main.html
index d7f65b742ce4..9884da94e6bc 100755
--- a/benchmarks/parsed-expressions-bp/main.html
+++ b/benchmarks/parsed-expressions-bp/main.html
@@ -31,6 +31,11 @@
+
+
+
+
+
@@ -134,6 +139,17 @@
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/ng/compile.js b/src/ng/compile.js
index f0d3fe197bfd..b6e813fea17b 100644
--- a/src/ng/compile.js
+++ b/src/ng/compile.js
@@ -1699,7 +1699,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
attrs[attrName], newIsolateScopeDirective.name);
};
lastValue = isolateBindingContext[scopeName] = parentGet(scope);
- var unwatch = scope.$watch($parse(attrs[attrName], function parentValueWatch(parentValue) {
+ var parentValueWatch = function parentValueWatch(parentValue) {
if (!compare(parentValue, isolateBindingContext[scopeName])) {
// we are out of sync and need to copy
if (!compare(parentValue, lastValue)) {
@@ -1711,7 +1711,9 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
}
}
return lastValue = parentValue;
- }), null, parentGet.literal);
+ };
+ parentValueWatch.externalInput = true;
+ var unwatch = scope.$watch($parse(attrs[attrName], parentValueWatch), null, parentGet.literal);
isolateScope.$on('$destroy', unwatch);
break;
diff --git a/src/ng/parse.js b/src/ng/parse.js
index 69e71e201698..ff34a1120689 100644
--- a/src/ng/parse.js
+++ b/src/ng/parse.js
@@ -112,7 +112,6 @@ var OPERATORS = extend(createMap(), {
'/':function(self, locals, a,b){return a(self, locals)/b(self, locals);},
'%':function(self, locals, a,b){return a(self, locals)%b(self, locals);},
'^':function(self, locals, a,b){return a(self, locals)^b(self, locals);},
- '=':noop,
'===':function(self, locals, a, b){return a(self, locals)===b(self, locals);},
'!==':function(self, locals, a, b){return a(self, locals)!==b(self, locals);},
'==':function(self, locals, a,b){return a(self, locals)==b(self, locals);},
@@ -125,8 +124,11 @@ var OPERATORS = extend(createMap(), {
'||':function(self, locals, a,b){return a(self, locals)||b(self, locals);},
'&':function(self, locals, a,b){return a(self, locals)&b(self, locals);},
// '|':function(self, locals, a,b){return a|b;},
- '|':function(self, locals, a,b){return b(self, locals)(self, locals, a(self, locals));},
- '!':function(self, locals, a){return !a(self, locals);}
+ '!':function(self, locals, a){return !a(self, locals);},
+
+ //Tokenized as operators but parsed as assignment/filters
+ '=':true,
+ '|':true
});
/* jshint bitwise: true */
var ESCAPE = {"n":"\n", "f":"\f", "r":"\r", "t":"\t", "v":"\v", "'":"'", '"':'"'};
@@ -375,6 +377,10 @@ Lexer.prototype = {
};
+function isConstant(exp) {
+ return exp.constant;
+}
+
/**
* @constructor
*/
@@ -492,7 +498,8 @@ Parser.prototype = {
return extend(function(self, locals) {
return fn(self, locals, right);
}, {
- constant:right.constant
+ constant:right.constant,
+ inputs: [right]
});
},
@@ -500,15 +507,16 @@ Parser.prototype = {
return extend(function(self, locals){
return left(self, locals) ? middle(self, locals) : right(self, locals);
}, {
- constant: left.constant && middle.constant && right.constant
+ constant:left.constant && middle.constant && right.constant
});
},
- binaryFn: function(left, fn, right) {
+ binaryFn: function(left, fn, right, isBranching) {
return extend(function(self, locals) {
return fn(self, locals, left, right);
}, {
- constant:left.constant && right.constant
+ constant: left.constant && right.constant,
+ inputs: !isBranching && [left, right]
});
},
@@ -537,12 +545,12 @@ Parser.prototype = {
var left = this.expression();
var token;
while ((token = this.expect('|'))) {
- left = this.binaryFn(left, token.fn, this.filter());
+ left = this.filter(left);
}
return left;
},
- filter: function() {
+ filter: function(inputFn) {
var token = this.expect();
var fn = this.$filter(token.text);
var argsFn;
@@ -556,7 +564,10 @@ Parser.prototype = {
}
}
- return valueFn(function $parseFilter(self, locals, input) {
+ var inputs = [inputFn].concat(argsFn || []);
+
+ return extend(function $parseFilter(self, locals) {
+ var input = inputFn(self, locals);
if (args) {
args[0] = input;
@@ -569,6 +580,9 @@ Parser.prototype = {
}
return fn(input);
+ }, {
+ constant: !fn.externalInput && inputs.every(isConstant),
+ inputs: !fn.externalInput && inputs
});
},
@@ -586,9 +600,11 @@ Parser.prototype = {
this.text.substring(0, token.index) + '] can not be assigned to', token);
}
right = this.ternary();
- return function $parseAssignment(scope, locals) {
+ return extend(function $parseAssignment(scope, locals) {
return left.assign(scope, right(scope, locals), locals);
- };
+ }, {
+ inputs: [left, right]
+ });
}
return left;
},
@@ -613,7 +629,7 @@ Parser.prototype = {
var left = this.logicalAND();
var token;
while ((token = this.expect('||'))) {
- left = this.binaryFn(left, token.fn, this.logicalAND());
+ left = this.binaryFn(left, token.fn, this.logicalAND(), true);
}
return left;
},
@@ -622,7 +638,7 @@ Parser.prototype = {
var left = this.equality();
var token;
if ((token = this.expect('&&'))) {
- left = this.binaryFn(left, token.fn, this.logicalAND());
+ left = this.binaryFn(left, token.fn, this.logicalAND(), true);
}
return left;
},
@@ -757,7 +773,6 @@ Parser.prototype = {
// This is used with json array declaration
arrayDeclaration: function () {
var elementFns = [];
- var allConstant = true;
if (this.peekToken().text !== ']') {
do {
if (this.peek(']')) {
@@ -766,9 +781,6 @@ Parser.prototype = {
}
var elementFn = this.expression();
elementFns.push(elementFn);
- if (!elementFn.constant) {
- allConstant = false;
- }
} while (this.expect(','));
}
this.consume(']');
@@ -781,41 +793,38 @@ Parser.prototype = {
return array;
}, {
literal: true,
- constant: allConstant
+ constant: elementFns.every(isConstant),
+ inputs: elementFns
});
},
object: function () {
- var keyValues = [];
- var allConstant = true;
+ var keys = [], values = [];
if (this.peekToken().text !== '}') {
do {
if (this.peek('}')) {
// Support trailing commas per ES5.1.
break;
}
- var token = this.expect(),
- key = token.string || token.text;
+ var token = this.expect();
+ keys.push(token.string || token.text);
this.consume(':');
var value = this.expression();
- keyValues.push({key: key, value: value});
- if (!value.constant) {
- allConstant = false;
- }
+ values.push(value);
} while (this.expect(','));
}
this.consume('}');
return extend(function $parseObjectLiteral(self, locals) {
var object = {};
- for (var i = 0, ii = keyValues.length; i < ii; i++) {
- var keyValue = keyValues[i];
- object[keyValue.key] = keyValue.value(self, locals);
+ for (var i = 0, ii = values.length; i < ii; i++) {
+ object[keys[i]] = values[i](self, locals);
}
return object;
}, {
literal: true,
- constant: allConstant
+ constant: values.every(isConstant),
+ inputs: values
});
}
};
@@ -1043,6 +1052,9 @@ function $ParseProvider() {
parsedExpression.$$watchDelegate = parsedExpression.literal ?
oneTimeLiteralWatchDelegate : oneTimeWatchDelegate;
}
+ else if (parsedExpression.inputs) {
+ parsedExpression.$$watchDelegate = inputsWatchDelegate;
+ }
cache[cacheKey] = parsedExpression;
}
@@ -1056,6 +1068,81 @@ function $ParseProvider() {
}
};
+ function collectExpressionInputs(inputs, list) {
+ for (var i = 0, ii = inputs.length; i < ii; i++) {
+ var input = inputs[i];
+ if (!input.constant) {
+ if (input.inputs) {
+ collectExpressionInputs(input.inputs, list);
+ }
+ else if (-1 === list.indexOf(input)) {
+ list.push(input);
+ }
+ }
+ }
+
+ return list;
+ }
+
+ function simpleEquals(o1, o2) {
+ if (o1 == null || o2 == null) return o1 === o2; // null/undefined
+
+ if (typeof o1 === "object") {
+ //The same object is not supported because it may have been mutated
+ if (o1 === o2) return false;
+
+ if (typeof o2 !== "object") return false;
+
+ //Convert to primitive if possible
+ o1 = o1.valueOf();
+ o2 = o2.valueOf();
+
+ //If the type became a non-object then we can use the primitive check below
+ if (typeof o1 === "object") return false;
+ }
+
+ //Primitive or NaN
+ return o1 === o2 || (o1 !== o1 && o2 !== o2);
+ }
+
+ function inputsWatchDelegate(scope, listener, objectEquality, parsedExpression) {
+ var inputExpressions = parsedExpression.$$inputs ||
+ (parsedExpression.$$inputs = collectExpressionInputs(parsedExpression.inputs, []));
+
+ var inputs = [simpleEquals/*=something that will never equal an evaluated input*/];
+ var lastResult;
+
+ if (1 === inputExpressions.length) {
+ inputs = inputs[0];
+ inputExpressions = inputExpressions[0];
+ return scope.$watch(function expressionInputWatch(scope) {
+ var newVal = inputExpressions(scope);
+ if (!simpleEquals(newVal, inputs)) {
+ lastResult = parsedExpression(scope);
+ inputs = newVal;
+ }
+ return lastResult;
+ }, listener, objectEquality);
+ }
+
+ return scope.$watch(function expressionInputsWatch(scope) {
+ var changed = false;
+
+ for (var i=0, ii=inputExpressions.length; i');
- expect(logs).toEqual([1, 2, 1, 2]);
+ expect(logs.length).toBe(0);
}));
@@ -894,12 +897,15 @@ describe('ngRepeat', function() {
// This creates one item, but it has no parent so we can't get to it
$rootScope.items = [1, 2];
$rootScope.$apply();
+ expect(logs).toContain(1);
+ expect(logs).toContain(2);
+ logs.length = 0;
// This cleans up to prevent memory leak
$rootScope.items = [];
$rootScope.$apply();
expect(sortedHtml(element)).toBe('--');
- expect(logs).toEqual([1, 2, 1, 2]);
+ expect(logs.length).toBe(0);
}));
diff --git a/test/ng/parseSpec.js b/test/ng/parseSpec.js
index 618149cee2fe..03eac03e1119 100644
--- a/test/ng/parseSpec.js
+++ b/test/ng/parseSpec.js
@@ -1311,6 +1311,153 @@ describe('parser', function() {
});
});
+ describe('watched $parse expressions', function() {
+ it('should respect short-circuiting AND if it could have side effects', function() {
+ var bCalled = 0;
+ scope.b = function() { bCalled++; }
+
+ scope.$watch("a && b()");
+ scope.$digest();
+ scope.$digest();
+ expect(bCalled).toBe(0);
+
+ scope.a = true;
+ scope.$digest();
+ expect(bCalled).toBe(1);
+ scope.$digest();
+ expect(bCalled).toBe(2);
+ });
+ it('should respect short-circuiting OR if it could have side effects', function() {
+ var bCalled = false;
+ scope.b = function() { bCalled = true; }
+
+ scope.$watch("a || b()");
+ scope.$digest();
+ expect(bCalled).toBe(true);
+
+ bCalled = false;
+ scope.a = true;
+ scope.$digest();
+ expect(bCalled).toBe(false);
+ });
+ it('should respect the branching ternary operator if it could have side effects', function() {
+ var bCalled = false;
+ scope.b = function() { bCalled = true; }
+
+ scope.$watch("a ? b() : 1");
+ scope.$digest();
+ expect(bCalled).toBe(false);
+
+ scope.a = true;
+ scope.$digest();
+ expect(bCalled).toBe(true);
+ });
+ it('should not invoke filters unless the input/arguments change', function() {
+ var filterCalled = false;
+ $filterProvider.register('foo', valueFn(function(input) {
+ filterCalled = true;
+ return input;
+ }));
+
+ scope.$watch("a | foo:b:1");
+ scope.a = 0;
+ scope.$digest();
+ expect(filterCalled).toBe(true);
+
+ filterCalled = false;
+ scope.$digest();
+ expect(filterCalled).toBe(false);
+
+ scope.a++;
+ scope.$digest();
+ expect(filterCalled).toBe(true);
+ });
+ it('should invoke filters if they are marked as having externalInput', function() {
+ var filterCalled = false;
+ $filterProvider.register('foo', valueFn(extend(function(input) {
+ filterCalled = true;
+ return input;
+ }, {externalInput: true})));
+
+ scope.$watch("a | foo:b:1");
+ scope.a = 0;
+ scope.$digest();
+ expect(filterCalled).toBe(true);
+
+ filterCalled = false;
+ scope.$digest();
+ expect(filterCalled).toBe(true);
+ });
+ it('should not invoke interceptorFns unless the input changes', inject(function($parse) {
+ var called = false;
+ function interceptor(v) {
+ called = true;
+ return v;
+ }
+ scope.$watch($parse("a", interceptor));
+ scope.a = scope.b = 0;
+ scope.$digest();
+ expect(called).toBe(true);
+
+ called = false;
+ scope.$digest();
+ expect(called).toBe(false);
+
+ scope.a++;
+ scope.$digest();
+ expect(called).toBe(true);
+ }));
+ it('should invoke interceptorFns if the yare marked as having externalInput', inject(function($parse) {
+ var called = false;
+ function interceptor() {
+ called = true;
+ }
+ interceptor.externalInput = true;
+
+ scope.$watch($parse("a", interceptor));
+ scope.a = 0;
+ scope.$digest();
+ expect(called).toBe(true);
+
+ called = false;
+ scope.$digest();
+ expect(called).toBe(true);
+
+ scope.a++;
+ called = false;
+ scope.$digest();
+ expect(called).toBe(true);
+ }));
+ it('should treat constants inputted to filters as constants', inject(function($parse) {
+ var filterCalls = 0;
+ $filterProvider.register('foo', valueFn(function(input) {
+ filterCalls++;
+ return input;
+ }));
+
+ var parsed = $parse('{x: 1} | foo:1');
+
+ expect( parsed.constant ).toBe(true);
+
+ var watcherCalls = 0;
+ scope.$watch(parsed, function(input) {
+ expect(input).toEqual({x:1});
+ watcherCalls++;
+ });
+
+ scope.$digest();
+ expect(filterCalls).toBe(1);
+ expect(watcherCalls).toBe(1);
+
+ scope.$digest();
+ expect(filterCalls).toBe(1);
+ expect(watcherCalls).toBe(1);
+ }));
+ it('should not treat constants passed to filters with externalInput as constants', inject(function($parse) {
+
+ }));
+ });
+
describe('locals', function() {
it('should expose local variables', inject(function($parse) {
expect($parse('a')({a: 0}, {a: 1})).toEqual(1);