angular4自定義表單控件[(ngModel)]的實現(xiàn)
[(ngModel)]拆分
[(ngModel)] 將[]輸入()輸出組合起來,進行雙向數(shù)據(jù)綁定。拆分開來
- 輸入屬性
[ngModel](ngModelChange)輸出監(jiān)聽元素值的變化,并同步view value與model value。
model: string;
constructor() {
this.model = 'model init';
}
getModelChange(event: string) {
this.model = event; // view value 與 model value 同步
}
自定義組件上使用 [(ngModel)]
我們不能把[(ngModel)]用到非表單類的原生元素或第三方自定義組件上,除非寫一個合適的值訪問器,這種技巧超出了本章的范圍。
Angular文檔中描述到這里,就中止了。剛好我要定制一個模擬radio的組件,只能如文檔所說,依葫蘆畫瓢實現(xiàn) ControlValueAccessor。
ControlValueAccessor接口
ControlValueAccessor acts as a bridge between the Angular forms API and a native element in the DOM.
Implement this interface if you want to create a custom form control directive that integrates with Angular forms.
簡而言之,實現(xiàn)了這個接口的組件,就可以使用 Angular forms API,比如[(ngModel)] 。
interface ControlValueAccessor {
writeValue(obj: any): void
registerOnChange(fn: any): void
registerOnTouched(fn: any): void
setDisabledState(isDisabled: boolean)?: void
}
實現(xiàn)ControlValueAccessor步驟
模仿primeng中的自定義radio組件,寫了一個簡單的自定義radio組件。
- 創(chuàng)建一個
RADIO_VALUE_ACCESSOR常量用來在組件中注冊NG_VALUE_ACCESSOR - 實現(xiàn)
ControlValueAccessor中的3+1個方法
完整demo代碼如下:
import { NgModule, Component, Input, Output, ElementRef, OnInit, EventEmitter, forwardRef, ViewChild, ChangeDetectorRef } from '@angular/core';
import { CommonModule } from '@angular/common';
import { NG_VALUE_ACCESSOR, ControlValueAccessor } from '@angular/forms';
const RADIO_VALUE_ACCESSOR: any = {
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => PRadioComponent),
multi: true
};
@Component({
selector: 'app-p-radio',
template: `
<div class="p-radio">
<label class="radio-label" (click)="select()" *ngIf="label">
<div class="name" [class.checked-name]="rb.checked">{{label}}</div>
</label>
<div class="helper-hidden-accessible">
<input #rb type="radio" [attr.name]="name" [attr.value]="value" [checked]="checked">
</div>
<div class="radio-md" (click)="handleClick()">
<div class="radio-icon " [class.radio-checked]="rb.checked">
<div class="radio-inner"></div>
</div>
</div>
</div>
`,
styleUrls: ['./p-radio.component.scss'],
providers: [RADIO_VALUE_ACCESSOR]
})
export class PRadioComponent implements ControlValueAccessor {
@Input() name: string;
@Input() label: string;
@Input() value: string;
checked: boolean;
@ViewChild('rb') inputViewChild: ElementRef;
@Output() pRadioChange: EventEmitter<any> = new EventEmitter();
onModelChange: Function = () => { };
constructor(
private cd: ChangeDetectorRef
) { }
// model view -> view value
writeValue(value: any): void {
if (value) {
this.checked = (value === this.value);
if (this.inputViewChild.nativeElement) {
this.inputViewChild.nativeElement.checked = this.checked;
}
this.cd.markForCheck();
}
}
// view value ->model value
registerOnChange(fn: Function): void {
this.onModelChange = fn;
}
registerOnTouched(fn: Function): void { }
handleClick() {
this.select();
}
select() {
this.inputViewChild.nativeElement.checked = !this.inputViewChild.nativeElement.checked;
this.checked = !this.checked;
if (this.checked) {
this.onModelChange(this.value); // 同步view value 和 model value
} else {
this.onModelChange(null);
}
this.pRadioChange.emit(null);
}
}
@NgModule({
imports: [CommonModule],
exports: [PRadioComponent],
declarations: [PRadioComponent]
})
export class RadioButtonModule { }
方法何時被調(diào)用?
writeValue(obj: any): void
API中提到 (model -> view) 時,writeValue() 會被調(diào)用。
model value 和 view value分別指什么?
舉個調(diào)用PRadioComponent的例子:
這里checkedValue屬性就是model value,view value 為PRadioComponent內(nèi)部的某個屬性(PRadioComponent中定義為this.value)。
當(dāng)model view(checkedValue)發(fā)生改變時,PRadioComponent中的writeValue(obj: any)就會被調(diào)用,參數(shù)為當(dāng)前model value(checkedValue)的值,在函數(shù)中將參數(shù)值賦給內(nèi)部的view value,從而實現(xiàn)(model -> view)。接受到model value的值后,改變PRadioComponent的UI顯示。
registerOnChange(fn: any): void
這個方法的作用是同步 view value 和 model value (view -> model),
registerOnChange(fn: Function): void {
this.onModelChange = fn;
}
調(diào)用this.onModelChange()時候,將view value當(dāng)作參數(shù)傳入此方法中,即完成了同步,此例子中this.onModelChange(this.value);。
上面兩種方法是相對的:
writeValue(obj: any): model value發(fā)生改變 ,完成后UI發(fā)生改變(model value-> view value)registerOnChange(fn: any): 觸發(fā)事件(比如click),view value和UI發(fā)生改變,完成調(diào)用后model value與view value同步(view value-> model value)
registerOnTouched(fn: any): void
setDisabledState(isDisabled: boolean)?: void
目的只為在控件中簡單的使用[(ngModel)],所以這兩個方法沒有用到。registerOnTouched(fn: any)必須實現(xiàn),所以定義了一個空函數(shù)。
實際效果
初始值為'a',點擊改變view value,在Angury調(diào)試工具中看到值改為'b'。然后在調(diào)試工具中將checkedValue改為'a',視圖發(fā)生了改變。可見,完成了數(shù)據(jù)的雙向綁定。

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Angularjs實現(xiàn)帶查找篩選功能的select下拉框示例代碼
這篇文章主要介紹了Angularjs實現(xiàn)帶查找篩選功能的select下拉框的詳細(xì)過程及示例代碼,文中通過示例介紹的很詳細(xì),相信會對大家學(xué)習(xí)使用Angularjs具有一定的參考借鑒價值,有需要的朋友們可以一起來看看。2016-10-10
angularjs 表單密碼驗證自定義指令實現(xiàn)代碼
這篇文章主要介紹了angularjs 表單密碼驗證自定義指令實現(xiàn)代碼,需要的朋友可以參考下2016-10-10
解決angularjs前后端分離調(diào)用接口傳遞中文時中文亂碼的問題
今天小編就為大家分享一篇解決angularjs前后端分離調(diào)用接口傳遞中文時中文亂碼的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-08-08
AngularJS+Node.js實現(xiàn)在線聊天室
隨著互聯(lián)網(wǎng)和信息技術(shù)的發(fā)展,如何快速構(gòu)建高效、強大的動態(tài)網(wǎng)站成為很多人研究的熱點。該文將結(jié)合AngularJS和Node.js構(gòu)建一個在線聊天室,體現(xiàn)AngularJs和Node.js整合的優(yōu)點。2015-08-08
詳解Angular路由 ng-route和ui-router的區(qū)別
這篇文章主要介紹了詳解Angular路由 ng-route和ui-router的區(qū)別,分別介紹了兩種路由的用法和區(qū)別,有興趣的可以了解一下2017-05-05
angularJs中ng-model-options設(shè)置數(shù)據(jù)同步的方法
今天小編就為大家分享一篇angularJs中ng-model-options設(shè)置數(shù)據(jù)同步的方法,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-09-09

