30

我试图让 Angular 根据我的数据生成一个 CSS 滑块。我知道数据在那里并且能够为按钮生成它,但是由于某种原因,代码不会填充 ng-switch-when。当我检查代码时,我看到了两次(我知道这是正确的,因为我只有两个项目):

<div ng-repeat="assignment in assignments" ng-animate="'animate'" class="ng-scope">
    <!-- ngSwitchWhen: {{assignment.id}} -->
</div>

我的实际代码:

<div ng-init="thisAssignment='one'">
     <div class="btn-group assignments" style="display: block; text-align: center; margin-bottom: 10px">
         <span ng-repeat="assignment in assignments">
              <button ng-click="thisAssignment = '{{assignment.id}}'" class="btn btn-primary">{{assignment.num}}</button>
         </span>
     </div>

     <div class="well" style="height: 170px;">
         <div ng-switch="thisAssignment">
              <div class="assignments">
                   <div ng-repeat="assignment in assignments" ng-animate="'animate'">
                        <div ng-switch-when='{{assignment.id}}' class="my-switch-animation">
                        <h2>{{assignment.name}}</h2>
                        <p>{{assignment.text}}</p>
                   </div>
              </div>
         </div>  
    </div>  
</div>

编辑:这就是我试图模仿的,虽然是动态数据。http://plnkr.co/edit/WUCyCN68tDR1YzNnCWyS?p=preview

4

2 回答 2

49

来自文档——</p>

请注意,要匹配的属性值不能是表达式。它们被解释为要匹配的文字字符串值。例如,ng-switch-when="someVal" 将匹配字符串“someVal”而不是表达式 $scope.someVal 的值。

因此,换句话说,ng-switch 用于模板中的硬编码条件。

你会像这样使用它:

<div class="assignments">
  <div ng-repeat="assignment in assignments" ng-animate="'animate'">
    <div ng-switch="assignment.id">
      <div ng-switch-when='1' class="my-switch-animation">
      <h2>{{assignment.name}}</h2>
      <p>{{assignment.text}}</p>
    </div>
  </div>
</div>

现在这可能不完全适合您的用例,因此您可能必须重新考虑您的策略。

Ng-If 可能是您所需要的——另外,您需要注意“隔离”范围。基本上,当您使用某些指令(如 ng-repeat)时,您会创建与其父级隔离的新范围。因此,如果您thisAssignment在中继器内部进行更改,您实际上是在更改特定重复块内的变量,而不是整个控制器。

这是你要做什么的演示。

请注意,我将 selected 属性分配给things数组(它只是一个对象)。


2014 年 12 月 12 日更新:添加新代码块以阐明ng-switch. 上面的代码示例应该考虑不做什么。

正如我在评论中提到的。Switch 应该被认为与 JavaScript 开关完全一样。它用于硬编码的切换逻辑。因此,例如在我的示例帖子中,只有几种类型的帖子。您应该提前知道要打开的值的类型。

<div ng-repeat="post in posts">
  <div ng-switch on="post.type">

    <!-- post.type === 'image' -->
    <div ng-switch-when="image" class="post post-image">
      <img ng-src="{{ post.image }} />
      <div ng-bind="post.content"></div>
    </div>

    <!-- post.type === 'video' -->
    <div ng-switch-when="video" class="post post-video">
      <video ng-src="{{ post.video }} />
      <div ng-bind="post.content"></div>
    </div>
    
    <!-- when above doesn't match -->
    <div ng-switch-default class="post">
      <div ng-bind="post.content"></div>
    </div>
  </div>
</div>

您可以使用 实现相同的功能ng-if,您的工作是决定在您的应用程序中什么是有意义的。在这种情况下,后者更简洁,但也更复杂,如果模板更复杂,你会看到它变得更加复杂。基本的区别是ng-switch声明ng-if式的,命令式的。

<div ng-repeat="post in posts">
  <div class="post" ng-class="{
      'post-image': post.type === 'image', 
      'post-video': post.type === 'video'">
    <video ng-if="post.type === 'video'" ng-src="post.video" />
    <img ng-if="post.type === 'image'" ng-src="post.image" />
    <div ng-bind="post.content" />
  </div>
</div>
于 2014-02-05T00:26:23.557 回答
12

乔恩绝对是正确的。Angular 不支持动态ngSwitchWhen值。但我想要它。我发现使用我自己的指令代替ngSwitchWhen. 它不仅支持动态值,而且还支持每个语句的多个值(类似于 JS 切换失败)。

需要注意的是,它只在编译时评估一次表达式,因此您必须立即返回正确的值。出于我的目的,这很好,因为我想使用应用程序中其他地方定义的常量。可能可以对其进行修改以动态重新评估表达式,但这需要使用ngSwitch.

我使用的是 angular 1.3.15,但我使用 angular 1.4.7 进行了快速测试,它在那里也运行良好。

Plunker 演示

编码

module.directive('jjSwitchWhen', function() {
    // Exact same definition as ngSwitchWhen except for the link fn
    return {
        // Same as ngSwitchWhen
        priority: 1200,
        transclude: 'element',
        require: '^ngSwitch',
        link: function(scope, element, attrs, ctrl, $transclude) {
            var caseStms = scope.$eval(attrs.jjSwitchWhen);
            caseStms = angular.isArray(caseStms) ? caseStms : [caseStms];

            angular.forEach(caseStms, function(caseStm) {
                caseStm = '!' + caseStm;
                ctrl.cases[caseStm] = ctrl.cases[caseStm] || [];
                ctrl.cases[caseStm].push({ transclude: $transclude, element: element });
            });
        }
    };
});

用法

控制器
  $scope.types = {
      audio: '.mp3', 
      video: ['.mp4', '.gif'],
      image: ['.jpg', '.png', '.gif'] // Can have multiple matching cases (.gif)
  };
模板
  <div ng-switch="mediaType">
    <div jj-switch-when="types.audio">Audio</div>

    <div jj-switch-when="types.video">Video</div>

    <div jj-switch-when="types.image">Image</div>

    <!-- Even works with ngSwitchWhen -->
    <div ng-switch-when=".docx">Document</div>

    <div ng-switch-default>Invalid Type</div>
  <div>
于 2015-10-28T18:37:57.093 回答