详解Angular Forms中自定义ngModel绑定值的方式
在 angular 应用中,我们有两种方式来实现表单绑定——“模板驱动表单”与“响应式表单”。这两种方式通常能够很好的处理大部分的情况,但是对于一些特殊的表单控件,例如 input[type=datetime] 、 input[type=file] ,我们需要重写默认的表单绑定方式,让我们绑定的变量不再仅仅只是一个字符串,而是一个 date 或者 file 对象。为了达成这一目的,我们需要自定义表单控件的 controlvalueaccessor 。
controlvalueaccessor 接口是 angular forms api 与 dom 之间的桥梁,通过提供不同的 controlvalueaccessor ,我们就可以使用统一的 angular forms api 来操作不同的 html 表单元素。
在我们使用 ngmodel 或者 formcontrol 的时候,这两个 directive 会向 angular 的依赖注入容器申请实现了 controlvalueaccessor 接口的对象,这是一种典型的面向接口编程的设计。例如,如果我们需要为 input[type=file] 提供一个用来绑定 file 对象的 controlvalueaccessor ,只需要在依赖注入容器中提供一个 filecontrolvalueaccessor 的实现就可以了。不过,我们并不想覆盖其他类型 input 元素的 controlvalueaccessor ,因为那样肯定会对已有代码造成大范围的破坏。所以在这里,我们需要使用 angular 的分层注入能力——在 elementinjector 中提供 filecontrolvalueaccessor 。关于 elementinjector 更多的内容,请看这里 。
下面演示的两个 directive 您都可以在这里查看 。
首先让我们来创建一个 directive,这个指令将会选中 input[type=file][appinputfile]
元素,这样我们就可以有选择的为文件选择器的 elementinjector 定义新的 provider。
@directive({ selector: 'input[type=file][inputfile]', // <1> providers: [ { provide: ng_value_accessor, // <2> useexisting: forwardref(() => inputfiledirective), // <3> multi: true // <4> } ] }) export class inputfiledirective implements controlvalueaccessor, oninit, ondestroy { // 当文件选择器选择的文件发生改变时调用的回调函数 onchange: (any) => any; // 当文件选择器选择的被操作后调用的回调函数 ontouched: () => any; // 监听宿主元素的 change 事件 @hostlistener('change', ['$event.target.files']) onelchange = (files: filelist) => { this.onchange(files); }; // 监听宿主元素的 blur 事件 @hostlistener('blur', []) oneltouched = () => { this.ontouched(); }; constructor(private el: elementref<htmlinputelement>) { // <5> } ngoninit(): void { this.el.nativeelement.addeventlistener('change', this.listener); } // 来自 controlvalueaccessor 接口,用来设置元素的值 writevalue(obj: any): void { this.el.nativeelement.value = obj; } // 来自 controlvalueaccessor 接口,用来将一个函数注册为 onchange 回调函数 registeronchange(fn: any): void { this.onchange = fn; } // 来自 controlvalueaccessor 接口,用来将一个函数注册为 ontouched 回调函数 registerontouched(fn: any): void { this.ontouched = fn; } // 来自 controlvalueaccessor 接口,设置表单元素是否启用 setdisabledstate?(isdisabled: boolean): void { this.el.nativeelement.disabled = isdisabled; } }
上面的代码片段中你可以看到有几处类似 // <1>
的注释,这是我用来在下面的文章中引用该行代码的标记,语法借鉴自 asciidoc
- 通过定义一个复合的选择器,我们可以有选择的对
input[type=file]
重写controlvalueaccessor
-
controlvalueaccessor
的注入 token 是一个常量 ——ng_value_accessor
- 由于 directive 的定义在这行代码的下面,所以需要使用
forwardref
来引用这个依赖的实现。 - 这里需要将 multiple 设置为 true,因为 angular 默认的
controlvalueaccessor
就是提供了多个实现的。在解析依赖的时候,angular 会优先选择我们自定义的实现。 - 为了代码更加简单,我在这里选择了不利于服务端渲染的
elementref.nativeelement
来读取原生 html 元素的属性,如果你对服务端渲染有需求,你应该使用renderer2
来读写元素的属性。
有了这个 directive,我们就可以在 angular forms 中绑定 file 对象了:
<input type="file" [(ngmodel)]="foo.files" inputfile />
date 类型的数据也是日常开发中比较头疼的一个地方,因为在 json 中, date 类型往往会被序列化为字符串,而在前端代码中,我们又需要将其反序列化为 date 对象,最终在页面上展示的时候,我们又需要按照产品需求再将其序列化为制定格式的字符串。现在,有了 controlvalueaccessor
的帮助,我们就可以实现让 input[type=datetime]
与 date
对象进行双向绑定的功能,同时还能够定制 date 对象在输入框中的显示格式。
@directive({ // tslint:disable-next-line:directive-selector selector: 'input[type=datetime][valueasdate]', providers: [ { provide: ng_value_accessor, useexisting: forwardref(() => datevaluedirective), multi: true } ] }) export class datevaluedirective implements controlvalueaccessor { /** * see https://date-fns.org/v2.0.0-alpha.25/docs/format * 自定义日期展示格式 * @type {string} * @memberof datevaluedirective */ // tslint:disable-next-line:no-input-rename @input('valueasdate') format: string; private datevalue: date; @hostlistener('input', ['$event.target.value']) onchange = (_: any) => { }; @hostlistener('blur', []) ontouched = () => { }; get element() { return this.elementref.nativeelement; } constructor( private elementref: elementref, private renderer: renderer2 // <1> ) { } parsedate(str: string) { return parsedate(str, this.format, new date(), { awareofunicodetokens: true }); } formatdate(date: date) { return formatdate(date, this.format, { awareofunicodetokens: true }); } /** * 设置组件的值的时候,先把新的值存到一个成员变量中,然后再把新的值格式化为 string */ writevalue(date: date): void { this.datevalue = date; this.renderer.setproperty(this.element, 'value', this.formatdate(date)); } /** * 在 input 元素值发生变化的时候,先尝试把变化后的值转换成 date 对象 * 如果转换失败,那么依然使用之前的值 * 否则,将新的值传递给回调函数 */ registeronchange(fn: any): void { const onchange = (value: string) => { const date = this.parsedate(value); if (isvaliddate(date)) { this.datevalue = date; fn(date); } else { fn(this.datevalue); } }; this.onchange = onchange; } registerontouched(fn: any): void { this.ontouched = fn; } setdisabledstate?(isdisabled: boolean): void { this.renderer.setproperty(this.element, 'disabled', isdisabled); } }
这里演示了使用 renderer2
来读写元素属性的操作
整个指令的内容仍然非常简单,但是却能够为我们的日常开发带来不小的便利,使用了这个指令后,我们就可以非常容易的为 date 对象进行双向绑定。
<input type="datetime" valueasdate="m/d/yyyy h:mm:ss a" [(ngmodel)]="foo.date">
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
上一篇: [HTML] websocket的模拟日志监控界面
下一篇: 详解Vue-axios 设置请求头问题