如何有条件地对对象属性的 ng-repeat 应用 Angular 过滤器?

How to conditionally apply Angular filters on ng-repeat of object attributes?

我有一个具有 100 多个属性的对象,例如 "name"、"price"、"expiry date"...等 我正在使用 ng-repeat 遍历对象的所有键对值并将它们显示在 table.

<table class="table">
    <tr ng-repeat="x in attr_array">
        <td><b>{{x.key}}</b></td>
        <td>{{x.value}}</td>
    </tr>
</table>

但我想对某些属性使用 Angular 日期过滤器,例如任何日期字段:

{{ x.value | date: 'MMM d, y'}}

最好还有其他过滤器。我该怎么做呢?

这不是简单而优雅的方法。你不能有动态过滤器,所以你的选择是优先顺序:

  • 在控制器中动态预格式化日期字段
  • 使用 ngIf/ngShow 个开关
  • 编写自定义过滤器,根据某些配置应用另一个过滤器。

这是自定义过滤器方法的示例:

angular.module('demo', [])

.controller('MainCtrl', function($scope) {
  $scope.attr_array = [
    {key: 'created', value: new Date, filter: ['date', 'MMM d, y']},
    {key: 'name', value: 'Thomas Mann', filter: 'uppercase'},
    {key: 'price', value: 1233.45, filter: 'number'},
    {key: 'description', value: 'Some long string', filter: ['limitTo', 10] }
  ]
})

.filter('transform', function($filter) {
  return function(value, filter) {
    var filterData = [].concat(filter);
    filterData.splice(1, 0, value);
    return $filter(filterData.shift()).apply(this, filterData);
  }
})
<script src="https://code.angularjs.org/1.4.9/angular.js"></script>

<table ng-app="demo" ng-controller="MainCtrl" class="table">
  <tr ng-repeat="x in attr_array">
    <td><b>{{ x.key }}</b>
    </td>
    <td>{{ x.value | transform:x.filter }}</td>
  </tr>
</table>

我尝试重现您的问题并用 ng-if 解决了它。 angular 命名空间中似乎有一个函数来检查我通过范围注入到视图中的每种类型,如日期、字符串、数字。
另请注意,我使用 ng-repeat="(key, value) in ..." 假设您正在迭代一个对象 source.

<!DOCTYPE html>
<html>

<head>
  <script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.4.0/angular.min.js"></script>
  <meta charset="utf-8">
  <meta name="viewport" content="width=device-width">
  <title>JS Bin</title>
</head>

<body ng-app="app" ng-controller="RootController">
  <table>
    <tr ng-repeat="(key, value) in attr_array">
      <td><b>{{key}}</b>
      </td>
      <td>
        <span ng-if="isDate(value)">{{value | date: 'MMM d, y'}}</span>
        <span ng-if="isNumber(value)">{{value | number: 4}}</span>
        <span ng-if="isString(value)">{{value | uppercase}}</span>
      </td>
    </tr>
  </table>
  <script>
    angular.module('app', [])
      .controller('RootController', function($scope) {
        $scope.isDate = angular.isDate;
        $scope.isNumber = angular.isNumber;
        $scope.isString = angular.isString;
        $scope.attr_array = {
          date: new Date(),
          str: "hello",
          nm: 50.2
        };
      });
  </script>
</body>

</html>