天天看点

Angular 2 表单

本章节我们将为大家介绍如何使用组件和模板构建一个 Angular 表单。

利用 Angular 模板,我们可以创建各种类型表单,例如:登录表单,联系人表单,商品详情表单等,而且我们也为这些表单的字段添加数据校验。

接下来我们一步步来实现表单的功能。

导入初始化项目。

完整的项目创建可以参考:Angular 2 TypeScript 环境配置

或者直接下载源代码:点我下载

解压后,修改目录名为angular-forms,修改 angular-forms/package.json 文件中的 <b>"name": "angular-quickstart"</b> 为 <b>"name": "angular-forms"</b>。

完成后,我们执行 <b>cnpm install</b> 来载入依赖包。

以下创建了一个简单的模型类 Site,包含了三个必需字段:id,name,url,一个可选字段:alexa。

在 angular-forms/app 目录下创建 site.ts 文件,代码如下:

export class Site {

constructor(

public id: number,

public name: string,

public url: string,

public alexa?: number

) { }

}

以下代码中,标为 public 的为公有字段,alexa 后添加一个问号(?)表示可选字段。

每个 Angular 表单分为两部分:一个基于 HTML 的模板,和一个基于代码的组件,它用来处理数据和用户交互。

在 angular-forms/app 目录下创建 site-form.component.ts 文件,代码如下:

import { Component } from '@angular/core';

import { Site } from './site';

@Component({

moduleId: module.id,

selector: 'site-form',

templateUrl: 'site-form.component.html'

})

export class SiteFormComponent {

urls = ['www.runoob.com', 'www.google.com',

'www.taobao.com', 'www.facebook.com'];

model = new Site(1, '菜鸟教程', this.urls[0], 10000);

submitted = false;

onSubmit() { this.submitted = true; }

// TODO: 完成后移除

get diagnostic() { return JSON.stringify(this.model); }

实例中导入了 Component 装饰器和 Site 模型。

@Component 选择器 "site-form" 表示我们可以通过一个 <b>&lt;site-form&gt;</b> 标签,把此表单扔进父模板中。

templateUrl 属性指向一个独立的HTML模板文件,名叫 site-form.component.html。

diagnostic 属性用于返回这个模型的JSON形式。

修改 app.module.ts 来定义应用的根模块,模块中指定了引用到的外部及声明属于本模块中的组件,比如 SiteFormComponent。

因为模板驱动的表单有它们自己的模块,所以我们得把 FormsModule 添加到本应用的 imports 数组中,这样我们才能使用表单。

app/app.module.ts 文件代码如下

import { NgModule } from '@angular/core';

import { BrowserModule } from '@angular/platform-browser';

import { FormsModule } from '@angular/forms';

import { AppComponent } from './app.component';

import { SiteFormComponent } from './site-form.component';

@NgModule({

imports: [

BrowserModule,

FormsModule

],

declarations: [

AppComponent,

SiteFormComponent

bootstrap: [ AppComponent ]

export class AppModule { }

修改根组件文件 app.component.ts,将 SiteFormComponent 放在其中。

selector: 'my-app',

template: '&lt;site-form&gt;&lt;/site-form&gt;'

export class AppComponent { }

创建模板文件 site-form.component.html ,代码如下所示:

&lt;div class="container"&gt;

&lt;h1&gt;网站表单&lt;/h1&gt;

&lt;form&gt;

&lt;div class="form-group"&gt;

&lt;label for="name"&gt;网站名&lt;/label&gt;

&lt;input type="text" class="form-control" id="name" required&gt;

&lt;/div&gt;

&lt;label for="alexa"&gt;alexa 排名&lt;/label&gt;

&lt;input type="text" class="form-control" id="alexa"&gt;

&lt;button type="submit" class="btn btn-default"&gt;提交&lt;/button&gt;

&lt;/form&gt;

required 属性设置的该字段为必需字段,如果没有设置则是可选。

在 angular-forms 目录下输入以下命令:

打开 index.html 文件,把以下样式链接添加到 &lt;head&gt; 中:

&lt;link rel="stylesheet" href="node_modules/bootstrap/dist/css/bootstrap.min.css"&gt;

执行 <b>npm start</b> 后,访问:http://localhost:3000/,输出效果如下:

Angular 2 表单

接下来我们使用 ngModel 进行双向数据绑定,通过监听 DOM 事件,来实现更新组件的属性。

修改 app/site-form.component.html ,使用 ngModel 把我们的表单绑定到模型。代码如下所示:

{{diagnostic}}

&lt;input type="text" class="form-control" id="name"

required

[(ngModel)]="model.name" name="name"&gt;

&lt;input type="text" class="form-control" id="alexa"

[(ngModel)]="model.alexa" name="alexa"&gt;

&lt;label for="url"&gt;网站 URL &lt;/label&gt;

&lt;select class="form-control" id="url"

[(ngModel)]="model.url" name="url"&gt;

&lt;option *ngFor="let p of urls" [value]="p"&gt;{{p}}&lt;/option&gt;

&lt;/select&gt;

每一个 input 元素都有一个 id 属性,它被 label 元素的 for 属性用来把标签匹配到对应的 input 。

每一个 input 元素都有一个 name 属性, Angular 的表单模块需要使用它为表单注册控制器。

运行以上实例输出结果如下:

Angular 2 表单

<b>{{diagnostic}} </b> 只是用于测试时候输出数据使用。

我们还可以通过 ngModel 跟踪修改状态与有效性验证,它使用了三个 CSS 类来更新控件,以便反映当前状态。

状态

为 true 时的类

为 false 时的类

控件已经被访问过

<code>ng-touched</code>

<code>ng-untouched</code>

控件值已经变化

<code>ng-dirty</code>

<code>ng-pristine</code>

控件值是有效的

<code>ng-valid</code>

<code>ng-invalid</code>

这样我们就可以添加自定义 CSS 来反应表单的状态。

在 angular-forms 目录下创建 forms.css 文件,代码如下:

.ng-valid[required], .ng-valid.required {

border-left: 5px solid #42A948; /* green */

.ng-invalid:not(form) {

border-left: 5px solid #a94442; /* red */

&lt;link rel="stylesheet" href="forms.css"&gt;

修改 app/site-form.component.html ,代码如下所示:

[(ngModel)]="model.name" name="name"

#name="ngModel" &gt;

&lt;div [hidden]="name.valid || name.pristine"

class="alert alert-danger"&gt;

网站名是必需的

模板中通过把 div 元素的 hidden 属性绑定到 name 控件的属性,我们就可以控制"name"字段错误信息的可见性了。

删除掉 name 字段的数据,显示结果如下所示:

Angular 2 表单

接下来我们创建一个用于添加网站的表单,在 app/site-form.component.html 添加一个按钮:

&lt;button type="button" class="btn btn-default" (click)="newSite()"&gt;添加网站&lt;/button&gt;

将以上按钮事件绑定到组件方法上:

active = true;

newSite() {

this.model = new Site(5, '', '');

this.active = false;

setTimeout(() =&gt; this.active = true, 0);

我们给组件添加一个 active 标记,把它初始化为 true 。当我们添加一个新的网站时,它把 active 标记设置为 false , 然后通过一个快速的 setTimeout 函数迅速把它设置回 true 。

我们可以使用 Angular 的指令 NgSubmit 来提交表单, 并且通过事件绑定机制把它绑定到 SiteFormComponent.submit() 方法上。

&lt;form *ngIf="active" (ngSubmit)="onSubmit()" #siteForm="ngForm"&gt;

我们定义了一个模板引用变量 #siteForm ,并且把它初始化为 "ngForm" 。

这个 siteForm 变量现在引用的是 NgForm 指令,它代表的是表单的整体。

site-form.component.ts 文件完整代码如下:

app/site-form.component.html 完整代码如下:

&lt;div [hidden]="submitted"&gt;

&lt;button type="submit" class="btn btn-default" [disabled]="!siteForm.form.valid"&gt;提交&lt;/button&gt;

&lt;button type="button" class="btn btn-default" (click)="newSite()"&gt;新增网站&lt;/button&gt;

&lt;div [hidden]="!submitted"&gt;

&lt;h2&gt;你提交的信息如下:&lt;/h2&gt;

&lt;div class="row"&gt;

&lt;div class="col-xs-3"&gt;网站名&lt;/div&gt;

&lt;div class="col-xs-9 pull-left"&gt;{{ model.name }}&lt;/div&gt;

&lt;div class="col-xs-3"&gt;网站 alexa 排名&lt;/div&gt;

&lt;div class="col-xs-9 pull-left"&gt;{{ model.alexa }}&lt;/div&gt;

&lt;div class="col-xs-3"&gt;网站 URL &lt;/div&gt;

&lt;div class="col-xs-9 pull-left"&gt;{{ model.url }}&lt;/div&gt;

&lt;br&gt;

&lt;button class="btn btn-default" (click)="submitted=false"&gt;编辑&lt;/button&gt;

模板中我们把 hidden 属性绑定到 SiteFormComponent.submitted 属性上。

主表单从一开始就是可见的,因为 submitted 属性是 false ,当我们提交了这个表单则隐藏,submitted 属性是 true:

最终的目录结构为:

Angular 2 表单

本文所使用的源码可以通过以下方式下载,不包含 node_modules 和 typings 目录。

源代码下载

完整实例演示 GIf 如下:

Angular 2 表单
下一篇: NPM 使用介绍