71

我想创建一个角度为 2 的大表单。但我想创建这个包含多个组件的表单,如下例所示。

应用组件

<form novalidate #form1="ngForm" [formGroup]="myForm">
<div>
    <address></address>
</div>
<div>
    <input type="text" ngModel required/>
</div>

<input type="submit" [disabled]="!form1.form.valid" > </form>

地址组件

<div>
<input type="text" ngModel required/> </div>

当我使用上面的代码时,它在我需要的浏览器中可见,但是当我删除地址组件中的文本时,提交按钮没有被禁用。
但是当我删除应用程序组件输入框中的文本时,该按钮被正确禁用。

4

5 回答 5

118

我会使用一种效果很好的反应形式,至于您的评论:

这个还有其他简单的例子吗?也许没有循环的同一个例子

我可以给你一个例子。您需要做的就是嵌套 aFormGroup并将其传递给孩子。

假设您的表单看起来像这样,并且您想将表单组传递address给孩子:

ngOnInit() {
  this.myForm = this.fb.group({
    name: [''],
    address: this.fb.group({ // create nested formgroup to pass to child
      street: [''],
      zip: ['']
    })
  })
}

然后在您的父母中,只需传递嵌套的表单组:

<address [address]="myForm.get('address')"></address>

在您的孩子中,@Input用于嵌套表单组:

@Input() address: FormGroup;

在您的模板中使用[formGroup]

<div [formGroup]="address">
  <input formControlName="street">
  <input formControlName="zip">
</div>

如果您不想创建实际的嵌套表单组,则不需要这样做,您可以将父表单传递给子表单,因此如果您的表单如下所示:

this.myForm = this.fb.group({
  name: [''],
  street: [''],
  zip: ['']
})

你可以通过任何你想要的控制。使用与上面相同的示例,我们只想显示streetand zip,子组件保持不变,但模板中的子标签将如下所示:

<address [address]="myForm"></address>

这是一个

第一个选项的演示,这是第二个演示

此处有关嵌套模型驱动表单的更多信息。

于 2017-04-07T11:47:15.503 回答
16

在模板驱动的表单中也有一种方法可以做到这一点。ngModel 会在每个组件上自动创建一个单独的表单,但您可以通过将其添加到您的组件来注入父组件的表单:

@Component({
viewProviders: [{ provide: ControlContainer, useExisting: NgForm}]
}) export class ChildComponent

但是,您必须确保每个输入都有一个唯一的名称。因此,如果您使用 *ngFor 来调用您的子组件,则必须将索引(或任何其他唯一标识符)放入 name ,例如:

[name]="'address_' + i"

如果你想将你的表单组织成 FormGroups,你可以使用 ngModelGroup 和

viewProviders: [{ provide: ControlContainer, useExisting: NgModelGroup }]

而不是 ngForm 并添加 [ngModelGroup]="yourNameHere" 到一些包含标签的子组件 html 中。

于 2018-11-02T12:05:50.343 回答
7

根据我的经验,这种表单域组合很难用模板驱动的表单来制作。嵌入在地址组件中的字段不会在表单中注册(NgForm.controls 对象),因此在验证表单时不会考虑它们。

  • 您可以创建一个具有所有验证的 ControlValueAccessor 组件(接受 ngModel 属性),但是很难显示验证错误和传播更改(地址被视为具有复杂值的单个表单字段)。
  • 您可能可以将表单引用传递给 Address 组件并在其中注册您的内部控件,但我没有尝试过,这似乎是一种奇怪的方法(我在任何地方都没有看到它)。
  • 您可以切换到反应式表单(而不是模板驱动),将表单组对象(表示地址)传递到地址组件,将验证保留在表单定义中。你可以在这里看到一个例子https://scotch.io/tutorials/how-to-build-nested-model-driven-forms-in-angular-2
于 2017-04-07T06:22:14.487 回答
6

我想分享在我的案例中完成这项工作的方法。我创建了以下指令:

import { Directive } from '@angular/core';
import { ControlContainer, NgForm } from '@angular/forms';

@Directive({
  selector: '[appUseParentForm]',
  providers: [
    {
      provide: ControlContainer,
      useFactory: function (form: NgForm) {
        return form;
      },
      deps: [NgForm]
    }
  ]
})
export class UseParentFormDirective {
}

现在,如果您在子组件上使用此指令,例如:

<address app-use-parent-form></address>

来自 AddressComponent 的控件将被添加到 form1。因此,表单有效性也将取决于子组件内的控件状态。

仅使用 Angular 6 检查

于 2018-06-14T07:20:58.020 回答
0

使用最新版本的 Angular 2+,您可以在多个组件中生成表单,而无需将 formGroup 实例作为输入传递给子组件。

您可以通过将此提供程序导入您的子组件来实现此目的:

viewProviders: [{provide: ControlContainer,useExisting: FormGroupDirective}]

然后您也可以通过注入 formGroupDirective 来访问主表单:

constructor(private readonly formGroupDirective: FormGroupDirective)
于 2022-01-22T22:11:42.120 回答