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 ...
 
随机推荐
- C++获取工程路径、exe路径
			
编码过程中有时候会用到获取工程所在路径或者exe所在的路径信息,这里稍微记录下. 获取工程路径 char pBuf[MAX_PATH]; //存放路径的变量 GetCurrentDirectory(M ...
 - html   2
			
一.列表 信息资源的一种展示形式 二.列表的分类 1.有序列表 <ol> <li>列表项1</li> <li>列表项2</li> </ ...
 - getResource()和getResourceAsStream以及路径问题
			
用JAVA获取文件,听似简单,但对于很多像我这样的新人来说,还是掌握颇浅,用起来感觉颇深,大常最经常用的,就是用JAVA的File类,如要取得c:/test.txt文件,就会这样用File file ...
 - Leeetcode--581. Shortest Unsorted Continuous Subarray
			
Given an integer array, you need to find one continuous subarray that if you only sort this subarray ...
 - Jsp+Struts2+JavaBean+DAO开发模式(1)
			
DAO模式就实现了把数据库表的操作转化对Java类的操作,从而提高程序的可读性,并实现更改数据库的方便性.其架构图如下图. 一共分为五个组件(component) jsp提交页面(一下四其中的一个例子 ...
 - 任务调度及远端管理(基于Quartz.net)
			
这篇文章我们来了解一些项目中的一个很重要的功能:任务调度 可能有些同学还不了解这个,其实简单点说任务调度与数据库中的Job是很相似的东西 只不过是运行的物理位置与管理方式有点不一样,从功能上来说我觉得 ...
 - Android 页面跳转之生命周期调用顺序问题
			
Android Activity 常用技巧 Android Activity 启动模式和任务栈 Android 页面跳转之生命周期调用顺序问题 一.页面跳转逻辑分析 1.1 跳转逻辑分析 Androi ...
 - Yii2 三层设计模式:SQL Command、Query builder、Active Record(ORM)
			
用Yii2也有一段时间了,发现Yii2 Framework对Database的操作有非常良好的结构和弹性. 接下来介绍三种数据库操作方式. SQL Command Level: // Get DB c ...
 - Shell-3--变量
			
用户自定义变量 环境变量 位置参数变量 预定义变量
 - Testing - 软件测试知识梳理 - 相关词汇
			
测试策略 描述测试工程的总体方法和目标:根据测试需求,描述在什么测试阶,依据什么测试要素和目标,进行什么种类的测试,使用什么样的测试方法和工具. 测试策略的制定主要包含如下内容: 确定测试过程要使用的 ...