Using $index with the AngularJS 'ng-options

2019-01-10 10:49发布

Say that I bind an array to a select tag using the following:

<select ng-model="selData" ng-options="$index as d.name for d in data">

In this case, the associated option tags are assigned a sequence of index values: (0, 1, 2, ...). However, when I select something from the drop-down, the value of selData is getting bound to undefined. Should the binding actually work?

On the other hand, say that I instead do the following:

<select ng-model="selData" ng-options="d as d.name for d in data">

Here, the option tags get the same index, but the entire object is bound on change. Is it working this way by design, or this behavior simply a nice bug or side-effect of AngularJS?

5条回答
萌系小妹纸
2楼-- · 2019-01-10 11:10

Don't use $index inside select tags. Use $index inside the option tags if you want to use the array indexes as either values or options.

<option ng-repeat="user in users" value="{{user.username}}">{{$index+1}}</option>

If you want to use inside values just put it in the value attribute as binding expression like

<option ng-repeat="user in users" value="{{$index+1}}">{{user.username}}</option>

and my controller code be like:

var users = ['username':'bairavan', 'username':'testuser'];
查看更多
我想做一个坏孩纸
3楼-- · 2019-01-10 11:14

Since you can't use $index but you can try indexOf.

HTML

<div ng-app ng-controller="MyCtrl">
    <select 
          ng-model="selectedItem"
          ng-options="values.indexOf(selectedItem) as selectedItem for selectedItem in values"></select>
    selectedItem: {{selectedItem}}
</div>

Controller

function MyCtrl($scope) {
    $scope.values = ["Value1","Value2"];
    $scope.selectedItem = 0;
}

Demo Fiddle

Comment:

Array.prototype.indexOf is not supported in IE7 (8)

查看更多
在下西门庆
4楼-- · 2019-01-10 11:16

Since arrays are very similar to objects in JavaScript, you can use the syntax for "object data sources". The trick is in the brackets in the ng-options part:

var choices = [
  'One',
  'Two',
  'Three'
];

In the template:

<select
  ng-model="model.choice"
  ng-options="idx as choice for (idx, choice) in choices">
</select>

In the end, model.choice will have the value 0, 1, or 2. When it's 0, you will see One; 1 will display Two, etc. But in the model, you will see the index value only.

I adapted this information from "Mastering Web Application Development with AngularJS" by PACKT Publishing, and verified at the Angular reference documentation for select.

查看更多
Deceive 欺骗
5楼-- · 2019-01-10 11:19

You can also use ng-value='$index' in <option> tag.

<select ng-model="selData">
 <option ng-repeat="d in data track by $index" ng-value="$index">
   {{d.name}}
 </option>
</select>
查看更多
Melony?
6楼-- · 2019-01-10 11:22

$index is defined for ng-repeat, not select. I think this explains the undefined. (So, no, this shouldn't work.)

Angular supports binding on the entire object. The documentation could be worded better to indicate this, but it does hint at it: "ngOptions ... should be used instead of ngRepeat when you want the select model to be bound to a non-string value."

查看更多
登录 后发表回答