建了个群有兴趣的朋友能够加一下 QQ 群:Angular 修仙之路(1)群 - 153742079 (已满),请加 Angular 修仙之路(2)群 - 648681235。html
第一节 - 建立最简单的输入框git
第二节 - 添加简单的验证功能github
第三节 - 显示验证失败的错误信息typescript
第四节 - 建立表单shell
第五节 - ngModelGroup简介npm
第六节 - 表单添加验证状态样式json
第七节 - 表单控件的状态api
第八节 - 使用单选控件服务器
第九节 - 使用多选控件app
本教程的开发环境及开发语言:
安装 Angular CLI (可选)
npm install -g @angular/cli
建立新的项目
ng new PROJECT-NAME
启动本地服务器
cd PROJECT-NAME ng serve
Angular 4 中有两种表单:
Template Driven Forms - 模板驱动式表单 (相似于 AngularJS 1.x 中的表单 )
Reactive Forms - 响应式表单
本文主要介绍 Template Driven Forms (模板驱动式表单) 的基础知识,相关的知识点会以问答的形式进行介绍。
在 Angular 表单中,咱们经过 ngModel
指令来实现双向绑定。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <input type="text" [(ngModel)]="username"> {{username}} `, }) export class AppComponent { username = 'semlinker'; }
目前 Angular 支持的内建 validators
以下:
required - 设置表单控件值是非空的
email - 设置表单控件值的格式是 email
minlength - 设置表单控件值的最小长度
maxlength - 设置表单控件值的最大长度
pattern - 设置表单控件的值需匹配 pattern 对应的模式
接下来咱们来添加最简单的 必填
校验。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <input type="text" required [(ngModel)]="username"> {{username}} `, }) export class AppComponent { username = 'semlinker'; }
在 Angular 中,咱们能够经过 #userName="ngModel"
方式获取 ngModel
对象,而后经过 userName.valid
判断表单控件是否经过验证。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <input type="text" required [(ngModel)]="username" #userName="ngModel"> {{userName.valid}} `, }) export class AppComponent { username = 'semlinker'; }
在 Angular 中,咱们能够经过 #userName="ngModel"
方式获取 ngModel
对象,而后经过该对象的 errors
属性,来获取对应验证规则 (如 required, minlength 等) 的验证状态。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <input type="text" required minlength="3" [(ngModel)]="username" #userName="ngModel"> <hr> <div *ngIf="userName.errors?.required">请您输入用户名</div> <div *ngIf="userName.errors?.minlength"> 用户名的长度必须大于 {{userName.errors?.minlength.requiredLength}},当前的长度为 {{userName.errors?.minlength.actualLength}} </div> `, }) export class AppComponent { username = 'semlinker'; }
在 Angular 中,咱们能够使用熟悉的 <form>
标签来建立表单。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <form> <input type="text" required minlength="3" name="username" [(ngModel)]="username" #userName="ngModel"> <hr> <div *ngIf="userName.errors?.required">请您输入用户名</div> <div *ngIf="userName.errors?.minlength"> 用户名的长度必须大于 {{userName.errors?.minlength.requiredLength}},当前的长度为 {{userName.errors?.minlength.actualLength}} </div> <button type="submit">提交</button> </form> `, }) export class AppComponent { username = 'semlinker'; }
须要注意的是,在使用 <form>
标签后,咱们的 username
输入框,必须添加 name
属性。
在 Angular 中,咱们能够经过 #loginForm="ngForm"
方式获取 ngForm
对象,而后经过 loginForm.value
来获取表单的值。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <form #loginForm="ngForm" (ngSubmit)="onSubmit(loginForm.value)"> <input type="text" required minlength="3" name="username" [(ngModel)]="username" #userName="ngModel"> <hr> <div *ngIf="userName.errors?.required">请您输入用户名</div> <div *ngIf="userName.errors?.minlength"> 用户名的长度必须大于 {{userName.errors?.minlength.requiredLength}},当前的长度为 {{userName.errors?.minlength.actualLength}} </div> <button type="submit">提交</button> {{loginForm.value | json}} </form> `, }) export class AppComponent { username = 'semlinker'; onSubmit(value) { console.dir(value); } }
ngModelGroup 指令是 Angular 表单中提供的另外一特殊指令,能够对表单输入内容进行分组,方便咱们在语义上区分不一样性质的输入。例如联系人的信息包括姓名及住址,如今需对姓名和住址进行精细化信息收集,姓名可精细化成姓和名字,地址可精细化成城市、区、街等。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <form #loginForm="ngForm" (ngSubmit)="onSubmit(loginForm.value)"> <fieldset ngModelGroup="user"> <input type="text" required minlength="3" name="username" [(ngModel)]="username" #userName="ngModel"> <hr> <div *ngIf="userName.errors?.required">请您输入用户名</div> <div *ngIf="userName.errors?.minlength"> 用户名的长度必须大于 {{userName.errors?.minlength.requiredLength}},当前的长度为 {{userName.errors?.minlength.actualLength}} </div> <input type="password" ngModel name="password"> </fieldset> <button type="submit">提交</button> <hr> {{loginForm.value | json}} </form> `, }) export class AppComponent { username = 'semlinker'; onSubmit(value) { console.dir(value); } }
以上代码成功运行后,{{loginForm.value | json}}
的输出结果:
{ "user": { "username": "semlinker", "password": "123" } }
在 Angular 表单中,若验证经过则会在表单控件上添加 ng-valid
类,若验证失败则会在表单控件上添加 ng-invalid
类。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', styles: [` input.ng-invalid { border: 3px solid red; } input.ng-valid { border: 3px solid green; } ` ], template: ` <form #loginForm="ngForm" (ngSubmit)="onSubmit(loginForm.value)"> <fieldset ngModelGroup="user"> <input type="text" required minlength="3" name="username" [(ngModel)]="username" #userName="ngModel"> <hr> <div *ngIf="userName.errors?.required">请您输入用户名</div> <div *ngIf="userName.errors?.minlength"> 用户名的长度必须大于 {{userName.errors?.minlength.requiredLength}},当前的长度为 {{userName.errors?.minlength.actualLength}} </div> <input type="password" required ngModel name="password"> </fieldset> <button type="submit">提交</button> <hr> {{loginForm.value | json}} </form> `, }) export class AppComponent { username = 'semlinker'; onSubmit(value) { console.dir(value); } }
valid
状态外,还包含哪些状态?在 Angular 中表单控件有如下 6 种状态,咱们能够经过 #userName="ngModel"
方式获取 ngModel
对象,进而获取控件的状态信息。具体状态以下:
valid - 表单控件有效
invalid - 表单控件无效
pristine - 表单控件值未改变
dirty - 表单控件值已改变
touched - 表单控件已被访问过
untouched - 表单控件未被访问过
import { Component } from '@angular/core'; @Component({ selector: 'app-root', styles: [` input.ng-invalid { border: 3px solid red; } input.ng-valid { border: 3px solid green; } ` ], template: ` <form #loginForm="ngForm" (ngSubmit)="onSubmit(loginForm.value)"> <fieldset ngModelGroup="user"> <input type="text" required minlength="3" name="username" [(ngModel)]="username" #userName="ngModel"> <hr> <p>Name控件的valid状态:{{userName.valid}} - 表示控件有效</p> <p>Name控件的invalid状态:{{userName.invalid}} - 表示控件无效</p> <p>Name控件的pristine状态:{{userName.pristine}} - 表示控件值未改变</p> <p>Name控件的dirty状态:{{userName.dirty}} - 表示控件值已改变</p> <p>Name控件的touched状态:{{userName.touched}} - 表示控件已被访问过</p> <p>Name控件的untouched状态:{{userName.untouched}} - 表示控件未被访问过</p> <div *ngIf="userName.errors?.required">请您输入用户名</div> <div *ngIf="userName.errors?.minlength"> 用户名的长度必须大于 {{userName.errors?.minlength.requiredLength}},当前的长度为 {{userName.errors?.minlength.actualLength}} </div> <input type="password" required ngModel name="password"> </fieldset> <button type="submit">提交</button> <hr> {{loginForm.value | json}} </form> `, }) export class AppComponent { username = 'semlinker'; onSubmit(value) { console.dir(value); } }
在 Angular 中,咱们经过 <input name="***" type="radio">
方式添加单选控件。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <form #loginForm="ngForm"> Angular版本: <div *ngFor="let version of versions;"> <input [attr.id]="version" name="version" ngModel required [value]="version" type="radio"> <label [attr.for]="version">{{version}}</label> </div> <hr> {{loginForm.value | json}} </form> `, }) export class AppComponent { versions = ['1.x', '2.x', '3.x']; }
在 Angular 中,咱们经过 <select name="***">
方式添加多选控件。
import { Component } from '@angular/core'; @Component({ selector: 'app-root', template: ` <form #loginForm="ngForm"> Angular版本: <select name="version" [ngModel]="versions[0]"> <option *ngFor="let version of versions;" [value]="version"> {{version}} </option> </select> <hr> {{loginForm.value | json}} </form> `, }) export class AppComponent { versions = ['1.x', '2.x', '3.x']; }
import { Component } from '@angular/core'; @Component({ selector: 'app-root', styles: [` select.ng-invalid + label:after { content: '<-- 请选择版本!' } ` ], template: ` <form #loginForm="ngForm"> Angular版本: <div> <select name="version" [ngModel]="version" required> <option *ngFor="let version of versions;" [value]="version"> {{version}} </option> </select> <label></label> </div> <hr> {{loginForm.value | json}} </form> `, }) export class AppComponent { versions = ['','1.x', '2.x', '3.x']; }