zl程序教程

您现在的位置是:首页 >  前端

当前栏目

细说angular Form addControl方法

Angular方法 Form 细说
2023-09-14 09:00:58 时间

在本篇博文中,我们将接触angular的验证。angular的验证是由form 指令和ngModel协调完成的。今天博主在这里想要说的是在验证在的一种特殊情况,当验证控件没有没有name属性这是不会被form捕获的。或者是你希望在ngRepeat中使用动态表达式。

下面且让我们先来从angular源码中看起如下:

首先是ngModel:


从上面我们能够看出ngModel指令会在编译时期的post link阶段会通过form的 addControl方法把自己的controller注册到父节点上的form的formController中。

在看看ngModel controller初始化代码:

复制代码
 var NgModelController = [$scope, $exceptionHandler, $attrs, $element, $parse,

 function($scope, $exceptionHandler, $attr, $element, $parse) {

 ....

 this.$pristine = true;

 this.$dirty = false;

 this.$valid = true;

 this.$invalid = false;

 this.$name = $attr.name;
复制代码

我们从上面我们可以看到 ngModel的$name属性并不支持表达式计算。

而FormController的addControl代码则是:

复制代码
 /**

 * @ngdoc function

 * @name ng.directive:form.FormController#$addControl

 * @methodOf ng.directive:form.FormController

 * @description

 * Register a control with the form.

 * Input elements using ngModelController do this automatically when they are linked.

 form.$addControl = function(control) {

 // Breaking change - before, inputs whose name was "hasOwnProperty" were quietly ignored

 // and not added to the scope. Now we throw an error.

 assertNotHasOwnProperty(control.$name, input);

 controls.push(control);

 if (control.$name) {

 form[control.$name] = control;

 };
复制代码

从上面我们可以清楚的看见虽然ngModel注册了自己,但是这里也不一定能注册成功,ngModel心必须有自己的$name才能被注册成功。

从上面的代码中可以得出,当我们的验证失效的时候,我们可以有一个万能的方式就是 手动注册到form controller

手动注册form controller

为了我写了一个dy-name的插件,其会在post link阶段解析表达式,并把自己注册到父form controller。

如下:

复制代码
.directive("dyName", [

 function() {

 return {

 require: "ngModel",

 link: function(scope, elm, iAttrs, ngModelCtr) {

 ngModelCtr.$name = scope.$eval(iAttrs.dyName)

 var formController = elm.controller(form) || {

 $addControl: angular.noop

 formController.$addControl(ngModelCtr);

 scope.$on($destroy, function() {

 formController.$removeControl(ngModelCtr);

 ])
复制代码
 使用方式:
复制代码
 div ng-repeat="item in demo.fields" 

 div 

 label : /label 

 div 

 input type="number" dy-name="item.field" ng-model="demo.data[item.field]" min="10" max="500" ng-required="true"/ 

 /div 

 /div 

 /div 
复制代码


其实实现为在post link阶段获取到form controller,并把自己注册到form controller,而且为了消除对象的级联,将会在scope摧毁阶段remove自己。

其效果请看jsbin $addControl

注意:在formController.$addControl方法的参数传入的不是界面控件,而是ngModelController.或者名字为$addController更合适。