关于如何在 ngRepeat 指令中实现项目删除有很多问题,正如我发现的那样,它归结为使用ngClick并触发一些传递它项目的$index的删除函数。
但是,我在任何地方都找不到我有多个 ngRepeats 的示例:
<div ng-controller="MyController">
<div ng-repeat="email in user.emails">
{{ email }} <a href>Remove</a>
</div>
<div ng-repeat="phone in user.phones">
{{ phone }} <a href>Remove</a>
</div>
</div>
为此,我需要创建$scope.removePhone和$scope.removeEmail将使用ngClick on Remove anchor调用。但我正在寻找更通用的解决方案。特别是因为我有很多页面有很多 ngRepeats。
我正在考虑编写一个指令,该指令将放置在Remove anchor上,并会执行以下操作:
- 在父元素中查找ngRepeat 。
- 阅读它正在迭代的内容(第一种情况是'user.emails',第二种情况是'user.phones')
- 从那个模型中移除$index元素。
所以标记看起来像这样:
<div ng-controller="MyController">
<div ng-repeat="email in user.emails">
{{ email }} <a href remove-directive="$index">Remove</a>
</div>
<div ng-repeat="phone in user.phones">
{{ phone }} <a href remove-directive="$index">Remove</a>
</div>
</div>
我正在寻找可能实现的目标吗?最好的方法是什么?
当前的hacky解决方案
这是我目前的做法。它既丑陋又丑陋,但在我找到更漂亮的方法之前可以完成工作。
myAppModule.controller('MyController', function ($scope, $parse, $routeParams, User) {
$scope.user = User.get({id: $routeParams.id});
$scope.remove = function ($index, $event) {
// TODO: Find a way to make a directive that does this. This is ugly. And probably very wrong.
var repeatExpr = $($event.currentTarget).closest('[ng-repeat]').attr('ng-repeat');
var modelPath = $parse(repeatExpr.split('in')[1].replace(/^\s+|\s+$/g, ''));
$scope.$eval(modelPath).splice($index, 1);
};
});
在 DOM 中:
<div ng-repeat="email in user.email" class="control-group">
<label class="control-label">
{{ "Email Address"|_trans }}
</label>
<div class="controls">
<input type="text" ng-model="email.address">
<span class="help-inline"><a href ng-click="remove($index, $event)">{{ "Delete"|_trans }}</a></span>
</div>
</div>