当前位置: 首页 > news >正文

如何在 Angular 中为响应式表单创建自定义验证器

简介

Angular 的 @angular/forms 包提供了一个 Validators 类,支持诸如 requiredminLengthmaxLengthpattern 等有用的内置验证器。然而,可能存在需要更复杂或自定义规则进行验证的表单字段。在这种情况下,您可以使用自定义验证器。

在 Angular 中使用响应式表单时,您可以使用函数定义自定义验证器。如果验证器不需要被重复使用,它可以直接存在于组件文件中作为一个函数。否则,如果验证器需要在其他组件中重复使用,它可以存在于一个单独的文件中。

在本教程中,您将构建一个带有可重用自定义验证器的响应式表单,以检查 URL 是否符合特定条件。

先决条件

要完成本教程,您需要:

  • 在本地安装 Node.js,您可以按照《如何安装 Node.js 并创建本地开发环境》进行操作。
  • 一些关于设置 Angular 项目的基本知识。

本教程已经在 Node v15.2.1、npm v6.14.8、@angular/core v11.0.0 和 @angular/forms v11.0.0 下进行了验证。

步骤 1 – 设置项目

为了本教程的目的,您将从使用 @angular/cli 生成的默认 Angular 项目开始构建。

npx @angular/cli new angular-reactive-forms-custom-validtor-example --style=css --routing=false --skip-tests

这将配置一个新的 Angular 项目,其中样式设置为 “CSS”(而不是 “Sass”、“Less” 或 “Stylus”),没有路由,并且跳过了测试。

进入新创建的项目目录:

cd angular-reactive-forms-custom-validator-example

为了使用响应式表单,您将使用 ReactiveFormsModule 而不是 FormsModule

在代码编辑器中打开 app.module.ts 并添加 ReactiveFormsModule

import { BrowserModule } from '@angular/platform-browser';
import { NgModule } from '@angular/core';
import { ReactiveFormsModule } from '@angular/forms';import { AppComponent } from './app.component';@NgModule({declarations: [AppComponent],imports: [BrowserModule,ReactiveFormsModule,],providers: [],bootstrap: [AppComponent]
})
export class AppModule { }

到此为止,您应该已经拥有一个带有 ReactiveFormsModule 的新 Angular 项目。

步骤 2 – 构建自定义验证器

本教程的示例自定义验证器将接受一个 URL 字符串,并确保它以 https 协议开头并以 .io 顶级域名结尾。

首先,在您的终端中,创建一个 shared 目录:

mkdir src/shared

然后,在这个新目录中,创建一个新的 url.validator.ts 文件。在代码编辑器中打开此文件并添加以下代码:

import { AbstractControl } from '@angular/forms';export function ValidateUrl(control: AbstractControl) {if (!control.value.startsWith('https') || !control.value.includes('.io')) {return { invalidUrl: true };}return null;
}

这段代码使用了 AbstractControl 类,它是 FormControlFormGroupFormArray 的基类。这允许访问 FormControl 的值。

这段代码将检查值是否以 https 字符串开头。它还将检查值是否包含 .io 字符串。

如果验证失败,它将返回一个带有错误名称 invalidUrl 和值 true 的对象。

否则,如果验证通过,它将返回 null

到此为止,您的自定义验证器已经准备就绪。

步骤 3 – 使用自定义验证器

接下来,创建一个表单,其中包含 userNamewebsiteUrl

打开 app.component.ts 并用以下代码替换内容:

import { Component, OnInit } from '@angular/core';
import { FormBuilder, FormGroup, Validators } from '@angular/forms';import { ValidateUrl } from '../shared/url.validator';@Component({selector: 'app-root',templateUrl: './app.component.html',styleUrls: ['./app.component.css']
})
export class AppComponent implements OnInit {myForm: FormGroup;constructor(private fb: FormBuilder) {}ngOnInit() {this.myForm = this.fb.group({userName: ['', Validators.required],websiteUrl: ['', [Validators.required, ValidateUrl]]});}saveForm(form: FormGroup) {console.log('Valid?', form.valid); // true or falseconsole.log('Username', form.value.userName);console.log('Website URL', form.value.websiteUrl);}
}

在这段代码中,websiteUrl 表单控件同时使用了内置的 Validators.required 和自定义的 ValidateUrl 验证器。

第四步 – 访问模板中的错误信息

与您的表单交互的用户需要了解哪些数值未通过验证。在组件模板中,您可以使用自定义验证器返回值中定义的键。

打开 app.component.html 并用以下代码替换内容:

<form [formGroup]="myForm" (ngSubmit)="saveForm(myForm)"><div><label>用户名:<input formControlName="userName" placeholder="您的用户名"></label><div *ngIf="(myForm.get('userName').dirty ||myForm.get('userName').touched) &&myForm.get('userName').invalid">请提供您的用户名。</div></div><div><label>网站 URL:<input formControlName="websiteUrl" placeholder="您的网站"></label><div*ngIf="(myForm.get('websiteUrl').dirty ||myForm.get('websiteUrl').touched) &&myForm.get('websiteUrl').invalid">仅接受通过 HTTPS 提供且来自 .io 顶级域的 URL。</div></div>
</form>

此时,您可以编译您的应用程序:

npm start

然后在您的网络浏览器中打开它。您可以与 userNamewebsiteUrl 字段进行交互。确保您的 ValidateUrl 的自定义验证器对于应满足 https.io 条件的值(例如 https://example.io)能够正常工作。

结论

在本文中,您为 Angular 应用程序中的响应式表单创建了一个可重用的自定义验证器。

要了解模板驱动表单和响应式表单中自定义验证器的示例,请参阅 Angular 中的自定义表单验证。

如果您想了解更多关于 Angular 的知识,请查看我们的 Angular 主题页面,了解练习和编程项目。

相关文章:

  • 如何对线上项目Debug
  • C语言希尔排序详解!!!速过
  • redis的缓存穿透,缓存并发,缓存雪崩,缓存问题及解决方案
  • 【MySQL】事务的一致性究竟怎么理解?
  • Spring Boot项目打包及依赖管理-瘦身
  • css中选择器的优先级
  • flink operator 1.7 更换日志框架log4j 到logback
  • 最近火的一键穿衣AI,这款服装设计软件也不赖
  • 【动态规划专栏】专题二:路径问题--------6.地下城游戏
  • 2024-02-20(数位DP)
  • RuntimeError: CUDA out of memory.【多种场景下的解决方案】
  • Delphi语言教程
  • Uipath 读取Word模板实现录用通知书PDF批量生成
  • CSS篇--transform
  • 基于Java SSM框架实现老年人食谱管理系统项目【项目源码+论文说明】
  • 深入了解以太坊
  • 9月CHINA-PUB-OPENDAY技术沙龙——IPHONE
  • CSS实用技巧干货
  • exports和module.exports
  • Java读取Properties文件的六种方法
  • JS变量作用域
  • Laravel5.4 Queues队列学习
  • Promise初体验
  • scala基础语法(二)
  • SpringBoot几种定时任务的实现方式
  • vue学习系列(二)vue-cli
  • yii2权限控制rbac之rule详细讲解
  • 关于Flux,Vuex,Redux的思考
  • 跨域
  • 罗辑思维在全链路压测方面的实践和工作笔记
  • 设计模式走一遍---观察者模式
  • 数据库写操作弃用“SELECT ... FOR UPDATE”解决方案
  • 我是如何设计 Upload 上传组件的
  • 一加3T解锁OEM、刷入TWRP、第三方ROM以及ROOT
  • 源码之下无秘密 ── 做最好的 Netty 源码分析教程
  • 整理一些计算机基础知识!
  • ### Cause: com.mysql.jdbc.exceptions.jdbc4.MySQLTr
  • (02)vite环境变量配置
  • (html转换)StringEscapeUtils类的转义与反转义方法
  • (vue)页面文件上传获取:action地址
  • (附源码)springboot社区居家养老互助服务管理平台 毕业设计 062027
  • (论文阅读30/100)Convolutional Pose Machines
  • (完整代码)R语言中利用SVM-RFE机器学习算法筛选关键因子
  • (原创)boost.property_tree解析xml的帮助类以及中文解析问题的解决
  • (转)机器学习的数学基础(1)--Dirichlet分布
  • (状压dp)uva 10817 Headmaster's Headache
  • .NET Core跨平台微服务学习资源
  • .Net(C#)常用转换byte转uint32、byte转float等
  • .net遍历html中全部的中文,ASP.NET中遍历页面的所有button控件
  • .Net各种迷惑命名解释
  • .net开发引用程序集提示没有强名称的解决办法
  • @selector(..)警告提示
  • [AR Foundation] 人脸检测的流程
  • [C#]winform部署yolov9的onnx模型
  • [C++] cout、wcout无法正常输出中文字符问题的深入调查(1):各种编译器测试