使用FormArray製作動態表單。每建立一個表單,頁面就新增一個input顯示錶單填寫的標題,點選編輯再跳轉到點選表單的填寫內容。【相關教學推薦:《》】
// 封裝獲取modelList
get modelList() {
return this.formGroup.get('modelList') as FormArray
}
constructor(private fb: FormBuilder) {}
ngOnInit() {
// 一開始初始化arr為空陣列
this.formGroup = this.fb.group({
// 內部巢狀FormControl、FormArray、FormGroup
modelList: this.fb.array([])
})
}
// 模態框構造內部的表單
function newModel() {
return this.fb.group({
modelName: [''],
// 可以繼續巢狀下去,根據業務需求
})
}
// 省略模態框部分程式碼
// 傳遞到模態框的FormArray
selectedType: FormArray
登入後複製
<form [FormGroup]="formGroup">
<div FormArrayName="modelList">
<ng-container *nfFor="let item of modelList.controls;let i = index" [FormGroupName]="i">
<nz-input-group
[nzSuffix]="suffixIconSearch"
>
<input type="text" nz-input formControlName="modelName"/>
</nz-input-group>
<ng-template #suffixIconSearch>
<span
nz-icon
nzType="edit"
class="hover"
(click)="showModal(i)"
></span>
</ng-template>
</ng-container>
</div>
</form>
<nz-modal
[(nzVisible)]="isVisible"
nzTitle="Model"
[nzFooter]="modalFooter"
(nzOnCancel)="handleCancel()"
(nzOnOk)="handleOk()"
>
<ng-container *nzModalContent>
<form nz-form [formGroup]="selectedType">
<nz-form-item>
<nz-form-label nzRequired>Model Test</nz-form-label>
<nz-form-control>
<input
type="text"
nz-input
placeholder="請輸入ModelName"
formControlName="modelName"
/>
</nz-form-control>
</nz-form-item>
<nz-form-item>
<nz-form-control>
<product-config></product-config>
</nz-form-control>
</nz-form-item>
</form>
</ng-container>
<ng-template #modalFooter>
<button *ngIf="!isNewModel" nzDanger nz-button nzType="default" (click)="handleDelete()">刪除</button>
<button *ngIf="isNewModel" nz-button nzType="default" (click)="handleCancel()">取消</button>
<button nz-button nzType="primary" (click)="handleOk()">儲存</button>
</ng-template>
</nz-modal>
登入後複製
由於這種模態框比較特殊,割裂了表單的FormGroup之間的關係,在點選的時候需要傳遞引數到模態框顯示部分值,如果單純傳遞引數使用this.modelList.at(index)
獲取實體到模態框上進行賦值修改,在模態框點選儲存後會發現修改的值沒有在表單更新,而表單上對input值修改發現可以影響到模態框的內容。
但是模態框新增的表單卻可以響應到頁面中去。
點選編輯後,將點選的FormArray的元素傳遞給一個臨時變數 this.selectedType = <FormGroup>this.modelList.at(index);
,並且對模態框表單傳值。
模態框點選儲存再將原FormArray的值重新替換
this.modelList.removeAt(this.modelIndex)
this.modelList.insert(this.modelIndex, this.selectedType)
登入後複製
點選新增,建立一個新的FormGroup物件
儲存新增push到原頁面的FormArray中
newModelType(): FormGroup {
return this.fb.group({
modelName: ['', Validators.required],
configList: this.fb.array([]),
});
}
// ...省略
// 模態框顯示
show() {
this.isVisible = true
this.selectedType = this.newModelType();
}
// 儲存
save() {
this.isVisible = false
// 原頁面FormArray
this.modelList.push(this.selectedType);
}
登入後複製
最後發現這種寫法只能夠單向改變,頁面外input修改值會影響到模態框,但是模態框的值改變儲存卻讓外部沒有更新。通過console方式檢視頁面的FormArray內部引數發現其實是有改變的,只是angular沒有檢測到。這個時候判斷沒有發生響應的原因一般是沒有觸發angular檢測機制,仔細檢視檔案發現有一行很重要 在最下面寫著
原本第一次閱讀的時候,覺得我遵守了這種原則,因為在編輯的時候,我選擇了操控原FormArray進行元素刪除和插入,是遵循了這種規則,但是實際上在模態框賦值就已經違反了這種原則,我在賦值的時候拿了FormArray的元素範例賦值給模態框的臨時變數,然後更改範例的值,又重新刪除插入,本質上我操作的是同一個範例,所以angular沒有檢測到發生變化【雖然值發生改變】
在賦值的地方不能偷懶,仍然要重新建立新物件,再拿原物件的賦值。【相當於深拷貝】
this.selectedType = this.newModelType();
const old = this.modelList.at(index);
this.selectedType.setValue({
'modelName': old.get('modelName').value
})
登入後複製
這時候就可以正常更新了。
其實查到最後本質上還是迴歸檔案。在排查錯誤也走了很多坑,而且國內基本沒什麼angular的文章,還得靠外網論壇去找問題。
更多程式設計相關知識,請存取:!!
以上就是淺析Angular中怎麼結合使用FormArray和模態框的詳細內容,更多請關注TW511.COM其它相關文章!