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

feat($compile): allow $watchCollection to be used in bi-directional bindings #9725

Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
20 changes: 14 additions & 6 deletions src/ng/compile.js
Original file line number Diff line number Diff line change
Expand Up @@ -164,7 +164,9 @@
* value of `parentModel` on the parent scope. Any changes to `parentModel` will be reflected
* in `localModel` and any changes in `localModel` will reflect in `parentModel`. If the parent
* scope property doesn't exist, it will throw a NON_ASSIGNABLE_MODEL_EXPRESSION exception. You
* can avoid this behavior using `=?` or `=?attr` in order to flag the property as optional.
* can avoid this behavior using `=?` or `=?attr` in order to flag the property as optional. If
* you want to shallow watch for changes (i.e. $watchCollection instead of $watch) you can use
* `=*` or `=*attr` (`=*?` or `=*?attr` if the property is optional).
*
* * `&` or `&attr` - provides a way to execute an expression in the context of the parent scope.
* If no `attr` name is specified then the attribute name is assumed to be the same as the
Expand Down Expand Up @@ -691,7 +693,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
var EVENT_HANDLER_ATTR_REGEXP = /^(on[a-z]+|formaction)$/;

function parseIsolateBindings(scope, directiveName) {
var LOCAL_REGEXP = /^\s*([@=&])(\??)\s*(\w*)\s*$/;
var LOCAL_REGEXP = /^\s*([@&]|=(\*?))(\??)\s*(\w*)\s*$/;

var bindings = {};

Expand All @@ -706,9 +708,10 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
}

bindings[scopeName] = {
attrName: match[3] || scopeName,
mode: match[1],
optional: match[2] === '?'
mode: match[1][0],
collection: match[2] === '*',
optional: match[3] === '?',
attrName: match[4] || scopeName
};
});

Expand Down Expand Up @@ -1890,7 +1893,12 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
return lastValue = parentValue;
};
parentValueWatch.$stateful = true;
var unwatch = scope.$watch($parse(attrs[attrName], parentValueWatch), null, parentGet.literal);
var unwatch;
if (definition.collection) {
unwatch = scope.$watchCollection(attrs[attrName], parentValueWatch);
} else {
unwatch = scope.$watch($parse(attrs[attrName], parentValueWatch), null, parentGet.literal);
}
isolateScope.$on('$destroy', unwatch);
break;

Expand Down
49 changes: 49 additions & 0 deletions test/ng/compileSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -3065,6 +3065,8 @@ describe('$compile', function() {
optref: '=?',
optrefAlias: '=? optref',
optreference: '=?',
colref: '=*',
colrefAlias: '=* colref',
expr: '&',
exprAlias: '&expr'
},
Expand Down Expand Up @@ -3562,6 +3564,53 @@ describe('$compile', function() {
});


describe('collection object reference', function () {
it('should update isolate scope when origin scope changes', inject(function () {
$rootScope.collection = [{
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

this entire block is 1 space further to the right than it should be (lines 3592 to 3609)

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Done. And sorry for being so careless about indentation; I'm used to having code formatting automated on save in my own projects.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

ah don't worry too hard about it, we all mess up formatting, even on projects that use clang-format in a git hook!

name: 'Gabriel',
value: 18
}, {
name: 'Tony',
value: 91
}];
$rootScope.query = "";
$rootScope.$apply();

compile('<div><span my-component colref="collection | filter:query">');

expect(componentScope.colref).toEqual($rootScope.collection);
expect(componentScope.colrefAlias).toEqual(componentScope.colref);

$rootScope.query = "Gab";
$rootScope.$apply();

expect(componentScope.colref).toEqual([$rootScope.collection[0]]);
expect(componentScope.colrefAlias).toEqual([$rootScope.collection[0]]);
}));

it('should update origin scope when isolate scope changes', inject(function () {
$rootScope.collection = [{
name: 'Gabriel',
value: 18
}, {
name: 'Tony',
value: 91
}];

compile('<div><span my-component colref="collection">');

var newItem = {
name: 'Pablo',
value: 10
};
componentScope.colref.push(newItem);
componentScope.$apply();

expect($rootScope.collection[2]).toEqual(newItem);
}));
});


describe('executable expression', function() {
it('should allow expression execution with locals', inject(function() {
compile('<div><span my-component expr="count = count + offset">');
Expand Down