前言
成都創(chuàng)新互聯(lián)公司是一家專注于網(wǎng)站建設(shè)、成都網(wǎng)站建設(shè)與策劃設(shè)計(jì),霍山網(wǎng)站建設(shè)哪家好?成都創(chuàng)新互聯(lián)公司做網(wǎng)站,專注于網(wǎng)站建設(shè)十年,網(wǎng)設(shè)計(jì)領(lǐng)域的專業(yè)建站公司;建站業(yè)務(wù)涵蓋:霍山等地區(qū)。霍山做網(wǎng)站價(jià)格咨詢:13518219792
眾所周知Angular響應(yīng)式表單相比較模板驅(qū)動(dòng)表單更大操作性、更易測(cè)試性。因此,我更推薦這類表單創(chuàng)造方式。
當(dāng)一個(gè)用于修改用戶信息的表單,數(shù)據(jù)的來(lái)源總是來(lái)自遠(yuǎn)程;而對(duì)于一個(gè) FormGroup 的創(chuàng)建總在 ngOnInit 中完成。因此,這里會(huì)有一個(gè)表單更新值的問(wèn)題。
而通常我們會(huì)透過(guò) FormGroup 下的三種方式 setValue、patchValue、reset 將值寫入表單當(dāng)中。
當(dāng)然,或許我說(shuō)的這三種方式時(shí)你壓根就沒(méi)有做過(guò),那說(shuō)明在表單上你依賴的是雙向綁定 [(ngModel)],這本身就不是符合 Angular 響應(yīng)式表單的牛B之處了。因此,在此我們不討論這種這種方式。下面來(lái)一起看看詳細(xì)的介紹:
一、創(chuàng)建響應(yīng)式表單
我們模擬一個(gè)用戶信息修改的表單所需要的字段,可能包括:email、nickname 等。
如果以API的方式與現(xiàn)實(shí)字段之間產(chǎn)生一個(gè)關(guān)聯(lián),那么 FormGroup 表示一個(gè)表單,F(xiàn)ormControl 表示表單中的字段。因此,F(xiàn)ormControl 必須包裹在 FromGroup 下面。
下面,我們先簡(jiǎn)單的構(gòu)建一個(gè)響應(yīng)式表單。
別忘記導(dǎo)入 ReactiveFormsModule 模塊。
@Component({ selector: 'app-validation', template: ` <form [formGroup]="form" (ngSubmit)="_submitForm(form)"> <input type="email" formControlName="email"> <input type="text" formControlName="nickname"> <button type="submit" [disabled]="form.invalid">Submit</button> </form> ` }) export class UserEditComponent { constructor(private fb: FormBuilder, private route: ActivatedRoute) {} ngOnInit() { this.form = this.fb.group({ email: ['', Validators.compose([Validators.required, Validators.email])], nickname: ['', [Validators.required]] }); this.route.params .switchMap((params: Params) => loadUser(+params['id'])) .subscribe(data => { // Updating value }); } loadUser() { return Observable.of({ email: 'xx@xx.com', nickname: 'cipchk' }).delay(1000); } _submitForm({ value }) { // Save value } }
以上的這些代碼再熟悉不過(guò)了。假設(shè) UserEditComponent 是由路由 /user/edit/1 觸發(fā),那么會(huì)發(fā)生幾個(gè)幾件事情。
首先,創(chuàng)建一個(gè)空的響應(yīng)式表單 form。
this.form = this.fb.group({ email: ['', Validators.compose([Validators.required, Validators.email])], nickname: ['', [Validators.required]] });
表單的內(nèi)容有 email、nickname 兩個(gè)字段。
然而,HTML中,除了 formGroup、formControlName 的配置以外,也看不到任何有關(guān)對(duì)表單的校驗(yàn)代碼。但,當(dāng)我們輸入一個(gè)無(wú)效 Email 時(shí) input 會(huì)自動(dòng)加上 ng-invalid 類。
這便是響應(yīng)式表單的魅力。
現(xiàn)在我們回到正題,將分別針對(duì) setValue、patchValue、reset 三種不同更新表單值實(shí)際上會(huì)發(fā)生什么。
二、patchValue
正如名稱那般,打補(bǔ)丁。假如我們?cè)?email 文本框里輸入:xx@xx.com,接著調(diào)用:
this.form.patchValue({ nickname: 'cipchk' });
最終的結(jié)果是兩個(gè)字段同時(shí)擁有值,因?yàn)檫@里我們只對(duì) nickname 設(shè)置了值,而 email 并沒(méi)有,那只是先前人為錄入的數(shù)據(jù)。
那么 patchValue 實(shí)際上做了什么呢?
patchValue(value: {[key: string]: any}, options: {onlySelf?: boolean, emitEvent?: boolean} = {}): void { Object.keys(value).forEach(name => { if (this.controls[name]) { this.controls[name].patchValue(value[name], {onlySelf: true, emitEvent: options.emitEvent}); } }); this.updateValueAndValidity(options); }
首先,利用 Object.keys 查找主鍵,并以主鍵名查找相應(yīng)的 FromControl 實(shí)例對(duì)象:
Object.keys({ nickname: 'cipchk' }).forEach(name => { console.log(name); }); // [ 'nickname' ]
然后,更新值:
this.controls[name].patchValue(value[name], {onlySelf: true, emitEvent: options.emitEvent});
而 FromControl 實(shí)例的 patchValue 和 FromGroup 不同,他只是單純的更新 FromControle 實(shí)例對(duì)象中的 value 值。
value 相當(dāng)于表單實(shí)際值,還記得先前HTML中的 formControlName 就是將實(shí)例與DOM產(chǎn)生聯(lián)系,這也就是為什么不需要在DOM中使用雙向綁定的原因。
三、setValue
跟 patchValue 有一點(diǎn)不一樣,當(dāng)我們提供一個(gè) FromGroup 中并不存在的字段時(shí),會(huì)拋出一個(gè)錯(cuò)誤。除此之外,與 patchValue 并無(wú)不同。
setValue(value: {[key: string]: any}, options: {onlySelf?: boolean, emitEvent?: boolean} = {}): void { this._checkAllValuesPresent(value); Object.keys(value).forEach(name => { this._throwIfControlMissing(name); this.controls[name].setValue(value[name], {onlySelf: true, emitEvent: options.emitEvent}); }); this.updateValueAndValidity(options); }
主要是 this._throwIfControlMissing(name); 當(dāng)傳遞的對(duì)象有一個(gè)不是 FromControl 時(shí)直接拋棄一個(gè) Error。
_throwIfControlMissing(name: string): void { if (!Object.keys(this.controls).length) { throw new Error(` There are no form controls registered with this group yet. If you're using ngModel, you may want to check next tick (e.g. use setTimeout). `); } if (!this.controls[name]) { throw new Error(`Cannot find form control with name: ${name}.`); } }
四、reset
正常情況下,表單需要提供一個(gè)重置按鈕時(shí)調(diào)用此方法。
reset(formState: any = null, options: {onlySelf?: boolean, emitEvent?: boolean} = {}): void { this._applyFormState(formState); this.markAsPristine(options); this.markAsUntouched(options); this.setValue(this._value, options); }
除了恢復(fù)校驗(yàn)狀態(tài)以外。最后一句代碼是調(diào)用 setValue,這等同上一節(jié)說(shuō)的。因此,當(dāng)我們調(diào)用此方法時(shí),允許我們直接傳遞一個(gè)數(shù)據(jù)對(duì)象做為重置后的默認(rèn)值,比如:
<button (click)="form.reset({ nickname: 'xx' })">Reset</button>
重置表單后并設(shè)置 nickname 默認(rèn)值為:xx。
結(jié)論
每一種不同更新值方式都會(huì)有不一樣的結(jié)果,當(dāng)我們回頭過(guò)看開(kāi)頭中留下來(lái)的:
// Updating value
如果是你,你會(huì)怎么寫呢?
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來(lái)一定的幫助,如果有疑問(wèn)大家可以留言交流,謝謝大家對(duì)創(chuàng)新互聯(lián)的支持。
當(dāng)前題目:Angular中響應(yīng)式表單的三種更新值方法詳析
文章源于:http://www.chinadenli.net/article22/pgejcc.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供商城網(wǎng)站、App設(shè)計(jì)、定制開(kāi)發(fā)、外貿(mào)建站、面包屑導(dǎo)航、電子商務(wù)
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請(qǐng)盡快告知,我們將會(huì)在第一時(shí)間刪除。文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如需處理請(qǐng)聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時(shí)需注明來(lái)源: 創(chuàng)新互聯(lián)