从 JSON 数据的嵌套表单数组填充表单控件中的值
populate values in Form control from nested Form Array from JSON data
FormArrayformObj 具有 componentDetails 对象数组,而后者又具有嵌套的组件数组数组。
export class FormViewComponent implements OnInit {
public callbackForm: FormGroup;
formObj = {
"componentDetails": [{
"component": [{
"value": "Choice 1"
}, {
"value": "Choice 2"
}],
"cpv": "",
"label": "Description of Problem",
"type": "radio",
"mandatory": true
}]
};
loadObservableForm() {
this.formService.getData(this.id)
.subscribe(
(formObj) => {
this.formObj = formObj;
this.callbackForm = this._formBuild.group({
componentDetails: this._formBuild.array([])
});
this.addComponentDetails();
},
(err) => console.error(err),
() => console.log('observable complete')
);
}
addComponentDetails() {
const control = <FormArray> this.callbackForm.controls.componentDetails;
this.formObj.componentDetails.forEach(x => {
control.push(this.addControl(x));
});
}
addControl(x) {
const group = this._formBuild.group({
label: x.label,
cpv: x.cpv,
type: x.type,
mandatory: x.mandatory,
component: this._formBuild.array([this.addOptions(x)])
});
return group;
}
addOptions(z) {
const control = < FormArray > z.component;
z.component.forEach(y => {
control.push(this.addOptionRow(y.value));
});
}
addOptionRow(value) {
return this._formBuild.group({
value: value
});
}
}
模板HTML:
<form [formGroup]="callbackForm">
<div>
<div formArrayName="componentDetails">
<div *ngFor="let question of callbackForm.controls.componentDetails.controls; let i = index;" [formGroupName]="i">
<div class="row">
<div class="col-md-12 panel-group panel-group--compressed">
<div class="panel panel--default">
<fieldset>
<div class="row" *ngIf="question.controls.type.value === 'radio'">
<div class="col-md-12">
<p>{{ question.controls.label.value }}</p>
<div formArrayName="component">
<div *ngFor="let answer of question.controls.component.controls; let j = index" [formGroupName]="j">
<label class="radio radio--alt radio--stacked">
<input type="radio" name="radio-stacked">
<span class="radio__input"></span>
<span class="radio__label">{{ answer.value }}</span>
</label>
</div>
</div>
</div>
</div>
</fieldset>
</div>
</div>
</div>
</div>
</div>
</div>
</form>
问题:
在组件模板 HTML 中,我无法获取 {{ answer.value }} 的值。我尝试使用 answer.controls.value.value 和 answer.controls.value。没有任何效果。 question.controls.component.controls returns [对象对象]
在表单数组中调用 addOptions
时遇到问题:
component: this._formBuild.array([this.addOptions(x)])
将它放在下面,这样就可以了。我从您的代码中省略了 addOptions
和 addOptionRow
,而是应用了 addControl
:
中的逻辑
addControl(x) {
const group = this._formBuild.group({
label: x.label,
cpv: x.cpv,
type: x.type,
mandatory: x.mandatory,
component: this._formBuild.array([])
});
const ctrl = group.controls.component as FormArray;
x.component.forEach(y => {
ctrl.push(this._formBuild.group({
value: y.value
}))
})
return group;
}
并且您的模板在其他方面是正确的,但要显示标签,请执行以下操作:
{{ answer.value.value }}
而不是
{{ answer.value }}
StackBlitz
FormArrayformObj 具有 componentDetails 对象数组,而后者又具有嵌套的组件数组数组。
export class FormViewComponent implements OnInit {
public callbackForm: FormGroup;
formObj = {
"componentDetails": [{
"component": [{
"value": "Choice 1"
}, {
"value": "Choice 2"
}],
"cpv": "",
"label": "Description of Problem",
"type": "radio",
"mandatory": true
}]
};
loadObservableForm() {
this.formService.getData(this.id)
.subscribe(
(formObj) => {
this.formObj = formObj;
this.callbackForm = this._formBuild.group({
componentDetails: this._formBuild.array([])
});
this.addComponentDetails();
},
(err) => console.error(err),
() => console.log('observable complete')
);
}
addComponentDetails() {
const control = <FormArray> this.callbackForm.controls.componentDetails;
this.formObj.componentDetails.forEach(x => {
control.push(this.addControl(x));
});
}
addControl(x) {
const group = this._formBuild.group({
label: x.label,
cpv: x.cpv,
type: x.type,
mandatory: x.mandatory,
component: this._formBuild.array([this.addOptions(x)])
});
return group;
}
addOptions(z) {
const control = < FormArray > z.component;
z.component.forEach(y => {
control.push(this.addOptionRow(y.value));
});
}
addOptionRow(value) {
return this._formBuild.group({
value: value
});
}
}
模板HTML:
<form [formGroup]="callbackForm">
<div>
<div formArrayName="componentDetails">
<div *ngFor="let question of callbackForm.controls.componentDetails.controls; let i = index;" [formGroupName]="i">
<div class="row">
<div class="col-md-12 panel-group panel-group--compressed">
<div class="panel panel--default">
<fieldset>
<div class="row" *ngIf="question.controls.type.value === 'radio'">
<div class="col-md-12">
<p>{{ question.controls.label.value }}</p>
<div formArrayName="component">
<div *ngFor="let answer of question.controls.component.controls; let j = index" [formGroupName]="j">
<label class="radio radio--alt radio--stacked">
<input type="radio" name="radio-stacked">
<span class="radio__input"></span>
<span class="radio__label">{{ answer.value }}</span>
</label>
</div>
</div>
</div>
</div>
</fieldset>
</div>
</div>
</div>
</div>
</div>
</div>
</form>
问题:
在组件模板 HTML 中,我无法获取 {{ answer.value }} 的值。我尝试使用 answer.controls.value.value 和 answer.controls.value。没有任何效果。 question.controls.component.controls returns [对象对象]
在表单数组中调用 addOptions
时遇到问题:
component: this._formBuild.array([this.addOptions(x)])
将它放在下面,这样就可以了。我从您的代码中省略了 addOptions
和 addOptionRow
,而是应用了 addControl
:
addControl(x) {
const group = this._formBuild.group({
label: x.label,
cpv: x.cpv,
type: x.type,
mandatory: x.mandatory,
component: this._formBuild.array([])
});
const ctrl = group.controls.component as FormArray;
x.component.forEach(y => {
ctrl.push(this._formBuild.group({
value: y.value
}))
})
return group;
}
并且您的模板在其他方面是正确的,但要显示标签,请执行以下操作:
{{ answer.value.value }}
而不是
{{ answer.value }}