@@ -35,6 +35,12 @@ var ngOptionsMinErr = minErr('ngOptions');
35
35
* be bound to string values at present.
36
36
* </div>
37
37
*
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
+ *
38
44
* @param {string } ngModel Assignable angular expression to data-bind to.
39
45
* @param {string= } name Property name of the form under which the control is published.
40
46
* @param {string= } required The control is considered valid only if value is entered.
@@ -69,7 +75,9 @@ var ngOptionsMinErr = minErr('ngOptions');
69
75
* DOM element.
70
76
* * `trackexpr`: Used when working with an array of objects. The result of this expression will be
71
77
* 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`.
73
81
*
74
82
* @example
75
83
<example module="selectExample">
@@ -326,6 +334,8 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
326
334
327
335
var displayFn = $parse ( match [ 2 ] || match [ 1 ] ) ,
328
336
valueName = match [ 4 ] || match [ 6 ] ,
337
+ selectAs = / a s / . test ( match [ 0 ] ) && match [ 1 ] ,
338
+ selectAsFn = selectAs ? $parse ( selectAs ) : null ,
329
339
keyName = match [ 5 ] ,
330
340
groupByFn = $parse ( match [ 3 ] || '' ) ,
331
341
valueFn = $parse ( match [ 2 ] ? match [ 1 ] : valueName ) ,
@@ -336,7 +346,27 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
336
346
// We try to reuse these if possible
337
347
// - optionGroupsCache[0] is the options with no option group
338
348
// - 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
+ } ;
340
370
341
371
if ( nullOption ) {
342
372
// compile the element since there might be bindings in it
@@ -437,7 +467,7 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
437
467
var selectedSet = false ;
438
468
if ( multiple ) {
439
469
var modelValue = ctrl . $modelValue ;
440
- if ( trackFn && isArray ( modelValue ) ) {
470
+ if ( trackFn && isArray ( modelValue ) && ! selectAs ) {
441
471
selectedSet = new HashMap ( [ ] ) ;
442
472
var locals = { } ;
443
473
for ( var trackIndex = 0 ; trackIndex < modelValue . length ; trackIndex ++ ) {
@@ -501,27 +531,16 @@ var selectDirective = ['$compile', '$parse', function($compile, $parse) {
501
531
optionGroup = optionGroups [ optionGroupName ] = [ ] ;
502
532
optionGroupNames . push ( optionGroupName ) ;
503
533
}
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 ;
518
537
label = displayFn ( scope , locals ) ; // what will be seen by the user
519
538
520
539
// doing displayFn(scope, locals) || '' overwrites zero values
521
540
label = isDefined ( label ) ? label : '' ;
522
541
optionGroup . push ( {
523
542
// 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 ) ,
525
544
label : label ,
526
545
selected : selected // determine if we should be selected
527
546
} ) ;
0 commit comments