Angularjs - 在控制器范围内使用orderby过滤器

时间:2013-12-17 10:45:37

标签: angularjs pagination angular-ui watch angularjs-orderby

我有一个对象数组,即已过滤和分页,现在我想按不同的对象属性对列表项进行排序。

我尝试了orderBy过滤器,如下所示:

<th><a href='' ng-click="reverse = sortParam == 'title' && !reverse; sortParam = 'title'">Title</a></th>

<tr ng-repeat="item in pagedItems|filter:filterParam|orderBy:sortParam:reverse">
    <td>{{ item.title }}</td>
</tr>

这似乎工作正常,点击Title链接,按字母顺序排序或按字母顺序排序,具体取决于当前状态。

但问题是,只有pagedItems被排序,这是有道理的,因为我们正在将orderBy过滤器应用于pagedItems。我想要实现的是在应用过滤器时订购整个项目集(不仅仅是当前被分页的项目)。

为了达到这个目的,我想我会在控制器范围内使用一种方法。所以我把上面改为:

/** In the Template */

<th><a href='' ng-click="sortItems('title')">Title</a></th>

<tr ng-repeat="item in pagedItems|filter:filterParam">
    <td>{{ item.title }}</td>
</tr>


/** In the Controller */

$scope.sortItems = function(value) {
    $scope.filtered = $filter('orderBy')($scope.filtered, value);
};

$scope.$watch('currentPage + numPerPage + filtered', function() {
    $scope.pagedItems = getPagedItems($scope, 'filtered');
});

sortItems方法可以工作并更改顺序,但视图中的项目不会更新,因为$watch代码未被触发。我假设它可能没有被更改,因为$scope.filtered中的数据没有被更改,只是索引被更改。所以我在数组的末尾添加了空元素:

$scope.sortItems = function(value) {
    $scope.filtered = $filter('orderBy')($scope.filtered, value);
    $scope.filtered.push({});
};

现在,Everything按预期工作但我不能在数组中保留一个空对象,因为它会影响显示的项目,计数和数据。所以我想我会添加和删除一个空项目。所以将上面改为:

$scope.sortItems = function(value) {
    $scope.filtered = $filter('orderBy')($scope.filtered, value);
    $scope.filtered.push({});
    $scope.filtered.pop();
};

但是猜猜$watch代码没有再被解雇。

问题

我的问题是$watch根据它的长度查找数组中的更改吗?如果是,那么实现我想要的最好的方法是什么。任何帮助将不胜感激。

1 个答案:

答案 0 :(得分:5)

好的,我使用$broadcast$on解决了这个问题,如下所示:

$scope.sortList = function(value) {

    if (value === $scope.currentFilter) {
        value = value.indexOf('-') === 0 ? value.replace("-","") : "-" + value;
    }   

    $scope.currentFilter = value;
    $scope.filtered = $filter('orderBy')($scope.filtered, value);
    $scope.$broadcast('sorted');
}

$scope.$on('sorted', function() {
    $scope.pagedCandidates = getPagedItems($scope, 'filtered');
})