詳解Angular中實(shí)現(xiàn)自定義組件的雙向綁定的兩種方法
在 Angular 中,對(duì)于表單元素,通過(guò) [(ngModel)] 即可以簡(jiǎn)單地實(shí)現(xiàn)雙向綁定。對(duì)于自定義組件而言,希望實(shí)現(xiàn)同樣的效果可以怎么做呢?
1 實(shí)現(xiàn)自定義組件的 ngModel 指令
如果希望自定義組件能夠具有與表單元素相同的 ngModel 效果,可以通過(guò)在組件內(nèi)實(shí)現(xiàn) ControlValueAccessor 接口達(dá)到目的。
對(duì)于 [(ngModel)] ,需要至少實(shí)現(xiàn)該接口的如下方法:
interface ControlValueAccessor {
writeValue(obj: any): void
registerOnChange(fn: any): void
registerOnTouched(fn: any): void
}
最簡(jiǎn)單的核心實(shí)現(xiàn)示例參考如下。
import { ControlValueAccessor } from '@angular/forms/src/directives';
import { Component, forwardRef, Input } from '@angular/core';
import { NG_VALUE_ACCESSOR } from '@angular/forms';
@Component({
selector: 'custom-input',
template: `<input [(ngModel)]="value"/>`,
providers: [
{
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => UnionInputComponent),
multi: true
}
]
})
export class CustomInputComponent implements ControlValueAccessor {
constructor() { }
private innerValue: any = '';
private onTouchedCallback: () => void = function () { };
private onChangeCallback: (_: any) => void = function () { };
get value(): any {
return this.innerValue;
}
set value(v: any) {
if (v !== this.innerValue) {
this.innerValue = v;
this.onChangeCallback(v);
}
}
/**
* model view -> view value
*/
writeValue(value: any) {
if (value !== this.innerValue) {
this.innerValue = value;
}
}
/**
* view value ->model value
*/
registerOnChange(fn: any) {
this.onChangeCallback = fn;
}
registerOnTouched(fn: any) {
this.onTouchedCallback = fn;
}
}
2 使用 get/set 關(guān)鍵字實(shí)現(xiàn)父子組件的雙向綁定
其實(shí)實(shí)現(xiàn)雙向綁定內(nèi)部的本質(zhì)原理就是父子組件的事件綁定機(jī)制。簡(jiǎn)單舉例如下。
2.1 自定義子組件定義
import { Input, Output, Component, EventEmitter } from '@angular/core';
@Component({
selector: 'custom-input',
template: `<input [(ngModel)]="innerValue"/>`,
})
export class CustomInputComponent {
innerValue;
@Input()
get twoWayModel() {
return this.innerValue;
}
set twoWayModel(val) {
this.innerValue = val;
this.twoWayModelChange.emit(this.innerValue);
}
@Output() twoWayModelChange: EventEmitter<string> = new EventEmitter</string><string>();
}
2.2 使用自定義組件
在需要使用組件的地方,通過(guò) [(twoWayModel)] 即可實(shí)現(xiàn)雙向綁定的效果。
import { Input, Output } from '@angular/core';
import { Component, forwardRef, Input } from '@angular/core';
@Component({
selector: 'custom-input',
template: `<custom -input [(twoWayModel)]="inputValue" (twoWayModelChange)="onInputValueChange($event)"></custom>`
})
export class abcComponent {
inputValue;
onInputValueChange(val) {
console.log(val);
console.log(val === this.inputValue); // true
}
}
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
AngularJS模態(tài)框模板ngDialog的使用詳解
這篇文章主要介紹了AngularJS模態(tài)框模板ngDialog的使用詳解,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-05-05
詳解Angular.js的$q.defer()服務(wù)異步處理
相信大家都知道jquery和angular都有defer服務(wù),這篇文章暫以angular為例談?wù)剛€(gè)人的理解,在文章的最后并附上jquery的阮一峰總結(jié)的defer。有需要的朋友們也可以參考借鑒,下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧。2016-11-11
Angular6 Filter實(shí)現(xiàn)頁(yè)面搜索的示例代碼
這篇文章主要介紹了Angular6 Filter實(shí)現(xiàn)頁(yè)面搜索的示例代碼,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-12-12
Angular.js?實(shí)現(xiàn)帶手柄自由調(diào)整頁(yè)面大小的功能
這篇文章主要介紹了Angular.js?實(shí)現(xiàn)帶手柄自由調(diào)整頁(yè)面大小的功能,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-12-12
Angular基于Constructor?Parameter的依賴注入方式詳解
這篇文章主要為大家介紹了Angular基于Constructor?Parameter的依賴注入方式詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11
AngularJS定時(shí)器的使用與移除操作方法【interval與timeout】
這篇文章主要介紹了AngularJS定時(shí)器的使用與移除操作方法,結(jié)合實(shí)例形式分析了AngularJS中interval與timeout方法的相關(guān)使用技巧,需要的朋友可以參考下2016-12-12
Angular 實(shí)現(xiàn)輸入框中顯示文章標(biāo)簽的實(shí)例代碼
這篇文章主要介紹了Angular 實(shí)現(xiàn)輸入框中顯示文章標(biāo)簽的實(shí)例代碼,非常不錯(cuò),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2018-11-11

