強(qiáng)大的 Angular 表單驗(yàn)證

Angular 支持非常強(qiáng)大的內(nèi)置表單驗(yàn)證,maxlength、minlength、required 以及 pattern。使用 Angular 的內(nèi)置表單校驗(yàn)?zāi)軌蛲瓿山^大多數(shù)的業(yè)務(wù)場景的校驗(yàn)需求,但有時(shí)我們還需要實(shí)現(xiàn)更為復(fù)雜的表單校驗(yàn)功能,這時(shí)可以使用 Angular 提供的表單自定義校驗(yàn)(Custom Validator)。下面,我們就來了解一下如何使用 Angular 的自定義表單校驗(yàn)

效果圖:

QQ截圖20170428193800.png
  1. 首先,來創(chuàng)建我們的注冊組件(register),并在模版中顯示一個(gè)簡單的表單
  <h3 class="text-center">注冊</h3>

  <form>

    <div class="form-group">
      <label for="username">用戶名:</label>
      <input type="text" id="username" class="form-control" >
    </div>

  </form>

為了使表單看上去能夠漂亮一些,在 index.html 中引入 bootstrap 樣式文件:

<link rel="stylesheet"  integrity="sha384-BVYiiSIFeK1dGmJRAkycuHAHRg32OmUcww7on3RYdg4Va+PmSTsz/K68vbdEjh4u" crossorigin="anonymous">
  1. 接下來確定我們的驗(yàn)證需求:
    我們希望用戶名只能包含數(shù)字、字母和下劃線,且不能以下劃線開頭
  • 首先為 form 標(biāo)簽添加 formGroup 指令:
<form [formGroup]="registerForm" >
  • 并且為 input 標(biāo)簽添加 formControlName 指令:
<input formControlName="username" type="text" 
        id="username" class="form-control" >
  1. 在代碼中定義驗(yàn)證規(guī)則:
    從內(nèi)置表單模塊中導(dǎo)入以下類:
  import { FormBuilder, FormGroup, Validators } from '@angular/forms';

其中:
1. formBuilder 用來構(gòu)建表單數(shù)據(jù)
2. formGroup 表示表單類型
3. Validators 包含了表單內(nèi)置的驗(yàn)證規(guī)則,如: Validators.required
定義表單屬性

registerForm: FormGroup;

定義表單驗(yàn)證不通過時(shí)每一項(xiàng)顯示的錯(cuò)誤消息(目前我們只有 username )

    formErrors = {
      username: ''
    };

為每一項(xiàng)驗(yàn)證規(guī)則定義驗(yàn)證失敗時(shí)的說明文字(表單控件可能有多條驗(yàn)證規(guī)則,由不通過的驗(yàn)證說明構(gòu)成一條錯(cuò)誤消息)

    validationMessage = {
      'username': {
        'minlength': '用戶名長度最少為3個(gè)字符',
        'maxlength': '用戶名長度最多為10個(gè)字符',
        'required': '請?zhí)顚懹脩裘?
      }
    };

在構(gòu)造函數(shù)中添加 fb 屬性用來構(gòu)建表單

constructor(private fb: FormBuilder) { }

添加構(gòu)建表單的方法

    buildForm(): void {
    // 通過 formBuilder構(gòu)建表單
    this.registerForm = this.fb.group({
      /* 為 username 添加3項(xiàng)驗(yàn)證規(guī)則:
       * 1.必填, 2.最大長度為10, 3.最小長度為3
       * 其中第一個(gè)空字符串參數(shù)為表單的默認(rèn)值
      */
      'username': [ '', [
        Validators.required,
        Validators.maxLength(10),
        Validators.minLength(3)
      ]]
    });

接下來我們添加一個(gè)方法用來更新錯(cuò)誤信息

    onValueChanged(data?: any) {
      // 如果表單不存在則返回
      if (!this.registerForm) return;
      // 獲取當(dāng)前的表單
      const form = this.registerForm;

      // 遍歷錯(cuò)誤消息對象
      for (const field in this.formErrors) {
        // 清空當(dāng)前的錯(cuò)誤消息
        this.formErrors[field] = '';
        // 獲取當(dāng)前表單的控件
        const control = form.get(field);

        // 當(dāng)前表單存在此空間控件 && 此控件沒有被修改 && 此控件驗(yàn)證不通過
        if (control && control.dirty && !control.valid) {
          // 獲取驗(yàn)證不通過的控件名,為了獲取更詳細(xì)的不通過信息
          const messages = this.validationMessage[field];
          // 遍歷當(dāng)前控件的錯(cuò)誤對象,獲取到驗(yàn)證不通過的屬性
          for (const key in control.errors) {
            // 把所有驗(yàn)證不通過項(xiàng)的說明文字拼接成錯(cuò)誤消息
            this.formErrors[field] += messages[key] + '\n';
          }
        }
      }
    }

下面只需要在表單構(gòu)建結(jié)束后初始化錯(cuò)誤消息,并且在每次表單數(shù)據(jù)更改時(shí)更新錯(cuò)誤消息就可以了
在 buildForm 方法中添加如下代碼

    // 每次表單數(shù)據(jù)發(fā)生變化的時(shí)候更新錯(cuò)誤信息
    this.registerForm.valueChanges
      .subscribe(data => this.onValueChanged(data));

    // 初始化錯(cuò)誤信息
    this.onValueChanged();

此時(shí),我們已經(jīng)很好的控制了錯(cuò)誤信息,下面只需要在表單模版中添加錯(cuò)誤信息的顯示就可以了
在 input 標(biāo)簽下方添加如下代碼:

    <div *ngIf="formErrors.username" 
      class="showerr alert alert-danger" >{{ formErrors.username }}</div>

添加如下代碼到表單模版的 css 中:

    form {
      width: 90%;
      max-width: 45em;
      margin: auto;
    }

    .showerr {
      white-space: pre-wrap;
    }

現(xiàn)在我們就可以嘗試運(yùn)行了,在代碼不報(bào)錯(cuò)的情況下已經(jīng)能夠看到非常好的效果了
如果代碼報(bào)錯(cuò)或沒有出現(xiàn)想象中的效果則可以參照本文結(jié)尾的完整代碼進(jìn)行修改

  1. 雖然我們已經(jīng)搭建了整個(gè)布局,但是還沒有實(shí)現(xiàn)我們的最終目的:實(shí)現(xiàn)自定義的表單驗(yàn)證
    接下來我們創(chuàng)建一個(gè)正則驗(yàn)證器,新建文件 validate-register.ts :
    import { ValidatorFn, AbstractControl } from '@angular/forms';

    export function validateRex(type: string, validateRex: RegExp): ValidatorFn {
      return (control: AbstractControl): {[key: string]: any} => {
        // 獲取當(dāng)前控件的內(nèi)容
        const str = control.value;
        // 設(shè)置我們自定義的驗(yàn)證類型
        const res = {};
        res[type] = {str}
        // 如果驗(yàn)證通過則返回 null 否則返回一個(gè)對象(包含我們自定義的屬性)
        return validateRex.test(str) ? null : res;
      }
    }

下面我們在代碼中導(dǎo)入此函數(shù):

import { validateRex } from './validate-register';

修改 validationMessage 屬性為:

    // 為每一項(xiàng)表單驗(yàn)證添加說明文字
    validationMessage = {
      'username': {
        'minlength': '用戶名長度最少為3個(gè)字符',
        'maxlength': '用戶名長度最多為10個(gè)字符',
        'required': '請?zhí)顚懹脩裘?,
        'notdown': '用戶名不能以下劃線開頭',
        'only': '用戶名只能包含數(shù)字、字母、下劃線'
      }
    };

修改 buildForm 方法:

    // 通過 formBuilder構(gòu)建表單
    this.registerForm = this.fb.group({
      /* 為 username 添加 5 項(xiàng)驗(yàn)證規(guī)則:
       * 1.必填, 2.最大長度為10, 3.最小長度為3, 4.不能以下劃線開頭, 5.只能包含數(shù)字、字母、下劃線
       * 其中第一個(gè)空字符串參數(shù)為表單的默認(rèn)值
      */
      'username': [ '', [
        Validators.required,
        Validators.maxLength(10),
        Validators.minLength(3),
        validateRex('notdown', /^(?!_)/),
        validateRex('only', /^[1-9a-zA-Z_]+$/)
      ]]
    });

OK ! 大功告成了,趕緊運(yùn)行代碼嘗試一下吧,我們可以隨時(shí)添加各種驗(yàn)證規(guī)則,只需要修改 validationMessage 屬性和 buildForm 方法即可!
如果添加多個(gè)表單控件的話還需要修改 formErrors,例如添加 password 控件則修改 formErrors 為

formErrors = {
    username: '',
    password: ''
  };

大家可自行嘗試一下!

  1. 完整代碼:
    register.component.html:
    <h3 class="text-center">注冊</h3>

    <form [formGroup]="registerForm" >

      <div class="form-group">
        <label for="username">用戶名:</label>

        <input formControlName="username"
          type="text" id="username" #username
          class="form-control" >
        <div *ngIf="formErrors.username" class="showerr alert alert-danger" >{{ formErrors.username }}</div>
      </div>

    </form>
    ```

    register.component.css:
    ```
    form {
      width: 90%;
      max-width: 45em;
      margin: auto;
    }

    .showerr {
      white-space: pre-wrap;
    }
    ```

    register.component.ts:
    ```
    import { Component, OnInit } from '@angular/core';
    import { FormBuilder, FormGroup, Validators } from '@angular/forms';
    import { validateRex } from './validate-register';

    @Component({
      selector: 'app-register',
      templateUrl: './register.component.html',
      styleUrls: ['./register.component.css']
    })
    export class RegisterComponent implements OnInit {

      // 定義表單
      registerForm: FormGroup;

      // 表單驗(yàn)證不通過時(shí)顯示的錯(cuò)誤消息
      formErrors = {
        username: ''
      };

      // 為每一項(xiàng)表單驗(yàn)證添加說明文字
      validationMessage = {
        'username': {
          'minlength': '用戶名長度最少為3個(gè)字符',
          'maxlength': '用戶名長度最多為10個(gè)字符',
          'required': '請?zhí)顚懹脩裘?,
          'notdown': '用戶名不能以下劃線開頭',
          'only': '用戶名只能包含數(shù)字、字母、下劃線'
        }
      };

      // 添加 fb 屬性,用來創(chuàng)建表單
      constructor(private fb: FormBuilder) { }

      ngOnInit() {
        // 初始化時(shí)構(gòu)建表單
        this.buildForm();
      }

      // 構(gòu)建表單方法
      buildForm(): void {
        // 通過 formBuilder構(gòu)建表單
        this.registerForm = this.fb.group({
          /* 為 username 添加3項(xiàng)驗(yàn)證規(guī)則:
           * 1.必填, 2.最大長度為10, 3.最小長度為3, 4.不能以下劃線開頭, 5.只能包含數(shù)字、字母、下劃線
           * 其中第一個(gè)空字符串參數(shù)為表單的默認(rèn)值
          */
          'username': [ '', [
            Validators.required,
            Validators.maxLength(10),
            Validators.minLength(3),
            validateRex('notdown', /^(?!_)/),
            validateRex('only', /^[1-9a-zA-Z_]+$/)
          ]]
        });

        // 每次表單數(shù)據(jù)發(fā)生變化的時(shí)候更新錯(cuò)誤信息
        this.registerForm.valueChanges
          .subscribe(data => this.onValueChanged(data));

        // 初始化錯(cuò)誤信息
        this.onValueChanged();
      }

      // 每次數(shù)據(jù)發(fā)生改變時(shí)觸發(fā)此方法
      onValueChanged(data?: any) {
        // 如果表單不存在則返回
        if (!this.registerForm) return;
        // 獲取當(dāng)前的表單
        const form = this.registerForm;

        // 遍歷錯(cuò)誤消息對象
        for (const field in this.formErrors) {
          // 清空當(dāng)前的錯(cuò)誤消息
          this.formErrors[field] = '';
          // 獲取當(dāng)前表單的控件
          const control = form.get(field);

          // 當(dāng)前表單存在此空間控件 && 此控件沒有被修改 && 此控件驗(yàn)證不通過
          if (control && control.dirty && !control.valid) {
            // 獲取驗(yàn)證不通過的控件名,為了獲取更詳細(xì)的不通過信息
            const messages = this.validationMessage[field];
            // 遍歷當(dāng)前控件的錯(cuò)誤對象,獲取到驗(yàn)證不通過的屬性
            for (const key in control.errors) {
              // 把所有驗(yàn)證不通過項(xiàng)的說明文字拼接成錯(cuò)誤消息
              this.formErrors[field] += messages[key] + '\n';
            }
          }
        }
      }

    }

validate-register.ts:

    import { ValidatorFn, AbstractControl } from '@angular/forms';

    export function validateRex(type: string, validateRex: RegExp): ValidatorFn {
      return (control: AbstractControl): {[key: string]: any} => {
        // 獲取當(dāng)前控件的內(nèi)容
        const str = control.value;
        // 設(shè)置我們自定義的嚴(yán)重類型
        const res = {};
        res[type] = {str}
        // 如果驗(yàn)證通過則返回 null 否則返回一個(gè)對象(包含我們自定義的屬性)
        return validateRex.test(str) ? null : res;
      }
    }

本文結(jié)束,感謝閱讀!

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,606評論 6 533
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,582評論 3 418
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,540評論 0 376
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,028評論 1 314
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 71,801評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,223評論 1 324
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,294評論 3 442
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,442評論 0 289
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,976評論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 40,800評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 42,996評論 1 369
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,543評論 5 360
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,233評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,662評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,926評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 51,702評論 3 392
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 47,991評論 2 374

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理,服務(wù)發(fā)現(xiàn),斷路器,智...
    卡卡羅2017閱讀 134,785評論 18 139
  • HTML表單 在HTML中,表單是 ... 之間元素的集合,它們允許訪問者輸入文本、選擇選項(xiàng)、操作對象等等,然后將...
    蘭山小亭閱讀 3,434評論 2 14
  • 工廠模式類似于現(xiàn)實(shí)生活中的工廠可以產(chǎn)生大量相似的商品,去做同樣的事情,實(shí)現(xiàn)同樣的效果;這時(shí)候需要使用工廠模式。簡單...
    舟漁行舟閱讀 7,798評論 2 17
  • 細(xì)說 Angular 2+ 的表單(一):模板驅(qū)動型表單 響應(yīng)式表單 響應(yīng)式表單乍一看還是很像模板驅(qū)動型表單的,但...
    接灰的電子產(chǎn)品閱讀 3,388評論 5 22
  • 動態(tài)表單(React Forms)是一種動態(tài)構(gòu)建表單的技術(shù),用于解決有時(shí)候手動編寫和維護(hù)表單所需工作量和時(shí)間會過大...
    阿貍不歌閱讀 7,756評論 0 2