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

Commit be66e74

Browse files
committed
fix(select): make ngOptions support selectAs and trackBy together
This commit implements a function, "getSelected()", that takes the selectAs expression into account when determining if a particular option is selected or not. Previously, the selectAs part of the ngOptions comprehension expression was being ignored. Fixes #6564
1 parent e7cf04b commit be66e74

File tree

2 files changed

+90
-24
lines changed

2 files changed

+90
-24
lines changed

src/ng/directive/select.js

+37-18
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,12 @@ var ngOptionsMinErr = minErr('ngOptions');
3535
* be bound to string values at present.
3636
* </div>
3737
*
38+
* <div class="alert alert-info">
39+
* **Note:** The values that are stored on the underlying option elements (i.e. `option.value`) are set
40+
* to either the indeces of the array (for array data sources) or the property names of
41+
* the object (for object data sources).
42+
* </div>
43+
*
3844
* @param {string} ngModel Assignable angular expression to data-bind to.
3945
* @param {string=} name Property name of the form under which the control is published.
4046
* @param {string=} required The control is considered valid only if value is entered.
@@ -69,7 +75,9 @@ var ngOptionsMinErr = minErr('ngOptions');
6975
* DOM element.
7076
* * `trackexpr`: Used when working with an array of objects. The result of this expression will be
7177
* used to identify the objects in the array. The `trackexpr` will most likely refer to the
72-
* `value` variable (e.g. `value.propertyName`).
78+
* `value` variable (e.g. `value.propertyName`). With this the selection is preserved
79+
* even when the options are recreated (e.g. reloaded from the server).
80+
* Note that this does not work when there is a `select` expression that is different from the `trackexpr`.
7381
*
7482
* @example
7583
<example module="selectExample">
@@ -326,6 +334,8 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
326334

327335
var displayFn = $parse(match[2] || match[1]),
328336
valueName = match[4] || match[6],
337+
selectAs = / as /.test(match[0]) && match[1],
338+
selectAsFn = selectAs ? $parse(selectAs) : null,
329339
keyName = match[5],
330340
groupByFn = $parse(match[3] || ''),
331341
valueFn = $parse(match[2] ? match[1] : valueName),
@@ -336,7 +346,27 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
336346
// We try to reuse these if possible
337347
// - optionGroupsCache[0] is the options with no option group
338348
// - optionGroupsCache[?][0] is the parent: either the SELECT or OPTGROUP element
339-
optionGroupsCache = [[{element: selectElement, label:''}]];
349+
optionGroupsCache = [[{element: selectElement, label:''}]],
350+
idFn, isSelected;
351+
352+
isSelected = function isSelected(selectedSet, modelValue, locals) {
353+
var compareValueFn;
354+
if (selectAsFn) {
355+
compareValueFn = selectAsFn;
356+
} else if (trackFn) {
357+
compareValueFn = trackFn;
358+
} else {
359+
compareValueFn = valueFn;
360+
}
361+
var compareValue = compareValueFn(scope, locals);
362+
if (multiple) {
363+
return isDefined(
364+
selectedSet.remove(compareValue)
365+
);
366+
} else {
367+
return modelValue === compareValue;
368+
}
369+
};
340370

341371
if (nullOption) {
342372
// compile the element since there might be bindings in it
@@ -437,7 +467,7 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
437467
var selectedSet = false;
438468
if (multiple) {
439469
var modelValue = ctrl.$modelValue;
440-
if (trackFn && isArray(modelValue)) {
470+
if (trackFn && isArray(modelValue) && !selectAs) {
441471
selectedSet = new HashMap([]);
442472
var locals = {};
443473
for (var trackIndex = 0; trackIndex < modelValue.length; trackIndex++) {
@@ -501,27 +531,16 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
501531
optionGroup = optionGroups[optionGroupName] = [];
502532
optionGroupNames.push(optionGroupName);
503533
}
504-
if (multiple) {
505-
selected = isDefined(
506-
selectedSet.remove(trackFn ? trackFn(scope, locals) : valueFn(scope, locals))
507-
);
508-
} else {
509-
if (trackFn) {
510-
var modelCast = {};
511-
modelCast[valueName] = modelValue;
512-
selected = trackFn(scope, modelCast) === trackFn(scope, locals);
513-
} else {
514-
selected = modelValue === valueFn(scope, locals);
515-
}
516-
selectedSet = selectedSet || selected; // see if at least one item is selected
517-
}
534+
535+
selected = isSelected(selectedSet, modelValue, locals);
536+
selectedSet = selectedSet || selected;
518537
label = displayFn(scope, locals); // what will be seen by the user
519538

520539
// doing displayFn(scope, locals) || '' overwrites zero values
521540
label = isDefined(label) ? label : '';
522541
optionGroup.push({
523542
// either the index into array or key from object
524-
id: trackFn ? trackFn(scope, locals) : (keyName ? keys[index] : index),
543+
id: (keyName ? keys[index] : index),
525544
label: label,
526545
selected: selected // determine if we should be selected
527546
});

test/ng/directive/selectSpec.js

+53-6
Original file line numberDiff line numberDiff line change
@@ -591,6 +591,29 @@ describe('select', function() {
591591
});
592592

593593

594+
it('should bind to scope value through expression while tracking/identifying objects', function() {
595+
createSelect({
596+
'ng-model': 'selected',
597+
'ng-options': 'item.id as item.name for item in values track by item.trackId'
598+
});
599+
600+
var selectedIdx;
601+
602+
scope.$apply(function() {
603+
scope.values = [{id: 10, trackId: 100, name: 'A'}, {id: 20, trackId: 200, name: 'B'}];
604+
scope.selected = 10;
605+
});
606+
607+
expect(element.val()).toEqual('0');
608+
609+
scope.$apply(function() {
610+
scope.selected = 20;
611+
});
612+
613+
expect(element.val()).toEqual('1');
614+
});
615+
616+
594617
it('should render a list', function() {
595618
createSingleSelect();
596619

@@ -924,23 +947,23 @@ describe('select', function() {
924947
{id: 2, name: 'second'},
925948
{id: 3, name: 'third'},
926949
{id: 4, name: 'forth'}];
927-
scope.selected = {id: 2};
950+
scope.selected = scope.values[1];
928951
});
929952

930-
expect(element.val()).toEqual('2');
953+
expect(element.val()).toEqual('1');
931954

932955
var first = jqLite(element.find('option')[0]);
933956
expect(first.text()).toEqual('first');
934-
expect(first.attr('value')).toEqual('1');
957+
expect(first.attr('value')).toEqual('0');
935958
var forth = jqLite(element.find('option')[3]);
936959
expect(forth.text()).toEqual('forth');
937-
expect(forth.attr('value')).toEqual('4');
960+
expect(forth.attr('value')).toEqual('3');
938961

939962
scope.$apply(function() {
940963
scope.selected = scope.values[3];
941964
});
942965

943-
expect(element.val()).toEqual('4');
966+
expect(element.val()).toEqual('3');
944967
});
945968

946969

@@ -1459,7 +1482,31 @@ describe('select', function() {
14591482

14601483
expect(element.find('option')[0].selected).toEqual(false);
14611484
});
1462-
});
1485+
1486+
it('should bind to scope with multiple:true while tracking/identifying objects with multiple',
1487+
function() {
1488+
createSelect({
1489+
'ng-model': 'selected',
1490+
'multiple': true,
1491+
'ng-options': 'item.id as item.name for item in values track by item.trackId'
1492+
});
1493+
1494+
var selectedIdx;
1495+
1496+
scope.$apply(function() {
1497+
scope.values = [{id: 10, trackId: 100, name: 'A'}, {id: 20, trackId: 200, name: 'B'}];
1498+
scope.selected = [10];
1499+
});
1500+
1501+
expect(element.val()).toEqual(['0']);
1502+
1503+
scope.$apply(function() {
1504+
scope.selected = [20];
1505+
});
1506+
1507+
expect(element.val()).toEqual(['1']);
1508+
});
1509+
});
14631510

14641511

14651512
describe('ngRequired', function() {

0 commit comments

Comments
 (0)