angularjs - 为什么 ngIf 的优先级高于 {{ }} (插值)?

标签 angularjs

ngIf 优先级 (600) 设置为高于 {{ }}(100) 的原因是什么?难道它不应该有较低的优先级来允许ng-if属性值中的{{ }}吗?

我想在 $scope 变量中有一个条件:

Controller :

app.controller('MainCtrl', function($scope, $http, $parse) {
  $scope.hide = "check === 'hidden'";
  $scope.parsecond = function (cond) {
    return $parse(cond)($scope);
  };

});

模板:

  <body ng-controller="MainCtrl">
    <div ng-if="!{{hide}}">funky ng-if div</div>
    <div ng-hide="{{hide}}">ng-hide div</div>
    <div ng-if="!parsecond(hide)">ng-if div</div>
    <input type="input" ng-model="check" />
  </body>

ng-hide 工作正常,因为它解析 hide 变量的内容并返回“check === 'hidden'”,然后由 ng-hide 对其进行评估指令。

但是 ng-if 尝试在 interpolate 有机会解析字符串之前评估 {{hide}},因此 ng-if抛出异常。

我发现的唯一解决方案是调用一个函数,该函数基本上完成插值指令的工作并返回解析的内容。

Plnkr 显示问题:link

编辑:

阅读文档后,我找到了更好的方法,无需在 $scope 上使用自定义方法,因为 angularjs 已经有一个方法可以根据当前的 $ 解析变量范围 ($eval)。

所以我的解决方案是:

  <body ng-controller="MainCtrl">
    <div ng-if="!$eval(hide)">funky ng-if div</div>
    <div ng-hide="{{hide}}">ng-hide div</div>
    <div ng-if="!parsecond(hide)">ng-if div</div>
    <input type="input" ng-model="check" />
  </body>

更新了 plnkr:link

尽管这仍然不能解释为什么 ngIf 具有更高的优先级。

编辑2:

只是为了让人们明白这是不一样的:

例如:

Controller :

$scope.value = "hi";
$scope.condition = "value === 'bye'";

HTML:

<div ng-hide="condition"></div> <!--This will be evaluated to true since !!"value ==='bye'" = true. -->
<div ng-hide="{{condition}}"></div> <!--This will be evaluated to false since value !== 'bye' = false -->
<div ng-if="condition"></div> <!--This will be evaluated to true since !!"value === 'bye'" = true. -->
<div ng-if="{{condition}}"></div> <!--This will give an exception since ngIf directive ran before interpolation directive in the $compile step. -->
<div ng-if="$eval(condition)"></div> <!--This will be evaluated to false since value !== 'bye' = false. -->

我的结论是,如果您希望指令在字符串中而不是在作用域的属性中评估/设置监视,则使用 $parse 更安全。尽管我确实可以将 {{ }} 用于 ng-hide/ng-show 或任何优先级低于 100 的指令但我猜它不安全,因为我依赖于编译顺序,并且不能 100% 清楚它在未来的补丁中不会改变。

最佳答案

ng-if期望其值是一个 Angular 表达式 - under the hood它只是利用 $scope.$watch 。因此,如果您想条件包括 ng-if 的内容在范围内定义的某个变量上(比方说: scope.hide ),您可以输入 ng-if="hide"在你的标记中。这里没有双花括号。

现在回到你的问题:确实是 ng-if优先级为600 ,但是 $interpolate 是 Angular 的服务 - 不是指令。因此$interpolate没有定义优先级。你从哪里得到的100来自?

更新

您始终可以条件包括 ng-if 的内容在某些功能上(比如说 scope.conditionFn ),输入您的标记: ng-if="conditionFn()" .

更新2

我更新了您的PLNKR使其发挥作用。 ng-if之间的不一致和ng-hide在你的plunker中与$compile中发生的插值优先级无关.

更新3

看来你是对的,插值顺序在这里发挥了作用,但是......我真的没有看到任何在 Angular 表达式内部插值的好理由。原因ng-if具有相对较高优先级的是它从 DOM 中删除/添加嵌入的内容,而 ng-hide仅显示/隐藏嵌入的内容。我认为一个指令似乎起作用而另一个指令不起作用,这纯粹是巧合。 但是如果你不使用不必要的、多余的技巧,两者都会按预期工作,正如我的笨蛋所展示的那样。

关于angularjs - 为什么 ngIf 的优先级高于 {{ }} (插值)?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/23265948/

相关文章:

javascript - Jasmine ,AngularJS : Unit test a function which returns a value after timeout

angularjs - ag-Grid:尝试调用 sizeColumnsToFit() 但网格返回时宽度为零,也许网格在屏幕上还不可见?

javascript - Angular Controller 中的事件仅触发一次

javascript - 使用 Angular 使 <i> 标签可点击

javascript - Angularjs uib modal无法调用模态 Controller

javascript - Angularjs - 如何在确认对话框为 false 时取消下拉菜单中的更改事件?

javascript - AngularJS - 需要 $routeChangeStart 和 $locationChangeStart 的某种组合

javascript - AngularJS ngmap 将范围变量传递给 infoWindow

javascript - ng-class 的多个参数

javascript - 在模糊中屏蔽信用卡输入