本文使用 Angular 2 搭建一个 TODO MVC 的例子。如果你不知道什么是 TODO MVC 的话,引用官方的一句话是:“ToDoMVC Helping you select an MV* framework”。如果你没听说过什么是 TODO MVC,下图是本文完成后的大概样子。
本文需要做到:
- 使用 Angular Cli 初始化一个 TODO MVC 项目。
- 创建一个 Todo 类。用来描述 todo 的信息。
- 创建一个 TodoService,用来对 todo 信息的增删改查。
- 使用 AppComponent 来向用户展示。
1. 使用 Angular CLI 来初始化你的 Todo Application
使用 Angular CLI 是一个最简单高效的方法来创建你的 Angular 项目。如果你对 Angular 还不熟悉的话,可以看看我之前写的一篇文章,对 Angular 的历史,核心及项目结构有一个了解。
-
安装 Angular CLI
新版的angular-cli已改名成@angular/cli,更符合angular官方的命名规则
>npm install -g @angular/cli
等待安装完成后,若没有报错信息,在cmd中输入一下命令进行验证
>ng -v
_ _ ____ _ ___ / \ _ __ __ _ _ _| | __ _ _ __ / ___| | |_ _| / △ \ | '_ \ / _` | | | | |/ _` | '__| | | | | | | / ___ \| | | | (_| | |_| | | (_| | | | |___| |___ | | /_/ \_\_| |_|\__, |\__,_|_|\__,_|_| \____|_____|___| |___/ @angular/cli: 1.1.0 node: 6.10.3 os: win32 x64
出现以上结果,证明 angular cli 安装成功
-
初始化你的项目
使用 angular-cli 创建第一个工程。ng new 工程名
>ng new todo-app
请耐心等待。 创建新项目需要花费很多时间,安装npm包需要比较长的时间。
-
启动服务
>cd todo-app
>ng serve
正常情况下,启动成功后浏览器会自动打开。或者你可以使用 http://localhost:4200/. 来访问。
2. 创建 Todo Class
-
使用 Angular CLI 创建 Todo 对象
ng g cl Todo
该命令会创建一个 TypeScript 类:
src/app/todo.ts
文件 todo.ts 内容很简单,如下(当然你也可以手动创建):
export class Todo { }
-
修改 todo 对象
export class Todo { id: number; title: string = ''; complete: boolean = false; constructor(values: Object = {}) { Object.assign(this, values); } }
在这个类里面我们定义了三个属性和一个构造函数:
id: number类型, todo 的 id
title: string类型, todo 的标题
complete: boolean类型, 用来表示 todo 是否已经完成
constructor:构造函数,方便我们 new 一个 todo 对象:
2. 创建 TodoService
TodoService 用来管理 Todo 实例的增删改查功能。这里我们会把数据直接保存在内存里。在实际的生产中一般会访问远程接口的 API 等,但这不是本文的重点。
-
创建 TodoService 对象
ng g s Todo
该命令会创建一个两个文件:
src\app\todo.service.spec.ts
src\app\todo.service.ts
todo.service.ts :
import { Injectable } from '@angular/core'; @Injectable() export class TodoService { constructor() { } }
todo.service.spec.ts (测试):
import { TestBed, inject } from '@angular/core/testing'; import { TodoService } from './todo.service'; describe('TodoService', () => { beforeEach(() => { TestBed.configureTestingModule({ providers: [TodoService] }); }); it('should be created', inject([TodoService], (service: TodoService) => { expect(service).toBeTruthy(); })); });
-
修改 todo.service.ts 增加逻辑代码
import { Todo } from './todo'; import { Injectable } from '@angular/core'; @Injectable() export class TodoService { // 增加 todo 时模拟自增id id: number = 0; // 用于在内存里保存 todo 信息 todos: Todo[] = []; constructor() { } add(todo: Todo): Todo { if (!todo.id) { todo.id = ++this.id; } this.todos.push(todo); return todo; } deleteById(id: number): void { this.todos = this.todos .filter(todo => todo.id !== id); } update(todo: Todo): Todo { let t = this.findById(todo.id); if (!t) { return null; } Object.assign(t, todo); return t; } findById(id: number): Todo { return this.todos .filter(todo => todo.id === id) .pop(); } findAll(): Todo[] { return this.todos; } toggleTodoComplete(todo: Todo){ todo.complete = !todo.complete; let u = this.update(todo); return u; } }
-
修改 todo.service.spec.ts 增加测试代码
编写完 todo.service.ts 逻辑代码后,你需要确认你的服务接口是不是正常。这时候需要在todo.service.spec.ts编写 TodoService 的测试代码
import { TestBed, inject } from '@angular/core/testing'; import { TodoService } from './todo.service'; import { Todo } from './todo'; describe('TodoService', () => { beforeEach(() => { TestBed.configureTestingModule({ providers: [TodoService] }); }); it('should be created', inject([TodoService], (service: TodoService) => { expect(service).toBeTruthy(); })); it('get all todos', inject([TodoService], (service: TodoService) => { expect(service.findAll()).toEqual([]); })); it('add todo ', inject([TodoService], (service: TodoService) => { let todo1 = new Todo({ title: 'test', complete: false }); service.add(todo1); expect(service.findById(1)).toEqual(todo1); })); it('toggle todo complete', inject([TodoService], (service: TodoService) => { let todo = new Todo({title: 'test', complete: false}); service.add(todo); let updatedTodo = service.toggleTodoComplete(todo); expect(updatedTodo.complete).toEqual(true); service.toggleTodoComplete(todo); expect(updatedTodo.complete).toEqual(false); })); });
这里只写了部分测试代码,使用一下命令测试:
ng test
运行后浏览器会自动打开
-
看看 todo.service.spec.ts 的结构
ng test 使用 jasmine 来进行测试,关于更多的内容你可以查看 jasmine 官方文档
测试代码会调用全局的 jasmine 函数 describe 。describe 函数包含两个参数,一个描述,另外一个便是测试方法。
beforeEach(() => { TestBed.configureTestingModule({ providers: [TodoService] }); });
TestBed 是 @angular/core/testing 提供的用来配置和创建 Angular Test 模块。我们使用 TestBed.configureTestingModule()方法配置相关信息。
providers 属性告诉测试模块,当我们运行测试用例时需要什么东西。it('get all todos', inject([TodoService], (service: TodoService) => { expect(service.findAll()).toEqual([]); }));
it 类似于 java 里的测试单元,该方法的第一个参数是一个名称,第二个参数是真正的测试方法。TestBad 注册器会为我们注入TodoService,以便我们在我们的测试方法里面访问到这个service。
3. 编写 AppComponent 组件
当我们初始化完这个项目的时候,Angular CLI 已经自动为我们创建了一个主组件 AppComponent ,包含了一下4个文件:
src/app/app.component.css // css 样式文件
src/app/app.component.html // html 结构文件
src/app/app.component.spec.ts // 测试文件
src/app/app.component.ts //逻辑代码
到这里整个
-
修改 app.component.html
-
我们先看一下app.component.html原来的样子:
{{ title }}
-
我们需要修改成我们的需要的结构:
Todos
0"> -
-
如果你对上面的模版语法还不熟悉,没关系,这里对 Angular 2 的模版语法做一个简单的介绍:
- [property]="expression": 属性绑定,会将表达式的值设置给 property 属性。
- [style.color]="expression": 属性绑定,会将表达式的值设置给样式属性 color。
- [class.special]="expression": 属性绑定,如果表达式为真,则为 class 属性添加 special 样式。
- (event)="statement": 事件绑定,当发生 event 事件时执行 statement。
- [(property)]="expression": 双向数据绑定,表达式的值变化会影响 property。property 的值变化也会影响表达式的值
-
-
app.component.html 各个部分解析
-
首先在最上面是一个 input ,用来创建新的todo对象:
-
[(ngModel)]="newTodo.title"
:双向数据绑定,在这里会绑定 newTodo.title 和这个 input 的 value 值。
-
(keyup.enter)="addTodo()"
:事件绑定,当回车键发送 keyup 事件时,执行 addTodo() 这个方法。
-
这里我们还没有在 AppComponent 里定义 newTodo 和 addTodo。 很快我们就会讲到,这里先试着了解 Angular 模版的语法。
-
接下来是一个 section 里面用于定义 todo 列表的展示方式:
0"> -
*ngIf="todos.length > 0"
:angular 内置指令,当 todos.length > 0 时才显示 section 和 section 的子元素。
-
-
接下来是 ul 里套着一个li:
-
*ngFor="let todo of todos"
:angular 内置指令,循环 todos 对象,生成 n 条 li 标签,n = todos.length。并为每条 li 赋值一个 todo 对象。
-
[class.completed]="todo.complete"
:当 todo.complete 等于 true 时,为该标签的 class 添加 completed 。
-
最后是是一个 view 用于展示每个 todo 的详细信息
-
(click)="toggleTodoComplete(todo)"
:当发生点击事件时,会执行 toggleTodoComplete 方法 todo 对象作为参数传入。
-
[checked]="todo.complete"
:绑定 todo.complete 的值到 checked 属性。
-
(click)="removeTodo(todo)"
:当点击 button 按钮时,执行 removeTodo 方法,并将 todo 对象作为参数传入。
-
-
到这里整个 app.component.html 的所有部门都介绍完毕。当然你可能会一头雾水,里面的 todos ,newTodo 还是 addTodo() 等,都是些什么东西。他们要定义在哪里? 又是如何访问的。别着急,下面我们马上就会讲到。
-
编写 app.component.ts
-
我们在先看一下 Angular CLI 默认为我们创建的这个文件里都有什么:
import { Component } from '@angular/core'; @Component({ selector: 'app-root', templateUrl: './app.component.html', styleUrls: ['./app.component.css'] }) export class AppComponent { title = 'app works!'; }
这个文件很简单。首先是一个 @Component 的装饰器,装饰器作用在类上用来告诉 Angular 这个类的一些附加属性。而这些附加的属性就是称之为元数据。
-
selector: 'app-root'
:css3 选择器,当 Angular 在模版里发现了
这个标签后,便会用本组件去渲染。 -
templateUrl: './app.component.html
:模版Url,这里就是我们上面一小节讲到的 app.component.html。用于告诉 Angular 本组件该如何渲染。
-
styleUrls: ['./app.component.css']
:css Url,样式文件的路径。一个完美的 css 能让我们的组件看起来更赏心悦目。
-
-
接下来,我们要为 app.component.ts 注入 TodoService 服务。
import {TodoService} from './todo.service'; @Component({ // ... providers: [TodoService] }) export class AppComponent { // ... constructor(private todoService: TodoService) { } }
-
providers: [TodoService]
:我们在元数据里添加了一个 providers 属性。Angular 会根据 providers 的值会到注入器中查找是否已经有 TodoService 实例,如果没有,会自动帮我们实例化一个 TodoService 对象。并存放在注入器中。
-
constructor(private todoService: TodoService) {}
:这句话便是真正的注入, Angular 注入器会为这个组件注入 TodoService 的实例,并赋值给 todoService 属性。其实这里等价于:
// ... export class AppComponent { private todoService:TodoService; // ... constructor(todoService: TodoService) { this.todoService = todoService; } }
-
-
增加我们的逻辑代码:
import { Component } from '@angular/core'; import { TodoService } from './todo.service'; import { Todo } from './todo'; @Component({ selector: 'app-root', templateUrl: './app.component.html', styleUrls: ['./app.component.css'], providers: [TodoService] }) export class AppComponent { newTodo: Todo = new Todo(); constructor(private todoService:TodoService){} addTodo() { this.todoService.add(this.newTodo); this.newTodo = new Todo(); } toggleTodoComplete(todo) { this.todoService.toggleTodoComplete(todo); } removeTodo(todo) { this.todoService.deleteById(todo.id); } get todos() { return this.todoService.findAll(); } }
你可能不信,增加完上面的代码,我们的应用已经能正常运行了,不过我们还是来看看这个文件的内容:
-
newTodo: Todo = new Todo();
:newTodo 属性用于新增 todo 对象时绑定属性。还记得我们页面的第一个 input 标签吗?
这里对 input 值和 newTodo.title 的值做了双向绑定,只要这两个值中的一个变化,另外一个都会跟着变化。
-
addTodo()
:addTodo() { this.todoService.add(this.newTodo); this.newTodo = new Todo(); }
可能你已经猜到了,当你在页面上按下回车时,那个
(keyup.enter)="addTodo()"
调用方法的实现,就是在这里。addTodo会调用注入的 todoService 的 add 方法。把 newTodo 进行保存。并将 this.newTodo 赋值一个新的 Todo 对象。 -
toggleTodoComplete(todo)
:toggleTodoComplete(todo) { this.todoService.toggleTodoComplete(todo); }
页面
,当你点击这个 checkbox 时,变会调用 toggleTodoComplete 方法并将 todo 对象传入,toggleTodoComplete 使用 todoService 服务的来转变 todo 的 complete 属性。当todo.complete 的值发生变化时,
[checked]
的值也会跟着变化。 -
removeTodo(todo)
:removeTodo(todo) { this.todoService.deleteById(todo.id); }
页面
,当你点击删除按钮是,便会调用 removeTodo 方法,并传入 todo 对象作为方法的参数。而removeTodo 会根据传入的参数调用 todoService 服务的 deleteById 来删除这个 todo 对象。
-
get todos()
:get todos() { return this.todoService.findAll(); }
如果你对 java bean 的定义熟悉的话你应该能猜个大概。这个方法是一个属性方法。相当于定义了一个 todos 属性。 并为这个 todos 属性添加了 get 方法。所以当页面
这个标签获取 todos 的数据时,便会调用
get todos()
方法。而get todos()
会调用 todoService 服务的 findAll 方法,并返回所有的 todos。
-
-
编写我们的样式文件
样式文件并不是本文的重点,这里贴出样式的代码,省的小伙伴自己编写。
-
./src/style.css
: 这个文件用于设置全局css属性:html, body { margin: 0; padding: 0; } button { margin: 0; padding: 0; border: 0; background: none; font-size: 100%; vertical-align: baseline; font-family: inherit; font-weight: inherit; color: inherit; -webkit-appearance: none; appearance: none; -webkit-font-smoothing: antialiased; -moz-font-smoothing: antialiased; font-smoothing: antialiased; } body { font: 14px 'Helvetica Neue', Helvetica, Arial, sans-serif; line-height: 1.4em; background: #f5f5f5; color: #4d4d4d; min-width: 230px; max-width: 550px; margin: 0 auto; -webkit-font-smoothing: antialiased; -moz-font-smoothing: antialiased; font-smoothing: antialiased; font-weight: 300; } button, input[type="checkbox"] { outline: none; } .hidden { display: none; }
-
/src/app/app.component.css
: 编写我们组件的样式.todoapp { background: #fff; margin: 130px 0 40px 0; position: relative; box-shadow: 0 2px 4px 0 rgba(0, 0, 0, 0.2), 0 25px 50px 0 rgba(0, 0, 0, 0.1); } .todoapp input::-webkit-input-placeholder { font-style: italic; font-weight: 300; color: #e6e6e6; } .todoapp input::-moz-placeholder { font-style: italic; font-weight: 300; color: #e6e6e6; } .todoapp input::input-placeholder { font-style: italic; font-weight: 300; color: #e6e6e6; } .todoapp h1 { position: absolute; top: -155px; width: 100%; font-size: 100px; font-weight: 100; text-align: center; color: rgba(175, 47, 47, 0.15); -webkit-text-rendering: optimizeLegibility; -moz-text-rendering: optimizeLegibility; text-rendering: optimizeLegibility; } .new-todo, .edit { position: relative; margin: 0; width: 100%; font-size: 24px; font-family: inherit; font-weight: inherit; line-height: 1.4em; border: 0; outline: none; color: inherit; padding: 6px; border: 1px solid #999; box-shadow: inset 0 -1px 5px 0 rgba(0, 0, 0, 0.2); box-sizing: border-box; -webkit-font-smoothing: antialiased; -moz-font-smoothing: antialiased; font-smoothing: antialiased; } .new-todo { padding: 16px 16px 16px 60px; border: none; background: rgba(0, 0, 0, 0.003); box-shadow: inset 0 -2px 1px rgba(0,0,0,0.03); } .main { position: relative; z-index: 2; border-top: 1px solid #e6e6e6; } label[for='toggle-all'] { display: none; } .toggle-all { position: absolute; top: -55px; left: -12px; width: 60px; height: 34px; text-align: center; border: none; /* Mobile Safari */ } .toggle-all:before { content: '❯'; font-size: 22px; color: #e6e6e6; padding: 10px 27px 10px 27px; } .toggle-all:checked:before { color: #737373; } .todo-list { margin: 0; padding: 0; list-style: none; } .todo-list li { position: relative; font-size: 24px; border-bottom: 1px solid #ededed; } .todo-list li:last-child { border-bottom: none; } .todo-list li.editing { border-bottom: none; padding: 0; } .todo-list li.editing .edit { display: block; width: 506px; padding: 13px 17px 12px 17px; margin: 0 0 0 43px; } .todo-list li.editing .view { display: none; } .todo-list li .toggle { text-align: center; width: 40px; /* auto, since non-WebKit browsers doesn't support input styling */ height: auto; position: absolute; top: 0; bottom: 0; margin: auto 0; border: none; /* Mobile Safari */ -webkit-appearance: none; appearance: none; } .todo-list li .toggle:after { content: url('data:image/svg+xml;utf8,'); } .todo-list li .toggle:checked:after { content: url('data:image/svg+xml;utf8,'); } .todo-list li label { white-space: pre-line; word-break: break-all; padding: 15px 60px 15px 15px; margin-left: 45px; display: block; line-height: 1.2; transition: color 0.4s; } .todo-list li.completed label { color: #d9d9d9; text-decoration: line-through; } .todo-list li .destroy { display: none; position: absolute; top: 0; right: 10px; bottom: 0; width: 40px; height: 40px; margin: auto 0; font-size: 30px; color: #cc9a9a; margin-bottom: 11px; transition: color 0.2s ease-out; } .todo-list li .destroy:hover { color: #af5b5e; } .todo-list li .destroy:after { content: '×'; } .todo-list li:hover .destroy { display: block; } .todo-list li .edit { display: none; } .todo-list li.editing:last-child { margin-bottom: -1px; } .footer { color: #777; padding: 10px 15px; height: 20px; text-align: center; border-top: 1px solid #e6e6e6; } .footer:before { content: ''; position: absolute; right: 0; bottom: 0; left: 0; height: 50px; overflow: hidden; box-shadow: 0 1px 1px rgba(0, 0, 0, 0.2), 0 8px 0 -3px #f6f6f6, 0 9px 1px -3px rgba(0, 0, 0, 0.2), 0 16px 0 -6px #f6f6f6, 0 17px 2px -6px rgba(0, 0, 0, 0.2); } .todo-count { float: left; text-align: left; } .todo-count strong { font-weight: 300; } .filters { margin: 0; padding: 0; list-style: none; position: absolute; right: 0; left: 0; } .filters li { display: inline; } .filters li a { color: inherit; margin: 3px; padding: 3px 7px; text-decoration: none; border: 1px solid transparent; border-radius: 3px; } .filters li a.selected, .filters li a:hover { border-color: rgba(175, 47, 47, 0.1); } .filters li a.selected { border-color: rgba(175, 47, 47, 0.2); } .clear-completed, html .clear-completed:active { float: right; position: relative; line-height: 20px; text-decoration: none; cursor: pointer; } .clear-completed:hover { text-decoration: underline; } .info { margin: 65px auto 0; color: #bfbfbf; font-size: 10px; text-shadow: 0 1px 0 rgba(255, 255, 255, 0.5); text-align: center; } .info p { line-height: 1; } .info a { color: inherit; text-decoration: none; font-weight: 400; } .info a:hover { text-decoration: underline; } /* Hack to remove background from Mobile Safari. Can't use it globally since it destroys checkboxes in Firefox */ @media screen and (-webkit-min-device-pixel-ratio:0) { .toggle-all, .todo-list li .toggle { background: none; } .todo-list li .toggle { height: 40px; } .toggle-all { -webkit-transform: rotate(90deg); transform: rotate(90deg); -webkit-appearance: none; appearance: none; } } @media (max-width: 430px) { .footer { height: 50px; } .filters { bottom: 10px; } }
-
-
4. 总结
到这里我们就使用 Angular 2 创建了一个 TODO MVC 项目。让我们一起回顾下,本文我们都学习到了什么:
- 使用 Angular CLI 创建并初始化一个 Angular 项目。
- 使用 Angular CLI 创建 Todo 对象,TodoService 并实现里面的逻辑代码。
- 使用
ng test
测试我们 TodoService 的逻辑代码 - 对 Angular 模版语法有了一定的了解,属性绑定,事件绑定,双向绑定等。
- 对 Angular 组件有一定的了解,组件装饰器里的元数据与模版和样式文件关联,组件与模版的通过绑定进行交互,组件通过元数据和构造函数注入服务。