AngularJS自定义验证指令 - 如何避免使用隔离范围

时间:2017-12-27 06:58:44

标签: angularjs validation angularjs-directive angularjs-ng-model

我尝试在文本字段上使用多个角度验证器,但遇到了Multiple directives requesting isolated scope错误。 (请在结束前阅读作为副本。)

到目前为止我见过的所有解决方案,建议从违规指令中删除scope: {...},但是对于我的场景,我需要从控制器中评估变量(并$watch变化)。

我已尝试使用attrs.$observe,但我无法弄清楚如何将评估变量纳入$validator函数。 (另外,我不能$observe ngModel)。

请告诉我是否有另一种解决此问题的方法。

这是我可以放在一起的最小例子。 N.B。 maxLength验证者的范围已被注释掉,基本上禁用它:



angular

.module('app', [])

// validates the min length of a string...
.directive("minLen", function() {
    return {
      require: 'ngModel',
      restrict: 'A',
      scope: {
        ngModel: '=',
        minLen: '='
      },
      link: function(scope, element, attrs, ngModelCtrl) {
        scope.$watch('ngModel', function(){
          ngModelCtrl.$validate();
        });
        scope.$watch('minLen', function(){
          ngModelCtrl.$validate();
        });
        ngModelCtrl.$validators.minLength = function(modelValue, viewValue) {
          var value = modelValue || viewValue;
          return angular.isUndefined(scope.minLen) ||
            angular.isUndefined(value) ||
            value.length >= scope.minLen;
        };
      }
	};
})

.directive("maxLen", function() {
    return {
      require: 'ngModel',
      restrict: 'A',
// Commented out for now - causes error.      
//       scope: {
//         ngModel: '=',
//         maxLen: "="
//       },
      link: function(scope, element, attrs, ngModelCtrl) {
        scope.$watch('ngModel', function(){
          ngModelCtrl.$validate();
        });
        scope.$watch('maxLen', function(){
          ngModelCtrl.$validate();
        });
        ngModelCtrl.$validators.maxLength = function(modelValue, viewValue) {
          var value = modelValue || viewValue;
          return angular.isUndefined(scope.maxLen) ||
            angular.isUndefined(value) ||
            value.length >= scope.maxLen;
        };
      }
	};
})

// this controller just initialises variables...
.controller('CustomController', function() {
  var vm = this;
  vm.toggleText = function(){
    if (vm.text === 'aaa') {
      vm.text = 'bbbbb';
    } else {
      vm.text = 'aaa';
    }
  }
  vm.toggle = function(){
    if (vm.minLen === 3) {
      vm.minLen = 4;
      vm.maxLen = 12;
    } else {
      vm.minLen = 3;
      vm.maxLen = 10;
    }
  };
  vm.toggleText();
  vm.toggle();
  return vm;
})
;

<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.4.0/angular.min.js"></script>
<div ng-app="app" ng-controller="CustomController as ctrl">
  <ng-form name="ctrl.form">
    <label>Enter {{ctrl.minLen}}-{{ctrl.maxLen}} characters: </label>
    <input 
           name="text" 
           type="text" 
           ng-model="ctrl.text" 
           min-len="ctrl.minLen"
           max-len="ctrl.maxLen"
           />
  </ng-form>
  <br/><br/>
  <button ng-click="ctrl.toggle()">Modify validation lengths</button>
  <button ng-click="ctrl.toggleText()">Modify ngModel (text)</button>
  
  <h3>Validation (just list $error for now)</h3>
  <pre>{{ ctrl.form.text.$error | json }}</pre>
</div>
&#13;
&#13;
&#13;

2 个答案:

答案 0 :(得分:0)

删除隔离范围并使用scope.$watch评估属性:

app.directive("minLen", function() {
    return {
      require: 'ngModel',
      restrict: 'A',
      scope: false,
   //     ngModel: '=',
   //     minLen: '='
   //   },
      link: function(scope, element, attrs, ngModelCtrl) {
        var minLen = 0;
        scope.$watch(attrs.minLen, function(value){
          minLen = toInt(value) || 0;
          ngModelCtrl.$validate();
        });
        ngModelCtrl.$validators.minLength = function(modelValue, viewValue) {
          return ngModelCtrl.$isEmpty(viewValue) || viewValue.length >= minLen;
        };
      }
    };
})

无需观看ngModel属性,因为ngModelController会在模型更改时自动调用$validators集合中的函数。

当watch表达式是字符串时,string属性将被评估为Angular Expression

有关详细信息,请参阅AngularJS scope API Reference - $watch

答案 1 :(得分:0)

  1. 删除隔离范围
  2. 你不需要观察,观察ngModel - 当它发生变化时,angular会为你运行验证器。
  3. 决定如何使用您的指令:my-val-dir =“{{valName}}”vs my-val-dir =“valName”。在第一种情况下,你使用attrs。$ observe('myValDir'),在第二个$ watch(attrs.myValDir)&amp; $ EVAL。 当你的价值变得简单时,比如数字或短字符串 - 第一种方式似乎很好,当价值很大时,即数组 - 使用第二种方法。