angular4更改表单中显示的值_angular4自定义表单控件[(ngModel)]的实现

本文介绍了如何在Angular4中拆分[(ngModel)],实现自定义表单控件的双向数据绑定。通过实现ControlValueAccessor接口,可以创建一个能够与Angular forms API集成的组件,允许在自定义组件上使用[(ngModel)]。文中通过一个简单的自定义radio组件的示例,详细解释了writeValue、registerOnChange等方法的用途及其在数据同步过程中的作用。
摘要由CSDN通过智能技术生成

[(ngModel)]拆分

[(ngModel)] 将[]输入()输出组合起来,进行双向数据绑定。拆分开来

输入属性

[ngModel](ngModelChange)输出监听元素值的变化,并同步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的组件,只能如文档所说,依葫芦画瓢实现 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.

简而言之,实现了这个接口的组件,就可以使用 Angular forms API,比如[(ngModel)]。

interface ControlValueAccessor {

writeValue(obj: any): void

registerOnChange(fn: any): void

registerOnTouched(fn: any): void

setDisabledState(isDisabled: boolean)?: void

}

实现ControlValueAccessor步骤

模仿primeng中的自定义radio组件,写了一个简单的自定义radio组件。

创建一个RADIO_VALUE_ACCESSOR常量用来在组件中注册NG_VALUE_ACCESSOR

实现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: `

{{label}}

`,

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 = 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 { }

方法何时被调用?

writeValue(obj: any): void

API中提到 (model -> view) 时,writeValue() 会被调用。

model value 和 view value分别指什么?

举个调用PRadioComponent的例子:

这里checkedValue属性就是model value,view value 为PRadioComponent内部的某个属性(PRadioComponent中定义为this.value)。

当model view(checkedValue)发生改变时,PRadioComponent中的writeValue(obj: any)就会被调用,参数为当前model value(checkedValue)的值,在函数中将参数值赋给内部的view value,从而实现(model -> view)。接受到model value的值后,改变PRadioComponent的UI显示。

registerOnChange(fn: any): void

这个方法的作用是同步 view value 和 model value (view -> model),

registerOnChange(fn: Function): void {

this.onModelChange = fn;

}

调用this.onModelChange()时候,将view value当作参数传入此方法中,即完成了同步,此例子中this.onModelChange(this.value);。

上面两种方法是相对的:

writeValue(obj: any): model value发生改变 ,完成后UI发生改变(model value-> view value)

registerOnChange(fn: any): 触发事件(比如click),view value和UI发生改变,完成调用后model value与view value同步(view value-> model value)

registerOnTouched(fn: any): void

setDisabledState(isDisabled: boolean)?: void

目的只为在控件中简单的使用[(ngModel)],所以这两个方法没有用到。registerOnTouched(fn: any)必须实现,所以定义了一个空函数。

实际效果

初始值为'a',点击改变view value,在Angury调试工具中看到值改为'b'。然后在调试工具中将checkedValue改为'a',视图发生了改变。可见,完成了数据的双向绑定。

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持脚本之家。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值