所以我有这个设置:
型号:
export class MapDetailModel{
id: number;
lat: number;
lon: number;
alt: number;
long: number;
angle: number;
distance?: number;
pendenza?: number;
}
HTML:
<div class="element-list">
<form [formGroup]="survey" novalidate (ngSubmit)="onSubmit(survey)" *ngIf="items">
<div formArrayName="sections">
<div class="single-item" *ngFor="let section of getSections(survey); let i = index">
<div [formGroupName]="i">
Lat: <input type="text" formControlName="lat" class="frm-txt">
Lon: <input type="text" formControlName="lon" class="frm-txt">
Long: <input type="text" formControlName="long" class="frm-txt">
Angle: <input type="text" formControlName="angle" class="frm-txt">
a: {{i.angle}}
<a (click)="updateData(i)">Update data</a>
</div>
</div>
</div>
</form>
</div>
ts:
@Component({
//moduleId: module.id,
changeDetection: ChangeDetectionStrategy.OnPush,
selector: 'my-info-bar',
templateUrl: 'map-info-bar.component.html',
styleUrls: ['map-info-bar.component.css'],
providers: [],
})
export class MapInfoBarComponent implements OnInit, OnDestroy
{
zoomLevels: any[];
points: MapDetailModel[] = [];
isAlive = true;
survey: FormGroup;
items = false;
constructor(
private mapService: MapService,
private fb: FormBuilder,
private changeDetectorRef: ChangeDetectorRef
){}
ngOnInit() {
this.mapService.singlePointChanged
.takeWhile(() => this.isAlive)
.subscribe( evt => {
if(!!evt && !!this.survey.value.sections[0]){
let elem;
this.points.forEach(el => {
if(el.id == evt.id){
el = evt;
elem = el;
}
});
(<FormArray>this.survey.get('sections')).at(elem.id).patchValue(elem);
this.changeDetectorRef.detectChanges();
}
})
this.mapService.pointsChanged
.takeWhile(() => this.isAlive)
.subscribe( evt => {
if(!!evt){
this.points = evt;
if(!this.items){
this.survey = this.fb.group({
sections: this.fb.array([
this.initSection(0),
]),
});
this.items = true;
} else {
this.addSection(this.points.length - 1);
}
}
});
}
initSection(idx: number) {
return this.fb.group({
lat: [this.points[idx].lat],
lon: [this.points[idx].lon],
long: [this.points[idx].long],
angle: [this.points[idx].angle]
});
}
addSection(idx: number) {
const control = <FormArray>this.survey.get('sections');
control.push(this.initSection(idx));
}
getSections(form) {
return form.controls.sections.controls;
}
updateData(index: number){
const values = this.survey.value.sections[index] as MapDetailModel;
this.mapService.setChangePoint(values);
}
ngOnDestroy(): void {
this.isAlive = false;
}
}
到目前为止,一切工作都很好。我正确显示了数据,很高兴。问题是每当执行此行时:
(<FormArray>this.survey.get('sections')).at(elem.id).patchValue(elem);
它给了我这个错误:
ERROR TypeError: Cannot read property 'patchValue' of undefined
但是它的结构是这样的:
那么如何更新位置elem.id
处的formArray值?我不明白,如果应该.get
找到'section'
数据,怎么能不确定呢?
谢谢。
答案 0 :(得分:1)
TL; DR -这是因为您在初始化表单之前调用了此操作。
您当前正在做什么-仅在触发pointsChanged
后才初始化表单。在此之前,表单将保持未定义。
您应该考虑做什么-在ngOnInit
上初始化表单,或者在您声明表单后立即初始化表单,如下所示:
survey: FormGroup = this.fb.group({
sections: this.fb.array([]),
});
触发pointsChanged
时,请考虑对收到的数据使用patchValue
。