[译] 别再对 Angular 表单的 ControlValueAccessor 感到迷惑

原文连接: Never again be confused when implementing ControlValueAccessor in Angular forms

ceasy-control-value-accessor

若是你正在作一个复杂项目,必然会须要自定义表单控件,这个控件主要须要实现 ControlValueAccessor 接口(译者注:该接口定义方法可参考 API 文档说明,也可参考 Angular 源码定义)。网上有大量文章描述如何实现这个接口,但不多说到它在 Angular 表单架构里扮演什么角色,若是你不只仅想知道如何实现,还想知道为何这样实现,那本文正合你的胃口。css

首先我解释下为啥须要 ControlValueAccessor 接口以及它在 Angular 中是如何使用的。而后我将展现如何封装第三方组件做为 Angular 组件,以及如何使用输入输出机制实现组件间通讯(译者注:Angular 组件间通讯输入输出机制可参考官网文档),最后将展现如何使用 ControlValueAccessor 来实现一种针对 Angular 表单新的数据通讯机制。html

FormControl 和 ControlValueAccessor

若是你以前使用过 Angular 表单,你可能会熟悉 FormControl ,Angular 官方文档将它描述为追踪单个表单控件值和有效性的实体对象。须要明白,无论你使用模板驱动仍是响应式表单(译者注:即模型驱动),FormControl 都总会被建立。若是你使用响应式表单,你须要显式建立 FormControl 对象,并使用 formControlformControlName 指令来绑定原生控件;若是你使用模板驱动方法,FormControl 对象会被 NgModel 指令隐式建立(译者注:可查看 Angular 源码这一行):react

@Directive({
  selector: '[ngModel]...',
  ...
})
export class NgModel ... {
  _control = new FormControl();   <---------------- here

无论 formControl 是隐式仍是显式建立,都必须和原生 DOM 表单控件如 input,textarea 进行交互,而且颇有可能须要自定义一个表单控件做为 Angular 组件而不是使用原生表单控件,而一般自定义表单控件会封装一个使用纯 JS 写的控件如 jQuery UI's Slider。本文我将使用原生表单控件术语来区分 Angular 特定的 formControl 和你在 html 使用的表单控件,但你须要知道任何一个自定义表单控件均可以和 formControl 指令进行交互,而不是原生表单控件如 inputjquery

原生表单控件数量是有限的,可是自定义表单控件是无限的,因此 Angular 须要一种通用机制来桥接原生/自定义表单控件和 formControl 指令,而这正是 ControlValueAccessor 干的事情。这个对象桥接原生表单控件和 formControl 指令,并同步二者的值。官方文档是这么描述的(译者注:为清晰理解,该描述不翻译):git

 ControlValueAccessor acts as a bridge between the Angular forms API and a native element in the DOM.

任何一个组件或指令均可以经过实现 ControlValueAccessor 接口并注册为 NG_VALUE_ACCESSOR,从而转变成 ControlValueAccessor 类型的对象,稍后咱们将一块儿看看如何作。另外,这个接口还定义两个重要方法——writeValueregisterOnChange (译者注:可查看 Angular 源码这一行):github

interface ControlValueAccessor {
  writeValue(obj: any): void
  registerOnChange(fn: any): void
  registerOnTouched(fn: any): void
  ...
}

formControl 指令使用 writeValue 方法设置原生表单控件的值(译者注:你可能会参考 L186L41);使用 registerOnChange 方法来注册由每次原生表单控件值更新时触发的回调函数(译者注:你可能会参考这三行,L186L43,以及 L85),你须要把更新的值传给这个回调函数,这样对应的 Angular 表单控件值也会更新(译者注:这一点能够参考 Angular 它本身写的 DefaultValueAccessor 的写法是如何把 input 控件每次更新值传给回调函数的,L52L89);使用 registerOnTouched 方法来注册用户和控件交互时触发的回调(译者注:你可能会参考 L95)。api

下图是 Angular 表单控件 如何经过 ControlValueAccessor 来和原生表单控件交互的(译者注:formControl你写的或者 Angular 提供的 CustomControlValueAccessor 两个都是要绑定到 native DOM element 的指令,而 formControl 指令须要借助 CustomControlValueAccessor 指令/组件,来和 native DOM element 交换数据。):架构

angular_form_control-controlValueAccessor-native_form_control

再次强调,不论是使用响应式表单显式建立仍是使用模板驱动表单隐式建立,ControlValueAccessor 都老是和 Angular 表单控件进行交互。app

Angular 也为全部原生 DOM 表单元素建立了 Angular 表单控件(译者注:Angular 内置的 ControlValueAccessor):ide

Accessor Form Element
DefaultValueAccessor input,textarea
CheckboxControlValueAccessor input[type=checkbox]
NumberValueAccessor input[type=number]
RadioControlValueAccessor input[type=radio]
RangeValueAccessor input[type=range]
SelectControlValueAccessor select
SelectMultipleControlValueAccessor select[multiple]

从上表中可看到,当 Angular 在组件模板中中遇到 inputtextarea DOM 原生控件时,会使用DefaultValueAccessor 指令:

@Component({
  selector: 'my-app',
  template: `
      <input [formControl]="ctrl">
  `
})
export class AppComponent {
  ctrl = new FormControl(3);
}

全部表单指令,包括上面代码中的 formControl 指令,都会调用 setUpControl 函数来让表单控件和DefaultValueAccessor 实现交互(译者注:意思就是上面代码中绑定的 formControl 指令,在其自身实例化时,会调用 setUpControl() 函数给一样绑定到 input DefaultValueAccessor 指令作好安装工做,如 L85,这样 formControl 指令就能够借助 DefaultValueAccessor 来和 input 元素交换数据了)。细节可参考 formControl 指令的代码:

export class FormControlDirective ... {
  ...
  ngOnChanges(changes: SimpleChanges): void {
    if (this._isControlChanged(changes)) {
      setUpControl(this.form, this);

还有 setUpControl 函数源码也指出了原生表单控件和 Angular 表单控件是如何数据同步的(译者注:做者贴的多是 Angular v4.x 的代码,v5 有了点小小变更,但基本类似):

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 插件,来实现一个自定义表单控件吧。

简单的封装器

最基础实现是经过简单封装使其能在屏幕上显示出来,因此咱们须要一个 NgxJquerySliderComponent 组件,并在其模板里渲染出 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();
  }
}

这里咱们使用标准的 jQuery 方法在原生 DOM 元素上建立一个 slider 控件,而后使用 widget 属性引用这个控件。

一旦简单封装好了 slider 组件,咱们就能够在父组件模板里使用它:

@Component({
  selector: 'my-app',
  template: `
      <h1>Hello {{name}}</h1>
      <ngx-jquery-slider></ngx-jquery-slider>
  `
})
export class AppComponent { ... }

为了运行程序咱们须要加入 jQuery 相关依赖,简化起见,在 index.html 中添加全局依赖:

<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">

这里是安装依赖的源码

交互式表单控件

上面的实现还不能让咱们自定义的 slider 控件与父组件交互,因此还得使用输入/输出绑定来是实现组件间数据通讯:

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);
    }
  }
}

一旦 slider 组件建立,就能够订阅 slidestop 事件获取变化的值,一旦 slidestop 事件被触发了,就可使用输出事件发射器 valueChanges 通知父组件。固然咱们也可使用 ngOnChanges 生命周期钩子来追踪输入属性 value 值的变化,一旦其值变化,咱们就将该值设置为 slider 控件的值。

而后就是父组件中如何使用 slider 组件的代码实现:

<ngx-jquery-slider
    [value]="sliderValue"
    (valueChange)="onSliderValueChange($event)">
</ngx-jquery-slider>

源码在这里。

可是,咱们想要的是,使用 slider 组件做为表单的一部分,并使用模板驱动表单或响应式表单的指令与其数据通讯,那就须要让其实现 ControlValueAccessor 接口了。因为咱们将实现的是新的组件通讯方式,因此不须要标准的输入输出属性绑定方式,那就移除相关代码吧。(译者注:做者先实现标准的输入输出属性绑定的通讯方式,又要删除,主要是为了引入新的表单组件交互方式,即 ControlValueAccessor。)

实现自定义控件值访问器

实现自定义控件值访问器并不难,只须要两步:

  1. 注册 NG_VALUE_ACCESSOR 提供者
  2. 实现 ControlValueAccessor 接口

NG_VALUE_ACCESSOR 提供者用来指定实现了 ControlValueAccessor 接口的类,而且被 Angular 用来和 formControl 同步,一般是使用组件类或指令来注册。全部表单指令都是使用NG_VALUE_ACCESSOR 标识来注入控件值访问器,而后选择合适的访问器(译者注:这句话可参考这两行代码,L175L181)。要么选择DefaultValueAccessor 或者内置的数据访问器,不然 Angular 将会选择自定义的数据访问器,而且有且只有一个自定义的数据访问器(译者注:这句话参考 selectValueAccessor 源码实现)。

让咱们首先定义提供者:

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

放在外面就须要使用 forwardRef,关于缘由能够参考 What is forwardRef in Angular and why we need it 。当实现自定义 controlValueAccessor,我建议仍是放在类装饰器里吧(译者注:我的建议仍是学习 Angular 源码那样放在外面)。

一旦定义了提供者后,就让咱们实现 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', (event, 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) {  }

因为咱们对用户是否与组件交互不感兴趣,因此先把 registerOnTouched 置空吧。在registerOnChange 里咱们简单保存了对回调函数 fn 的引用,回调函数是由 formControl 指令传入的(译者注:参考 L85),只要每次 slider 组件值发生改变,就会触发这个回调函数。在 writeValue 方法内咱们把获得的值传给 slider 组件。

如今咱们把上面描述的功能作成一张交互式图:

jQuery_slider-slider_component-form_control

若是你把简单封装和 controlValueAccessor 封装进行比较,你会发现父子组件交互方式是不同的,尽管封装的组件与 slider 组件的交互是同样的。你可能注意到 formControl 指令实际上简化了与父组件交互的方式。这里咱们使用 writeValue 来向子组件写入数据,而在简单封装方法中使用 ngOnChanges;调用 this.onChange 方法输出数据,而在简单封装方法中使用 this.valueChange.emit(ui.value)

如今,实现了 ControlValueAccessor 接口的自定义 slider 表单控件完整代码以下:

@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 仓库

相关文章
相关标签/搜索