Angular使用ControlValueAccessor創(chuàng)建自定義表單控件
在 Angular 自定義表單控件,有時你想要的輸入不是標準的文本輸入、選擇或復(fù)選框。通過實現(xiàn)ControlValueAccessor 接口并將組件注冊為 NG_VALUE_ACCESSOR,您可以將自定義表單控件無縫地集成到模板驅(qū)動或響應(yīng)表單中,就像它是本地表單一樣!
ControlValueAccessor
ControlValueAccessor 是一個接口,充當Angular API 和 DOM 元素之間的橋梁
ControlValueAccessor 是一個連接表單模型和視圖(DOM元素)的接口,自定義的表單控件必須實現(xiàn)這個接口,它的作用是:
- 把 form 模型中值映射到視圖中
- 當視圖發(fā)生變化時,通知 form directives 或 form controls
Angular 引入這個接口的原因是,不同的輸入控件數(shù)據(jù)更新方式是不一樣的。例如,對于我們常用的文本輸入框來說,我們是設(shè)置它的 value 值,而對于復(fù)選框 (checkbox) 我們是設(shè)置它的 checked 屬性。實際上,不同類型的輸入控件都有一個 ControlValueAccessor,用來更新視圖
Angular 中常見的 ControlValueAccessor 有:
- DefaultValueAccessor - 用于 text 和 textarea 類型的輸入控件
- SelectControlValueAccessor - 用于 select 選擇控件
- CheckboxControlValueAccessor - 用于 checkbox 復(fù)選控件
export interface ControlValueAccessor { writeValue(obj: any) : void registerOnChange(fn: any) : void registerOnTouched(fn: any) : void }
writeValue(obj:any)是將表單模型中的值寫入視圖中。
writeValue(value: any): void { this._renderer.setProperty(this._elementRef.nativeElement, 'value', value); }
registerOnChange(fn:any)是一個方法,用于注冊在視圖中的某些內(nèi)容發(fā)生更改時應(yīng)調(diào)用的處理程序。它獲取一個函數(shù),告訴其他表單指令和表單控件更新其值。
registerOnChange(fn: (_: any) => void): void { this._onChange = fn; }
registerOnTouched(fn:any)與registerOnChange()此類似,它專門為控件接收觸摸事件時注冊一個處理程序。
registerOnTouched(fn: any): void { this._onTouched = fn; }
setDisabledState?(isDisabled: boolean): void; 是一個可選的方法,設(shè)置自定義表單的狀態(tài)
setDisabledState(isDisabled: boolean): void { this._renderer.setProperty(this._elementRef.nativeElement, 'disabled', isDisabled); }
AbstractValueAccessor
我們可以把 ControlValueAccessor 中的方法寫在一個抽象類中,不同的組件可以實現(xiàn)這個基類
export abstract class AbstractValueAccessor implements ControlValueAccessor { private _value: any = ''; get value(): any { return this._value; } set value(v: any) { if (v !== this._value) { this._value = v; this.onChange(v); this.onTouched(); } } writeValue(value: any) { this._value = value; } onChange = (_) => {}; onTouched = () => {}; registerOnChange(fn: (_: any) => void): void { this.onChange = fn; } registerOnTouched(fn: () => void): void { this.onTouched = fn; } } export function MakeProvider(type: any): { provide: any, useExisting: any, multi: boolean} { return { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => type), multi: true }; }
Example
自定義一個 list 控件,可以選擇年級
以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
高效利用Angular中內(nèi)置服務(wù)$http、$location等
這篇文章主要介紹了如何高效利用Angular中內(nèi)置服務(wù),大家知道的Angular內(nèi)置服務(wù)有哪些,感興趣的小伙伴們可以參考一下2016-03-03深入理解Angularjs向指令傳遞數(shù)據(jù)雙向綁定機制
這篇文章主要深入的給大家介紹了Angularjs向指令傳遞數(shù)據(jù),雙向綁定機制的相關(guān)資料,需要的朋友可以參考下2016-12-12AngularJS實現(xiàn)的鼠標拖動畫矩形框示例【可兼容IE8】
這篇文章主要介紹了AngularJS實現(xiàn)的鼠標拖動畫矩形框,涉及基于AngularJS的事件響應(yīng)及頁面元素屬性動態(tài)操作相關(guān)實現(xiàn)技巧,需要的朋友可以參考下2019-05-05AngularJS路由實現(xiàn)頁面跳轉(zhuǎn)實例
這篇文章主要為大家詳細介紹了AngularJS路由實現(xiàn)頁面跳轉(zhuǎn)的實例,具有一定的參考價值,感興趣的小伙伴們可以參考一下2017-03-03全面解析Angular中$Apply()及$Digest()的區(qū)別
$apply()和$digest()在AngularJS中是兩個核心概念,但是有時候它們又讓人困惑。這篇文章主要介紹了Angular中$Apply()及$Digest()區(qū)別詳細說明的相關(guān)資料,需要的朋友可以參考下2016-08-08