我正在尝试通过示例(full example on stackblitz)来描述问题
如果我尝试将反应形式的某些部分以带有子组件的简单“ formControls”或“ formGroups”的形式放置,则没有问题。 (请参阅上面关于stackblitz的示例)。 FormGroupDirective
可以正常工作。
但是,如果我尝试将FormArray放置在子组件中,则会因为以下原因而遇到麻烦:
<div [formGroupName]="i">
<!--
Error: formGroupName must be used
with a parent formGroup directive.
-->
<div>
反应形式是一种简单的形式:
ngOnInit () {
this.newForm = this.fb.group({
id: [{value: '4711', disabled: this.idReadOnly}, Validators.required],
chapter: this.fb.group({
name: ['Some Chapter', Validators.required]
}),
faq: this.fb.array(this.faqArray)
});
}
如上所述,id
和chapter
都没有问题,它们在custom-child-components中实现,例如:
<fe-input
[controlPath]="['id']"
placeholder="Child-FormControl ID"
></fe-input>
<my-form-group
[controlPath]="['chapter', 'name']"
placeholder="Child-FormGroup Chapter"
[required]="true"
>
</my-form-group>
在stackblitz的App中,您将看到工作部件“ ID”和“章”。
与formArray
相同的方法:
<my-form-array
[controlPath]="['faq']"
placeholder="Child-FormArray FAQ"
[required]="true"
></my-form-array>
应该按我的预期工作,但是<div [formGroupName]="i">
部分仅在子组件内导致上面已经提到的错误:
Error: formGroupName must be used
with a parent formGroup directive.
如果我在原始文件中使用了它(定义了反应形式),那么它将正常工作。
我很困惑,也许我只是忽略了一个简单的问题?有人能帮我吗?整个示例都可以在此处在线获得:( >> stackblitz)
更新1:
我已经将@yurzui的解决方案与
集成在一起
viewProviders: [{
provide: ControlContainer,
useExisting: FormGroupDirective
}]
,错误“ Error: formGroupName must be used..
”消失了。之后,我为formArray字段集成了自定义组件,它们无法获取控件值。我想我已经接近解决方案了。这是自定义组件:
import { Component, OnInit, Input } from '@angular/core';
import {
FormControl,
FormGroupDirective
} from '@angular/forms';
@Component({
selector: 'fe-input',
template: `
<mat-form-field>
<input
matInput
[placeholder]="placeholder"
[formControl]="control"
[required]="required"
/>
<mat-error>
This is a required field!
</mat-error>
</mat-form-field>
`,
styles: [
]
})
export class FormElementInputComponent implements OnInit {
// Values to be set:
@Input() controlPath: any;
@Input() placeholder: string;
@Input() controlValue: any;
@Input() required = false;
// That's the reuseable control
control: FormControl;
constructor(private fgd: FormGroupDirective) {}
ngOnInit() {
this.control = this.fgd.control.get(
this.controlPath
) as FormControl;
}
}
最后,这是my-form-array
的模板部分:
template: `
<div fxLayout="column">
<div *ngFor="let c of control.controls; index as i">
<div [formGroupName]="i">
<fe-input
[controlPath]="q"
placeholder="Question"
[required]="required"
></fe-input>
<fe-input
[controlPath]="a"
placeholder="Answer"
[required]="required"
></fe-input>
<div>
</div>
</div>
`,
答案 0 :(得分:1)
首先,您的FormArray是一个FormControls的FormArray,这些FormControls获得一个Object而不是FormGroup的FormArray作为值。创建类似
的表单时必须更改this.newForm = this.fb.group({
id: [{value: '4711', disabled: this.idReadOnly}, Validators.required],
chapter: this.fb.group({
name: ['Some Chapter', Validators.required]
}),
faq: this.fb.array(this.faqArray.map(x=>this.fb.group({
q:x.q,
a:x.a
})))
});
第二,您可以使用“其他方式”来使用FormGroup的FormArray,而不使用[formGroupName =“ i”]否则[formGroup] =“ control”。是的
<!--remove this that not work
<div *ngFor="let c of control.controls; index as i">
<div [formGroupName]="i">
....
<div>
</div>
-->
<!--use [formGroup] -->
<div *ngFor="let c of control.controls; index as i">
<div [formGroup]="c">
....
</div>
</div>
好吧,在formGroup内,您需要fe输入[controlPath] =“'q'”和[controlPath] =“'a'”
我也删除了表单组,因为它也是错误的,请尝试使用[formGroup]并使用viewProviders:[{提供:ControlContainer,useExisting:FormGroupDirective}]
您看到自己的forked stackblitz
更新,如果我们使用以下形式标记formArray
<my-form-array
[controlPath]="['faq','a','q']"
placeholder="Child-FormArray FAQ"
[required]="true"
></my-form-array>
我们可以在ngOnInit中更改my-form-array
this.control = this.fgd.control.get(
this.controlPath[0]
) as FormArray;
.html
<div *ngFor="let c of control.controls; index as i">
<div [formGroup]="c">
<fe-input *ngFor="let fields of controlPath.slice(1)"
[controlPath]="fields"
placeholder="Question"
[required]="required"
></fe-input>
</div>
答案 1 :(得分:0)
因此,在子组件中,您必须使用formArrayName="faq"
,您将使用FormGroupDirective
从父组件中获取,而在子组件的viewProviders数组中,您必须使用useExisting
>
viewProviders: [
{ provide: ControlContainer, useExisting: FormGroupDirective }
]
在您的html中
<div formArrayName="faq" *ngFor="let fa of faq['controls']; let faqIndex = index;">
<div [formGroupName]="faqIndex">
// here goes your code
</div>
</div>