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

fix(ngOptions): use watchCollection not deep watch of ngModel #11743

Closed
wants to merge 2 commits into from
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
22 changes: 16 additions & 6 deletions src/ng/directive/ngOptions.js
Original file line number Diff line number Diff line change
Expand Up @@ -31,11 +31,21 @@ var ngOptionsMinErr = minErr('ngOptions');
* be nested into the `<select>` element. This element will then represent the `null` or "not selected"
* option. See example below for demonstration.
*
* <div class="alert alert-warning">
* **Note:** By default, `ngModel` compares by reference, not value. This is important when binding to an
* array of objects. See an example [in this jsfiddle](http://jsfiddle.net/qWzTb/). When using `track by`
* in an `ngOptions` expression, however, deep equality checks will be performed.
* </div>
* ## Complex Models (objects or collections)
*
* **Note:** By default, `ngModel` watches the model by reference, not value. This is important when
* binding any input directive to a model that is an object or a collection.
*
* Since this is a common situation for `ngOptions` the directive additionally watches the model using
* `$watchCollection` when the select has the `multiple` attribute or when there is a `track by` clause in
* the options expression. This allows ngOptions to trigger a re-rendering of the options even if the actual
* object/collection has not changed identity but only a property on the object or an item in the collection
* changes.
*
* Note that `$watchCollection` does a shallow comparison of the properties of the object (or the items in the collection
* if the model is an array). This means that changing a property deeper inside the object/collection that the
* first level will not trigger a re-rendering.
*
*
* ## `select` **`as`**
*
Expand Down Expand Up @@ -515,7 +525,7 @@ var ngOptionsDirective = ['$compile', '$parse', function($compile, $parse) {
// We also need to watch to see if the internals of the model changes, since
// ngModel only watches for object identity change
if (ngOptions.trackBy) {
scope.$watch(attr.ngModel, function() { ngModelCtrl.$render(); }, true);
scope.$watchCollection(attr.ngModel, function() { ngModelCtrl.$render(); });
}
// ------------------------------------------------------------------ //

Expand Down
89 changes: 85 additions & 4 deletions test/ng/directive/ngOptionsSpec.js
Original file line number Diff line number Diff line change
Expand Up @@ -776,6 +776,54 @@ describe('ngOptions', function() {
});


it('should re-render if an item in an array source is added/removed', function() {
createSelect({
'ng-model': 'selected',
'multiple': true,
'ng-options': 'item.id as item.label for item in arr'
});

scope.$apply(function() {
scope.selected = [10];
});
expect(element).toEqualSelectValue([10], true);

scope.$apply(function() {
scope.selected.push(20);
});
expect(element).toEqualSelectValue([10, 20], true);


scope.$apply(function() {
scope.selected.shift();
});
expect(element).toEqualSelectValue([20], true);
});


it('should handle a options containing circular references', function() {
scope.arr[0].ref = scope.arr[0];
scope.selected = [scope.arr[0]];
createSelect({
'ng-model': 'selected',
'multiple': true,
'ng-options': 'item as item.label for item in arr'
});
expect(element).toEqualSelectValue([scope.arr[0]], true);

scope.$apply(function() {
scope.selected.push(scope.arr[1]);
});
expect(element).toEqualSelectValue([scope.arr[0], scope.arr[1]], true);


scope.$apply(function() {
scope.selected.pop();
});
expect(element).toEqualSelectValue([scope.arr[0]], true);
});


it('should support single select with object source', function() {
createSelect({
'ng-model': 'selected',
Expand Down Expand Up @@ -901,8 +949,7 @@ describe('ngOptions', function() {

// Update the properties on the object in the selected array, rather than replacing the whole object
Copy link
Contributor

Choose a reason for hiding this comment

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

I think this comment should be updated, and so the test description.

Copy link
Contributor Author

Choose a reason for hiding this comment

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

@lgalfaso - looking at this a little more I am now thinking that in the specific situation of multiple + track by we should handle the value of the track by changing within each of the items in the collection of selected values.
I will put together an additional PR, which also should eliminate any form of BC to keep @gkalpak happy :-P

scope.$apply(function() {
scope.selected[0].id = 20;
scope.selected[0].label = 'new twenty';
scope.selected[0] = {id: 20, label: 'new twenty'};
});

// The value of the select should change since the id property changed
Expand Down Expand Up @@ -1042,7 +1089,7 @@ describe('ngOptions', function() {
}).not.toThrow();
});

it('should setup equality watches on ngModel changes if using trackBy', function() {
it('should re-render if a propery of the model is changed when using trackBy', function() {

createSelect({
'ng-model': 'selected',
Expand All @@ -1064,7 +1111,7 @@ describe('ngOptions', function() {

});

it('should not setup equality watches on ngModel changes if not using trackBy', function() {
it('should not re-render if a property of the model is changed when not using trackBy', function() {

createSelect({
'ng-model': 'selected',
Expand All @@ -1085,6 +1132,40 @@ describe('ngOptions', function() {
expect(element.controller('ngModel').$render).not.toHaveBeenCalled();
});


it('should handle options containing circular references (single)', function() {
scope.arr[0].ref = scope.arr[0];
createSelect({
'ng-model': 'selected',
'ng-options': 'item for item in arr track by item.id'
});

expect(function() {
scope.$apply(function() {
scope.selected = scope.arr[0];
});
}).not.toThrow();
});


it('should handle options containing circular references (multiple)', function() {
scope.arr[0].ref = scope.arr[0];
createSelect({
'ng-model': 'selected',
'multiple': true,
'ng-options': 'item for item in arr track by item.id'
});

expect(function() {
scope.$apply(function() {
scope.selected = [scope.arr[0]];
});

scope.$apply(function() {
scope.selected.push(scope.arr[1]);
});
}).not.toThrow();
});
});


Expand Down