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

Commit 4962888

Browse files
committed
feat(ngOptions): add $value variable for easier use of trackBy+selectAs
1 parent c00903b commit 4962888

File tree

2 files changed

+23
-23
lines changed

2 files changed

+23
-23
lines changed

src/ng/directive/ngOptions.js

+15-11
Original file line numberDiff line numberDiff line change
@@ -66,9 +66,7 @@ var ngOptionsMinErr = minErr('ngOptions');
6666
*
6767
* ### `select` **`as`** and **`track by`**
6868
*
69-
* <div class="alert alert-warning">
70-
* Be careful when using `select` **`as`** and **`track by`** in the same expression.
71-
* </div>
69+
* When using `select` **`as`** and **`track by`** in the same expression use the `$value` variable.
7270
*
7371
* Given this array of items on the $scope:
7472
*
@@ -110,6 +108,14 @@ var ngOptionsMinErr = minErr('ngOptions');
110108
* expression evaluates to `items[0].subItem.id` (which is undefined). As a result, the model value
111109
* is not matched against any `<option>` and the `<select>` appears as having no selected value.
112110
*
111+
* here is the fixed version of the broken example above.
112+
*
113+
* ```html
114+
* <select ng-options="item.subItem as item.label for item in items track by $value.id" ng-model="selected"></select>
115+
* ```
116+
* ```js
117+
* $scope.selected = $scope.items[0].subItem;
118+
* ```
113119
*
114120
* @param {string} ngModel Assignable angular expression to data-bind to.
115121
* @param {string=} name Property name of the form under which the control is published.
@@ -283,7 +289,7 @@ var ngOptionsDirective = ['$compile', '$document', '$parse', function($compile,
283289
function(value, locals) { return trackByFn(scope, locals); } :
284290
function getHashOfValue(value) { return hashKey(value); };
285291
var getTrackByValue = function(value, key) {
286-
return getTrackByValueFn(value, getLocals(value, key));
292+
return getTrackByValueFn(value, getLocals(value, key, true));
287293
};
288294

289295
var displayFn = $parse(match[2] || match[1]);
@@ -292,12 +298,10 @@ var ngOptionsDirective = ['$compile', '$document', '$parse', function($compile,
292298
var valuesFn = $parse(match[8]);
293299

294300
var locals = {};
295-
var getLocals = keyName ? function(value, key) {
296-
locals[keyName] = key;
297-
locals[valueName] = value;
298-
return locals;
299-
} : function(value) {
301+
var getLocals = function(value, key, isViewValue) {
302+
if (keyName) locals[keyName] = key;
300303
locals[valueName] = value;
304+
locals['$value'] = isViewValue ? value : viewValueFn(value, locals);
301305
return locals;
302306
};
303307

@@ -343,7 +347,7 @@ var ngOptionsDirective = ['$compile', '$document', '$parse', function($compile,
343347
var key = (optionValues === optionValuesKeys) ? index : optionValuesKeys[index];
344348
var value = optionValues[key];
345349

346-
var locals = getLocals(value, key);
350+
var locals = getLocals(value, key, true);
347351
var selectValue = getTrackByValueFn(value, locals);
348352
watchedArray.push(selectValue);
349353

@@ -376,7 +380,7 @@ var ngOptionsDirective = ['$compile', '$document', '$parse', function($compile,
376380
for (var index = 0; index < optionValuesLength; index++) {
377381
var key = (optionValues === optionValuesKeys) ? index : optionValuesKeys[index];
378382
var value = optionValues[key];
379-
var locals = getLocals(value, key);
383+
var locals = getLocals(value, key, false);
380384
var viewValue = viewValueFn(scope, locals);
381385
var selectValue = getTrackByValueFn(viewValue, locals);
382386
var label = displayFn(scope, locals);

test/ng/directive/ngOptionsSpec.js

+8-12
Original file line numberDiff line numberDiff line change
@@ -1453,22 +1453,18 @@ describe('ngOptions', function() {
14531453
});
14541454

14551455

1456-
/**
1457-
* This behavior is broken and should probably be cleaned up later as track by and select as
1458-
* aren't compatible.
1459-
*/
14601456
describe('selectAs+trackBy expression', function() {
14611457
beforeEach(function() {
14621458
scope.arr = [{subItem: {label: 'ten', id: 10}}, {subItem: {label: 'twenty', id: 20}}];
14631459
scope.obj = {'10': {subItem: {id: 10, label: 'ten'}}, '20': {subItem: {id: 20, label: 'twenty'}}};
14641460
});
14651461

14661462

1467-
it('It should use the "value" variable to represent items in the array as well as for the ' +
1463+
it('It should use the "$value" variable to represent items in the array as well as for the ' +
14681464
'selected values in track by expression (single&array)', function() {
14691465
createSelect({
14701466
'ng-model': 'selected',
1471-
'ng-options': 'item.subItem as item.subItem.label for item in arr track by (item.id || item.subItem.id)'
1467+
'ng-options': 'item.subItem as item.subItem.label for item in arr track by $value.id'
14721468
});
14731469

14741470
// First test model -> view
@@ -1502,12 +1498,12 @@ describe('ngOptions', function() {
15021498
});
15031499

15041500

1505-
it('It should use the "value" variable to represent items in the array as well as for the ' +
1501+
it('It should use the "$value" variable to represent items in the array as well as for the ' +
15061502
'selected values in track by expression (multiple&array)', function() {
15071503
createSelect({
15081504
'ng-model': 'selected',
15091505
'multiple': true,
1510-
'ng-options': 'item.subItem as item.subItem.label for item in arr track by (item.id || item.subItem.id)'
1506+
'ng-options': 'item.subItem as item.subItem.label for item in arr track by $value.id'
15111507
});
15121508

15131509
// First test model -> view
@@ -1543,12 +1539,12 @@ describe('ngOptions', function() {
15431539
});
15441540

15451541

1546-
it('It should use the "value" variable to represent items in the array as well as for the ' +
1542+
it('It should use the "$value" variable to represent items in the array as well as for the ' +
15471543
'selected values in track by expression (multiple&object)', function() {
15481544
createSelect({
15491545
'ng-model': 'selected',
15501546
'multiple': true,
1551-
'ng-options': 'val.subItem as val.subItem.label for (key, val) in obj track by (val.id || val.subItem.id)'
1547+
'ng-options': 'val.subItem as val.subItem.label for (key, val) in obj track by $value.id'
15521548
});
15531549

15541550
// First test model -> view
@@ -1588,11 +1584,11 @@ describe('ngOptions', function() {
15881584
});
15891585

15901586

1591-
it('It should use the "value" variable to represent items in the array as well as for the ' +
1587+
it('It should use the "$value" variable to represent items in the array as well as for the ' +
15921588
'selected values in track by expression (single&object)', function() {
15931589
createSelect({
15941590
'ng-model': 'selected',
1595-
'ng-options': 'val.subItem as val.subItem.label for (key, val) in obj track by (val.id || val.subItem.id)'
1591+
'ng-options': 'val.subItem as val.subItem.label for (key, val) in obj track by $value.id'
15961592
});
15971593

15981594
// First test model -> view

0 commit comments

Comments
 (0)