Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit 631076a

Browse files
committed
fix($parse): do not shallow-watch computed property keys
Shallow watching is not enough when an object implements a non-pure toString
1 parent 341f8db commit 631076a

File tree

2 files changed

+30
-9
lines changed

2 files changed

+30
-9
lines changed

src/ng/parse.js

+2-1
Original file line numberDiff line numberDiff line change
@@ -744,7 +744,8 @@ function findConstantAndWatchExpressions(ast, $filter, parentIsPure) {
744744
allConstants = allConstants && property.value.constant;
745745
argsToWatch.push.apply(argsToWatch, property.value.toWatch);
746746
if (property.computed) {
747-
findConstantAndWatchExpressions(property.key, $filter, astIsPure);
747+
//`{[key]: value}` implicitly does `key.toString()` which may be non-pure
748+
findConstantAndWatchExpressions(property.key, $filter, /*parentIsPure=*/false);
748749
allConstants = allConstants && property.key.constant;
749750
argsToWatch.push.apply(argsToWatch, property.key.toWatch);
750751
}

test/ng/parseSpec.js

+28-8
Original file line numberDiff line numberDiff line change
@@ -3805,35 +3805,55 @@ describe('parser', function() {
38053805

38063806
it('should watch ES6 object computed property changes', function() {
38073807
var count = 0;
3808-
var values = [];
3808+
var lastValue;
38093809

38103810
scope.$watch('{[a]: true}', function(val) {
38113811
count++;
3812-
values.push(val);
3813-
}, true);
3812+
lastValue = val;
3813+
});
38143814

38153815
scope.$digest();
38163816
expect(count).toBe(1);
3817-
expect(values[0]).toEqual({'undefined': true});
3817+
expect(lastValue).toEqual({'undefined': true});
38183818

38193819
scope.$digest();
38203820
expect(count).toBe(1);
3821-
expect(values[0]).toEqual({'undefined': true});
3821+
expect(lastValue).toEqual({'undefined': true});
38223822

38233823
scope.a = true;
38243824
scope.$digest();
38253825
expect(count).toBe(2);
3826-
expect(values[1]).toEqual({'true': true});
3826+
expect(lastValue).toEqual({'true': true});
38273827

38283828
scope.a = 'abc';
38293829
scope.$digest();
38303830
expect(count).toBe(3);
3831-
expect(values[2]).toEqual({'abc': true});
3831+
expect(lastValue).toEqual({'abc': true});
38323832

38333833
scope.a = undefined;
38343834
scope.$digest();
38353835
expect(count).toBe(4);
3836-
expect(values[3]).toEqual({'undefined': true});
3836+
expect(lastValue).toEqual({'undefined': true});
3837+
});
3838+
3839+
it('should not shallow-watch ES6 object computed properties in case of stateful toString', function() {
3840+
var count = 0;
3841+
var lastValue;
3842+
3843+
scope.$watch('{[a]: true}', function(val) {
3844+
count++;
3845+
lastValue = val;
3846+
});
3847+
3848+
scope.a = {toString: function() { return this.b; }};
3849+
scope.a.b = 1;
3850+
3851+
//TODO: would be great if it didn't throw!
3852+
expect(function() { scope.$apply(); }).toThrowMinErr('$rootScope', 'infdig');
3853+
expect(lastValue).toEqual({1: true});
3854+
3855+
expect(function() { scope.$apply('a.b = 2'); }).toThrowMinErr('$rootScope', 'infdig');
3856+
expect(lastValue).toEqual({2: true});
38373857
});
38383858
});
38393859

0 commit comments

Comments
 (0)