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

  1. 通過定義一個復合的選擇器,我們可以有選擇的對 input[type=file] 重寫 ControlValueAccessor
  2. ControlValueAccessor 的注入 token 是一個常量 —— NG_VALUE_ACCESSOR
  3. 由于 Directive 的定義在這行代碼的下面,所以需要使用 forwardRef 來引用這個依賴的實現。
  4. 這里需要將 multiple 設置為 true,因為 Angular 默認的 ControlValueAccessor 就是提供了多個實現的。在解析依賴的時候,Angular 會優先選擇我們自定義的實現。
  5. 為了代碼更加簡單,我在這里選擇了不利于服務端渲染的 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-selectorselector: '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);}
}
  1. 這里演示了使用 Renderer2 來讀寫元素屬性的操作

整個指令的內容仍然非常簡單,但是卻能夠為我們的日常開發帶來不小的便利,使用了這個指令后,我們就可以非常容易的為 Date 對象進行雙向綁定。

<input type="datetime" valueAsDate="M/d/yyyy h:mm:ss a" [(ngModel)]="foo.date">

轉載于:https://www.cnblogs.com/JacZhu/p/10087447.html

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/250071.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/250071.shtml
英文地址,請注明出處:http://en.pswp.cn/news/250071.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

[web性能優化] - 使用在線工具對html、js、css進行壓縮

參考 1. 學習點 使用 在線工具對html、css、js進行壓縮學會分析壓縮前后的效率提高點 2. 解決方案: 2.1 HTML壓縮 在線壓縮nodejs提供了 html-minifier工具(在構建層對代碼進行壓縮)后端模板引擎渲染壓縮 2.2 CSS壓縮 使用html-minifier對html中的css進行壓縮使用clean-cs…

【LOJ】 #2540. 「PKUWC2018」隨機算法

題解 感覺極其神奇的狀壓dp \(dp[i][S]\)表示答案為i&#xff0c;然后不可選的點集為S 我們每次往答案里加一個點&#xff0c;然后方案數是&#xff0c;設原來可以選的點數是y&#xff0c;新加入一個點后導致了除了新加的點之外x個點不能選&#xff0c;那么方案就是把x個數在y …

Shiro的authc過濾器的執行流程

1.先執行isAccessAllowed()&#xff0c;通過subject.isAuthenticated()判斷當前session中的subject是否已經登陸過。如果在當前session即會話中已經登陸過&#xff0c;返回true&#xff0c;authc過濾器放行請求到loginUrl。 問題? 這里會有一個問題&#xff0c;如果我登陸成功…

SpringBoot之基礎

簡介 背景 J2EE笨重的開發 / 繁多的配置 / 低下的開發效率 / 復雜的部署流程 / 第三方技術集成難度大 特點 ① 快速創建獨立運行的spring項目以及主流框架集成 ② 使用嵌入式的Servlet容器, 應用無需達成war包 ③ starters自動依賴和版本控制 ④ 大量自動配置, 簡化開發, 也可修…

[Java核心技術(卷I)] - vscode手動編譯運行繼承類

參考 - P160~P161 主要有3個類: 一個測試類(ManagerTest)、一個子類(Manager)、一個父類(Employee) 注意點: -1. 使用 javac -d . *.java進行預編譯 目錄結構入下: 此時會生成目錄結構如下: 之后運行 java com.inheritance.ManagerTest 附上幾個類的代碼 // com.inhe…

mysql常用語句和函數

mysql語句如果長期不寫&#xff0c;就會忘掉&#xff0c;所以要時常復習&#xff0c;溫故而知新。 1.select length("中國人"),select char_length("中國人"); 2建立數據庫的語句 use new_schema;create table ta(id int primary key);這是小括號&#xff…

shiro框架@RequiresPermissions 解釋

RequiresAuthentication 驗證用戶是否登錄&#xff0c;等同于方法subject.isAuthenticated() 結果為true時。 RequiresUser 驗證用戶是否被記憶&#xff0c;user有兩種含義&#xff1a; 一種是成功登錄的&#xff08;subject.isAuthenticated() 結果為true&#xff09;&…

【Social Listening實戰】當數據分析遭遇心理動力學:用戶深層次的情感需求浮出水面...

本文轉自知乎 作者&#xff1a;蘇格蘭折耳喵 ————————————————————————————————————————————————————— 本文篇幅較長&#xff0c;分為五部分&#xff0c;在中間部分有關于心理分析工具的介紹&#xff0c;案例分散在第二部…

Python 字符串切片

#-*- coding:utf-8 -*-#字符串切片names "abcdefgh"切片語法 names[起始位置:終止位置:步長] 起始位置:即字符串的下標&#xff0c;可以是正序下標(0,1,2...)&#xff0c;也可以是逆序下標(-1,-2,-3...) 終止位置:也是字符串的下標&#xff0c;但是和起始位置下標不…

[Java核心技術(卷Ⅰ)] - 判斷相等

參考 - P184 public boolean equals(Object otherObject) {// a quick test to see if the objects are identicalif (this otherObject) return true;// must return false if the explicit parameter is nullif (otherObject null) return null;// if the classes dont ma…

Oracle 11g DG主庫節點2 ORA-00245: control file backup fail

--節點1報錯 Sun Dec 09 08:29:57 2018Control file backup creation failed: failure to open backup target file /u01/app/oracle/product/11.2.0/db_1/dbs/snapcf_zwdb.ctl.Errors in file /u01/app/oracle/diag/rdbms/zwdb/zwdb2/trace/zwdb2_arc0_167660.trc:ORA-27037: …

hive字符函數

轉載于:https://www.cnblogs.com/ggzhangxiaochao/p/9222732.html

java動態編譯

編譯&#xff0c;一般來說就是將源代碼轉換成機器碼的過程&#xff0c;比如在C語言中中&#xff0c;將C語言源代碼編譯成a.out,&#xff0c;但是在Java中的理解可能有點不同&#xff0c;編譯指的是將java 源代碼轉換成class字節碼的過程&#xff0c;而不是真正的機器碼&#xf…

[c++] - 簡單的冒泡

#include <iostream> using namespace std;int main() {// 利用冒泡排序實現升序序列int arr[9] {4, 2, 8, 0, 5, 7, 1, 3, 9};cout << "排序前: " << endl;for (int i 0; i < 9; i){cout << arr[i] << " ";}cout <…

Python爬蟲之解析網頁

常用的類庫為lxml, BeautifulSoup, re(正則) 以獲取豆瓣電影正在熱映的電影名為例,urlhttps://movie.douban.com/cinema/nowplaying/beijing/ 網頁分析 部分網頁源碼 <ul class"lists"><liid"3878007"class"list-item"data-title"…

騰訊企業郵箱報錯 smtp.exmail.qq.comport 465, isSSL false

一、報錯 "smtp.exmail.qq.com" port 465, isSSL false 通過網上搜索查詢一些資料&#xff0c;推測是郵箱的配置出問題了。 二、修改郵箱配置 1 // 創建屬性2 Properties props new Properties();3 props.setProperty("mail.transport.protocol", "s…

spring與JDK版本對應關系

搭建spring框架得時候要考慮jdk的版本&#xff0c;提供一下參考 JDK 8 中可以使用 Spring Framework 5.x JDK 7 中可以使用 Spring Framework 4.x JDK 6 中可以使用 Spring Framework 4.x JDK 5 中可以使用 Spring Framework 3.x

Markdown預覽功能不可用解決方案

初學者在使用Markdown時也許會遇到這個問題 原因是電腦缺少一個組件&#xff0c;解決方案很簡單&#xff0c;安裝上就好了&#xff0c;以下是鏈接 http://markdownpad.com/download/awesomium_v1.6.6_sdk_win.exe轉載于:https://www.cnblogs.com/j9oker/p/10092829.html

Linux 中yum的配置

1.進入yum的路徑 cd /etc/yum.repos.d 2.將原始的repo文件移入一個新建的backup文件下做備份 mv CentOS* backup 3.在/etc/yum.repos.d下新建一個自己的文件(這里的文件必須以repo結尾); vi zhi.repo 其中&#xff0c;第一行必須是[文件名]的格式  是一個標記 name*** 這是一…

[生態建設] - js判斷小技巧

0、參考 說明: 從幾個易得的點出發,逐步向外擴展延申,保證代碼的可靠性 1、判斷是否為某個類型 // 判斷是否為 null const isNull o > {return o null; };// 判斷是否為 undefined const isUndefined o > {return o undefined; };// 判斷是否為 null or undefined…