[译] 别再对 Angular 表单的 ControlValueAccessor 感到迷惑
原文链接: Never again be confused when implementing ControlValueAccessor in Angularforms
如果你正在做一个复杂项目,必然会需要自定义表单控件,这个控件主要需要实现 首先我解释下为啥需要 FormControl 和 ControlValueAccessor如果你之前使用过 Angular 表单,你可能会熟悉 FormControl ,Angular 官方文档将它描述为追踪单个表单控件值和有效性的实体对象。需要明白,不管你使用模板驱动还是响应式表单(译者注:即模型驱动), @Directive({ selector: '[ngModel]...',... }) export class NgModel ... { _control = new FormControl(); <---------------- here 不管 原生表单控件数量是有限的,但是自定义表单控件是无限的,所以 Angular 需要一种通用机制来桥接原生/自定义表单控件和 ControlValueAccessoracts as a bridge between the Angular forms API and a native element in the DOM. 任何一个组件或指令都可以通过实现 interface ControlValueAccessor { writeValue(obj: any): void registerOnChange(fn: any): void registerOnTouched(fn: any): void ... }
下图是
再次强调,不管是使用响应式表单显式创建还是使用模板驱动表单隐式创建, Angular 也为所有原生 DOM 表单元素创建了
从上表中可看到,当 Angular 在组件模板中中遇到 @Component({ selector: 'my-app',template: ` <input [formControl]="ctrl"> ` }) export class AppComponent { ctrl = new FormControl(3); } 所有表单指令,包括上面代码中的 export class FormControlDirective ... { ... ngOnChanges(changes: SimpleChanges): void { if (this._isControlChanged(changes)) { setUpControl(this.form,this); 还有 export function setUpControl(control: FormControl,dir: NgControl) { // initialize a form control // 调用 writeValue() 初始化表单控件值 dir.valueAccessor.writeValue(control.value); // setup a listener for changes on the native control // and set this value to form control // 设置原生控件值更新时监听器,每当原生控件值更新,Angular 表单控件值也更新 valueAccessor.registerOnChange((newValue: any) => { control.setValue(newValue,{emitModelToViewChange: false}); }); // setup a listener for changes on the Angular formControl // and set this value to the native control // 设置 Angular 表单控件值更新监听器,每当 Angular 表单控件值更新,原生控件值也更新 control.registerOnChange((newValue: any,...) => { dir.valueAccessor.writeValue(newValue); }); 只要我们理解了内部机制,就可以实现我们自定义的 Angular 表单控件了。 组件封装器由于 Angular 为所有默认原生控件提供了控件值访问器,所以在封装第三方插件或组件时,需要写一个新的控件值访问器。我们将使用上文提到的 jQuery UI 库的 slider 插件,来实现一个自定义表单控件吧。 简单的封装器最基础实现是通过简单封装使其能在屏幕上显示出来,所以我们需要一个 @Component({ selector: 'ngx-jquery-slider',template: ` <div #location></div> `,styles: ['div {width: 100px}'] }) export class NgxJquerySliderComponent { @ViewChild('location') location; widget; ngOnInit() { this.widget = $(this.location.nativeElement).slider(); } } 这里我们使用标准的 一旦简单封装好了 @Component({ selector: 'my-app',template: ` <h1>Hello {{name}}</h1> <ngx-jquery-slider></ngx-jquery-slider> ` }) export class AppComponent { ... } 为了运行程序我们需要加入 <script src="https://code.jquery.com/jquery-3.2.1.js"></script> <script src="https://code.jquery.com/ui/1.12.1/jquery-ui.js"></script> <link rel="stylesheet" href="//code.jquery.com/ui/1.12.1/themes/smoothness/jquery-ui.css"> 这里是安装依赖的源码。 交互式表单控件上面的实现还不能让我们自定义的 export class NgxJquerySliderComponent { @ViewChild('location') location; @Input() value; @Output() private valueChange = new EventEmitter(); widget; ngOnInit() { this.widget = $(this.location.nativeElement).slider(); this.widget.slider('value',this.value); this.widget.on('slidestop',(event,ui) => { this.valueChange.emit(ui.value); }); } ngOnChanges() { if (this.widget && this.widget.slider('value') !== this.value) { this.widget.slider('value',this.value); } } } 一旦 然后就是父组件中如何使用 <ngx-jquery-slider [value]="sliderValue" (valueChange)="onSliderValueChange($event)"> </ngx-jquery-slider> 源码在这里。 但是,我们想要的是,使用 实现自定义控件值访问器实现自定义控件值访问器并不难,只需要两步:
让我们首先定义提供者: @Component({ selector: 'ngx-jquery-slider',providers: [{ provide: NG_VALUE_ACCESSOR,useExisting: NgxJquerySliderComponent,multi: true }] ... }) class NgxJquerySliderComponent implements ControlValueAccessor {...} 我们直接在组件装饰器里直接指定类名,然而 Angular 源码默认实现是放在类装饰器外面: export const DEFAULT_VALUE_ACCESSOR: any = { provide: NG_VALUE_ACCESSOR,useExisting: forwardRef(() => DefaultValueAccessor),multi: true }; @Directive({ selector:'input',providers: [DEFAULT_VALUE_ACCESSOR] ... }) export class DefaultValueAccessor implements ControlValueAccessor {} 放在外面就需要使用 一旦定义了提供者后,就让我们实现 export class NgxJquerySliderComponent implements ControlValueAccessor { @ViewChild('location') location; widget; onChange; value; ngOnInit() { this.widget = $(this.location.nativeElement).slider(this.value); this.widget.on('slidestop',ui) => { this.onChange(ui.value); }); } writeValue(value) { this.value = value; if (this.widget && value) { this.widget.slider('value',value); } } registerOnChange(fn) { this.onChange = fn; } registerOnTouched(fn) { } 由于我们对用户是否与组件交互不感兴趣,所以先把 现在我们把上面描述的功能做成一张交互式图:
如果你把简单封装和 现在,实现了 @Component({ selector: 'my-app',template: ` <h1>Hello {{name}}</h1> <span>Current slider value: {{ctrl.value}}</span> <ngx-jquery-slider [formControl]="ctrl"></ngx-jquery-slider> <input [value]="ctrl.value" (change)="updateSlider($event)"> ` }) export class AppComponent { ctrl = new FormControl(11); updateSlider($event) { this.ctrl.setValue($event.currentTarget.value,{emitModelToViewChange: true}); } } 你可以查看程序的最终实现。 Github项目的 Github 仓库。 (编辑:李大同) 【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容! |