31

我需要一种方法来遍历 AngularJS 表单的注册控件。本质上,我正在尝试获取所有 $dirty 控件,但没有控件数组(FormController 除了包含控件本身之外,还有许多不同的属性/功能 - 每个都是它自己的对象)。

我一直在查看源代码,发现controlsFormController 中有一个数组,正是我要查找的数组。有没有办法访问这个值,或者扩展 FormController 以包含一个返回这个controls数组的函数?

编辑:Plnkr演示

此外,我意识到从技术上讲,我可以检查密钥字符串中的第一个字符是否为“$”,但我想避免这种情况,以防 FormController/directive 在未来版本的 Angular 中发生变化。

编辑2:另一个澄清:我的目标是能够确定哪些特定字段是$dirty,是否通过循环遍历整个控件列表(不包括$dirty、$invalid、$error、$name、和其他存在于 Form 对象中的属性)或通过扩展 FormController 并创建一个函数,该函数仅返回当前脏的控件(并且不等于它们的起始值)

编辑3:我正在寻找的解决方案需要适用于不同结构的表单/模型。示波器上的模型是通过 AJAX 生成的,因此它们的结构已经设置好(我想避免为我已经通过 AJAX 接收的所有数据硬编码一个新结构)。此外,我希望在多个表单/模型中使用此表单提交过程,它们中的每一个都有不同的 JSON 结构——因为它们适用于我们对象模型中的不同实体。这就是为什么我选择寻求一种方法来访问controlsFormController 中的对象(我将从FormController下面发布代码),因为它是唯一可以获得所有字段的平面数组的地方。

function FormController(element, attrs) {


var form = this,
      parentForm = element.parent().controller('form') || nullFormCtrl,
      invalidCount = 0, // used to easily determine if we are valid
      errors = form.$error = {},
      controls = [];

  // init state
  form.$name = attrs.name || attrs.ngForm;
  form.$dirty = false;
  form.$pristine = true;
  form.$valid = true;
  form.$invalid = false;

  parentForm.$addControl(form);

  // Setup initial state of the control
  element.addClass(PRISTINE_CLASS);
  toggleValidCss(true);

  // convenience method for easy toggling of classes
  function toggleValidCss(isValid, validationErrorKey) {
    validationErrorKey = validationErrorKey ? '-' + snake_case(validationErrorKey, '-') : '';
    element.
      removeClass((isValid ? INVALID_CLASS : VALID_CLASS) + validationErrorKey).
      addClass((isValid ? VALID_CLASS : INVALID_CLASS) + validationErrorKey);
  }

  /**
   * @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) {
    controls.push(control);

    if (control.$name && !form.hasOwnProperty(control.$name)) {
      form[control.$name] = control;
    }
  };

  /**
   * @ngdoc function
   * @name ng.directive:form.FormController#$removeControl
   * @methodOf ng.directive:form.FormController
   *
   * @description
   * Deregister a control from the form.
   *
   * Input elements using ngModelController do this automatically when they are destroyed.
   */
  form.$removeControl = function(control) {
    if (control.$name && form[control.$name] === control) {
      delete form[control.$name];
    }
    forEach(errors, function(queue, validationToken) {
      form.$setValidity(validationToken, true, control);
    });

    arrayRemove(controls, control);
  };

  // Removed extra code
}

如您所见,表单本身具有controls私有可用的数组。我想知道是否有办法让我扩展,FormController以便我可以公开该对象?或者创建一个公共函数,以便我至少可以查看私有数组?

4

4 回答 4

20

要直接解决该问题,请像这样修改@lombardo 的答案;

     var dirtyFormControls = [];
     var myForm = $scope.myForm;
     angular.forEach(myForm, function(value, key) {
         if (typeof value === 'object' && value.hasOwnProperty('$modelValue') && value.$dirty)
             dirtyFormControls.push(value)                        
     });

数组“dirtyFormControls”将包含脏的表单控件。

您还可以使用此技巧在“必需”验证和所有其他验证的表单提交时显示错误消息。在您的 submit() 函数中,您将执行以下操作;

 if (form.$invalid) {
     form.$setDirty();              
     angular.forEach(form, function(value, key) {
         if (typeof value === 'object' && value.hasOwnProperty('$modelValue'))
             value.$setDirty();                        
     });
    //show user error summary at top of form.
     $('html, body').animate({
         scrollTop: $("#myForm").offset().top
     }, 1000);
     return;
 }

在您的表单中,您将显示错误消息

    <span ng-messages="myForm['subject-' + $index].$error" ng-show="myForm['subject-' + $index].$dirty" class="has-error">
        <span ng-message="required">Course subject is required.</span>
    </span>

当您使用“ng-repeat”或类似的东西动态生成控件时,上述解决方案很有用。

于 2015-12-06T18:44:40.663 回答
11

您可以使用以下代码来迭代控件:

    var data = {};
    angular.forEach(myForm, function (value, key) {
        if (value.hasOwnProperty('$modelValue'))
            data[key] = value.$modelValue;
    });
于 2014-04-20T05:28:56.440 回答
2

从您的控制器中简单地尝试:

$scope.checkForm = function(myFormName){
     console.log(myFormName.$invalid);
}

更新:

<div ng-controller="MyController">
                <form name="form" class="css-form" novalidate>
                    <input type="text" ng-model="user.uname" name="uname" required /><br />
                    <input type="text" ng-model="user.usurname" name="usurname" required /><br />
                    <button ng-click="update(form)">SAVE</button>
                </form>
              </div>

app.controller('MyController',function($scope){
                $scope.user = {};
                $scope.update = function (form){
                    var editedFields = [];
                    angular.forEach($scope.user, function(value, key){
                        if(form[key].$dirty){
                           this.push(key + ': ' + value); 
                        }

                    }, editedFields);
                    console.log(editedFields);
                }
        });
于 2014-01-03T16:33:26.550 回答
0

我想出了一个不错的解决方案,但它仍然不是我想要的。我从另一个涉及从字符串创建 JSON 对象的问题中挽救了一些代码,并提出了以下内容:

本质上,我以与模型绑定相同的方式命名我的字段,然后在调用 form_submit 时构建一个新对象以进行提交。

Plnkr 演示

在演示中,如果您更改任何一个表单字段,然后点击提交,您将看到弹出的对象只有脏​​值。

于 2014-01-03T17:29:31.887 回答