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 更多的内容,请看这里 a-curios-case-of-the-host-decorator-and-element-injectors-in-angular。
下面演示的两个 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">
Angular Forms - 自定义 ngModel 绑定值的方式的更多相关文章
- angular中的 input select 值绑定无效,以及多出一个空白选项问题
问题: <!-- 问题标签 --> <select ng-model="sortType"> <option value="1"& ...
- angular 输入框自动绑定值最长为16位,超过16位则会报错
最近发现angular在使用input输入框的ng-model绑定scope变量的时候,发现,输入长串的数字将会出错.代码如下: <html> <head> <meta ...
- asp.net MVC 自定义模型绑定 从客户端中检测到有潜在危险的 Request.QueryString 值
asp.net mvc 自定义模型绑定 有潜在的Requset.Form 自定义了一个模型绑定器.前端会传过来一些敏感字符.调用bindContext. valueProvider.GetValue( ...
- angular select2 ng-model 取值 ng-change调用方法
页面: 引入文件 '/select2.css', '/select2-bootstrap.css', '/select2.min.js', '/ui-select2.js' html: <div ...
- vue中checkbox 样式自定义重写;循环遍历checkbox,拿到不同的v-model绑定值;及获取当前checked 状态,全选和全不选等功能。
开始写这个功能,不得不吐槽原始的checkbox,灰色小方块的丑陋,虽说eleUI,mintUI,等各种框架的单复选框已经对其优化,但还是不想要这种.那我们就来研究一下怎么处理它. <secti ...
- 如何用angularjs制作一个完整的表格之四__自定义ng-model标签的属性使其支持input之外的html元素
有的时候我们需要为非input类型的元素添加ng-model来实现双向的数据绑定,从而减少冗余代码,那么可以尝试一下的方式 例如:我页面中使用了contenteditable这个属性来实现用户可直接编 ...
- 关于.Net Core+Angular+Ueditor富文本编辑器的使用方式
博客:https://www.cnblogs.com/24klr/ 资料:https://www.jianshu.com/p/0b21a1324d47 GitHub:https://github.co ...
- Angular:自定义表单控件
分享一个最近写的支持表单验证的时间选择组件. import {AfterViewInit, Component, forwardRef, Input, OnInit, Renderer} from & ...
- SSRS用自定义对象绑定报表
有一个报表的数据源是一个对象的List, 这个对象List中还有层级,其中还有其他的对象List,这样的层级有三层.其数据是从数据库中取出来的.其LINQ的操作太多了而且复杂,所以不太可 能从LINQ ...
随机推荐
- web服务器原理(作业四)
Web服务器简介:Web服务器是指驻留于因特网上某种类型计算机的程序.当web浏览器(客户端)连到服务器上并请求文件时,服务器将处理该请求并将文件发送到该浏览器上,附带的信息会告诉浏览器如何查看该 ...
- Linux 防火墙相关
1.SELinux 防火墙 1.1 查看SELinux状态: 1) /usr/sbin/sestatus -v ##如果SELinux status参数为enabled即为开启状态 bamb ...
- java基础知识-方法
1.方法 定义:一段定义在类中的业务逻辑的代码. 目的:封装右业务关系的代码,实现代码的复用,即简化代码书写. 2.方法定义的格式 修饰符,返回值类型 方法名(数据类型1,形参名1,数据类型2,形参2 ...
- 【SP1811】 LCS - Longest Common Substring(SAM)
传送门 洛谷 Solution 考虑他要求的是最长公共子串对吧,那么我们对于一个串建后缀自动机,另一个串在后缀自动机上面跑就是了. 复杂度\(O(n+m)\)的,很棒! 代码实现 代码戳这里
- Android ------------------ 带边框的圆角矩形
<?xml version="1.0" encoding="utf-8"?><shape xmlns:android="http:/ ...
- JavaScript学习之路-语法
版权声明:未经博主允许不得转载 在JavaScript中如何写语法呢?这里你可以去看一些教学文档来得快一些,这里不介绍,有点基础的也可以复习一下. //定义变量并赋值 var a; //定义变量 va ...
- Liferay7 BPM门户开发之4: Activiti事件处理和监听Event handlers
事件机制从Activiti 5.15开始引入,这非常棒,他可以让你实现委托. 可以通过配置添加事件监听器,也可以通过Runtime API加入注册事件. 所有的事件参数子类型都来自org.activi ...
- Strom
storm 实时分析概念 离线分析 通常是 需要一段时间的数据积累 积累到一定数量数据后 开始离线分析 无论数据量多大 离线分析 有开始 也有结束 最终得到 ...
- Python常用模块——json & pickle
序列化模块 1.什么是序列化-------将原本的字典,列表等对象转换成一个字符串的过程就叫做序列化 2.序列化的目的 1.以某种存储形式使自定义对象持久化 2.将对象从一个地方传递到另一个地方 3. ...
- (转)mtr命令详解诊断网络路由
原文:https://blog.51cto.com/6226001001/1941355 http://www.zzbiji.com/2212.html----Linux下使用mtr做路由图进行网络分 ...