中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

Angular在模板驅動表單中自定義校驗器的方法

發布時間:2020-08-20 08:12:43 來源:腳本之家 閱讀:298 作者:mrr 欄目:web開發

引言

模板驅動表單相比較響應式表單可以少更少的代碼做同樣的事情,可也損失了自由度與更易測試,當然很多人并不在乎啦。

所以我相信很多人在編寫Angular不自由自主去更傾向于模板驅動表單的寫法。

表單最核心的是校驗體驗,在Angular中簡直就是發揮到了極致,比如:required、min、max、pattern 等,這些原本是HTML DOM元素中的表述,而Angular默認實現了一整套的校驗指令,比如:required 對應 RequiredValidator。

然后很多時候我們需要一些特殊的校驗,比如:數據比較、遠程校驗等。那在模板驅動表單風格中我們要如何優雅的實現這樣一個校驗器呢?

一、Angular是如何校驗?

一般在編寫一個手機文本框可能是這樣:

<input [(ngModel)]="user.mobile" #mobile="ngModel" autocomplete="off" type="tel" class="form-control" name="mobile" required maxlength="11">
<div *ngIf="mobile.errors">
  <p *ngIf="mobile.errors.required">手機號必填</p>
  <p *ngIf="mobile.errors.pattern">手機號格式不正確</p>
</div>

以上幾行很友好的實現從必填項、格式進行校驗,而這一切都是依靠 [(ngModel)] 統一采集,得以只需要利用一個模板引用變量訪問到每個校驗指令的錯誤信息。

1、[(ngModel)] 到底做了什么?

在解析這個問題前需要先了解一下 RequiredValidator 是如何定義的。

@Directive({
 providers: [{
   provide: NG_VALIDATORS,
   useExisting: forwardRef(() => RequiredValidator),
   multi: true
  }]
})
export class RequiredValidator {}

只看最核心向 NG_VALIDATORS 標識符注冊一個 RequiredValidator 指令。這樣就可以使 ngModel 指令中注入 NG_VALIDATORS 后就能得到這個指令對象。

ngModel 我把它簡化了一下:

export class NgModel extends NgControl {
  constructor(@Inject(NG_VALIDATORS) validators: Array<Validator|ValidatorFn>) {}
  
  get validator(): ValidatorFn|null {
    // 各種校驗并返回結果
  }
}

有關更多ng_model.ts可以深入閱讀源代碼。

Angular會在每一次表單值變更時,對所有的表單中已經安裝的校驗器進行一次遍歷。

二、編寫一個校驗器

誠如 required 校驗器一樣,依然是把自定義校驗器掛到 NG_VALIDATORS 當中。假如我們希望手機文本框只能輸入 159 開頭的一個校驗器。

定義Directive

@Directive({
  selector: '[user-mobile]',
  exportAs: 'userMobile',
  providers: [{
    provide: NG_VALIDATORS,
    useExisting: forwardRef(() => UserMobileDirective),
    multi: true
  }]
})
export class UserMobileDirective {}

一個非常普通的指令定義方法,只是多了一個將 UserMobileDirective 注冊到 NG_VALIDATORS 標識符當中而已。別問我為什么,一種約定。

export class UserMobileDirective implements Validator {
  validate(c: AbstractControl): { [key: string]: any; } {
    let value: string = c.value || '';
    if (!value.startsWith('159')) {
      return {
        mobile: {
          msg: '手機號必須是159開頭',
          actualValue: value
        }
      };
    }
    return null;
  }
}

只需要實現 Validator 接口的 validate 方法即可。

從 c 中獲取DOM值,當遇到非 159 開頭時,返回一個用于表述消息的對象即可,否則返回一個 null。這個對象會被統一采集在 ngModel.errors 對象下面。故而,只需要在DOM元素加上 user-mobile 指令即可。

<input user-mobile [(ngModel)]="user.mobile" #mobile="ngModel" autocomplete="off" type="tel" class="form-control" name="mobile" id="mobile" required maxlength="11">
<div *ngIf="mobile.errors">
  <p *ngIf="mobile.errors.required">手機號必填</p>
  <p *ngIf="mobile.errors.mobile">{{mobile.errors.mobile.msg}}</p>
</div>

接口還包括一個 registerOnValidatorChange 可選方法,當某些其它外部屬性的變更時,允許重新手動觸發校驗。

三、異步校驗器

如果說用戶手機校驗器需要檢查手機是否為黑名單的情況下,正常黑名單數據都存在遠程當中。這樣情況下需要發送HTTP請求,而這一過程就是異步。

Angular針對這類異步校驗有獨立的另一個標識符,即:NG_ASYNC_VALIDATORS,而其它代碼都是相通的。

@Directive({
  selector: '[user-async]',
  exportAs: 'userAsync',
  providers: [{
    provide: NG_ASYNC_VALIDATORS,
    useExisting: forwardRef(() => UserAsyncDirective),
    multi: true
  }]
})
export class UserAsyncDirective implements Validator {
  validate(c: AbstractControl): Observable<any> {
    return c.valueChanges
        // 去抖
        .debounceTime(300)
        // 抑制重復值
        .distinctUntilChanged()
        // 1、可以使用flatMap進行遠程校驗
        // .flatMap(value => value)
        // 2、本地模擬判斷
        .map((value: string) => {
          if ([ '15900000001', '15900000002' ].includes(value)) {
            return {
              mobile: {
                msg: '手機號為黑名',
                actualValue: value
              }
            }
          }
          return null;
        })
        .first();    
  }
}

除了 NG_ASYNC_VALIDATORS 核心的結構完全沒有變動。

而對于 validate 方法返回的是一個 Observable 類型,利用對 valueChanges 的訂閱可以制作一些像去抖動作。

而最后必須使用 first() 做為結尾,原因每一次校驗,對于結果而言只允許一個。

結論

本章介紹的是如何對模板驅動表單創建自定義校驗器,它相比較響應式表單自定義校驗器略為復雜一些。但是實際運用中,我們不應該只為某個構建表單風格做一種自定義校驗器,應該二者是共存的。

比如上面 159 開頭的示例。更合理的編寫方式應該是將校驗邏輯獨立:

export class MyValidators {
  static checkMobile(value: string): ValidationErrors|null {
    return !value.startsWith('159') ? { mobile: { msg: '手機號必須是159開頭' } } : null;
  }
}
// 校驗器類
export class UserMobileDirective implements Validator {
  validate(c: AbstractControl): { [key: string]: any; } {
    let value: string = c.value || '';
    return MyValidators.checkMobile(value);
  }
}

這樣,同一個校驗器,不管是模板驅動表單還是響應式表單,都能是通用的。

總結

以上所述是小編給大家介紹的Angular在模板驅動表單中自定義校驗器的方法,希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會及時回復大家的。在此也非常感謝大家對億速云網站的支持!

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

郁南县| 大兴区| 柯坪县| 平度市| 大新县| 甘泉县| 桐城市| 衡南县| 平定县| 梁平县| 靖江市| 蒙山县| 广东省| 新巴尔虎左旗| 溧水县| 镇雄县| 泊头市| 马山县| 宁晋县| 静安区| 应城市| 广宁县| 白水县| 武胜县| 沐川县| 喜德县| 子洲县| 临清市| 拉萨市| 信宜市| 湘乡市| 蓬莱市| 淳安县| 涞水县| 蒙自县| 全南县| 大石桥市| 定结县| 涟水县| 武宣县| 遵化市|