This commit is contained in:
Zhimin(Rex) YE 2016-05-12 20:33:44 +01:00
commit 9824aca0fe
4 changed files with 255 additions and 252 deletions

View File

@ -28,7 +28,7 @@ include ../_util-fns
lets verify we have the following structure after [Part 1](./toh-pt1.html). lets verify we have the following structure after [Part 1](./toh-pt1.html).
If not, well need to go back to Part 1 and figure out what we missed. If not, well need to go back to Part 1 and figure out what we missed.
在继续《英雄指南》的第二部分之前,我们先检查一下,完成[第一部分](./toh-pt1.html)之后,你是否已经有了如下目录结构。如果没有,你得先回到第一部分,看看错过了哪里。 在继续《英雄指南》的第二部分之前,先检查一下,完成[第一部分](./toh-pt1.html)之后,你是否已经有了如下目录结构。如果没有,你得先回到第一部分,看看错过了哪里。
.filetree .filetree
.file angular2-tour-of-heroes .file angular2-tour-of-heroes

View File

@ -17,7 +17,7 @@ include ../_util-fns
## 我们在哪儿 ## 我们在哪儿
Before we continue with our Tour of Heroes, lets verify we have the following structure. If not, well need to go back and follow the previous chapters. Before we continue with our Tour of Heroes, lets verify we have the following structure. If not, well need to go back and follow the previous chapters.
在继续《英雄指南》之前,我们先检查一下,你是否已经有了如下目录结构。如果没有,你得先回上一章,看看错过了哪里。 在继续《英雄指南》之前,先检查一下,你是否已经有了如下目录结构。如果没有,你得先回上一章,看看错过了哪里。
.filetree .filetree
.file angular2-tour-of-heroes .file angular2-tour-of-heroes

View File

@ -5,21 +5,21 @@ include ../_util-fns
# 服务 # 服务
The Tour of Heroes is evolving and we anticipate adding more components in the near future. The Tour of Heroes is evolving and we anticipate adding more components in the near future.
《英雄指南》继续前行。来,我们准备添加更多的组件。 《英雄指南》继续前行。接下来,我们准备添加更多的组件。
Multiple components will need access to hero data and we don't want to copy and Multiple components will need access to hero data and we don't want to copy and
paste the same code over and over. paste the same code over and over.
Instead, we'll create a single reusable data service and learn to Instead, we'll create a single reusable data service and learn to
inject it in the components that need it. inject it in the components that need it.
将会有更多的组件访问英雄数据,但我们不想一遍又一遍的复制粘贴同样的代码。 会有更多的组件访问英雄数据,但我们不想一遍又一遍的复制粘贴同样的代码。
我们的替代方案是,创建一个单一的、可复用的数据服务,然后学习把它注入到那些想要用它的组件中去。 我们的替代方案是,创建一个单一的、可复用的数据服务,然后学着把它注入到那些想用它的组件中去。
Refactoring data access to a separate service keeps the component lean and focused on supporting the view. Refactoring data access to a separate service keeps the component lean and focused on supporting the view.
It also makes it easier to unit test the component with a mock service. It also makes it easier to unit test the component with a mock service.
重构数据访问代码,把它隔离到一个独立的服务中去,可以让组件保持精简,专注于为视图提供支持。 我们将重构数据访问代码,把它隔离到一个独立的服务中去,让组件尽可能保持精简,专注于为视图提供支持。
如果想通过mock服务来对组件进行单元测试在这种方式下也会变得更容易。 在这种方式下借助mock服务来对组件进行单元测试也会更容易。
Because data services are invariably asynchronous, Because data services are invariably asynchronous,
we'll finish the chapter with a promise-based version of the data service. we'll finish the chapter with a promise-based version of the data service.
@ -37,7 +37,7 @@ include ../_util-fns
Before we continue with our Tour of Heroes, lets verify we have the following structure. Before we continue with our Tour of Heroes, lets verify we have the following structure.
If not, well need to go back and follow the previous chapters. If not, well need to go back and follow the previous chapters.
在继续《英雄指南》之前,我们先检查一下,你是否已经有了如下目录结构。如果没有,你得先回上一章,看看错过了哪里。 在继续《英雄指南》之前,先检查一下,你是否已经有了如下目录结构。如果没有,你得先回上一章,看看错过了哪里。
.filetree .filetree
.file angular2-tour-of-heroes .file angular2-tour-of-heroes
@ -62,7 +62,7 @@ include ../_util-fns
Open a terminal/console window. Open a terminal/console window.
Start the TypeScript compiler, watch for changes, and start our server by entering the command: Start the TypeScript compiler, watch for changes, and start our server by entering the command:
打开terminal/console窗口启动TypeScript编译器它会监视文件变更并启动开发服务器。我们只要敲: 打开terminal/console窗口启动TypeScript编译器它会监视文件变更并启动开发服务器。只要敲
code-example(format="." language="bash"). code-example(format="." language="bash").
npm start npm start
@ -80,10 +80,10 @@ code-example(format="." language="bash").
Soon we'll add a dashboard with the top performing heroes and create a separate view for editing hero details. Soon we'll add a dashboard with the top performing heroes and create a separate view for editing hero details.
All three views need hero data. All three views need hero data.
客户描述了我们这个应用更广阔的图景 客户向我们描绘了本应用更大的目标
他们告诉我们,想要在不同的页面中用多种方式显示英雄。 他们,想要在不同的页面中用多种方式显示英雄。
我们已经能从列表中选择一个英雄了。 现在我们已经能从列表中选择一个英雄了,但这还不够
很快,我们将添加一个仪表盘来表彰绩效最好的英雄,并且创建一个独立视图来编辑英雄的详情。 很快,我们将添加一个仪表盘来表彰绩效最好的英雄,并且创建一个独立视图来编辑英雄的详情。
所有这些视图都需要英雄的数据。 所有这些视图都需要英雄的数据。
At the moment the `AppComponent` defines mock heroes for display. At the moment the `AppComponent` defines mock heroes for display.
@ -91,13 +91,13 @@ code-example(format="." language="bash").
First, defining heroes is not the component's job. First, defining heroes is not the component's job.
Second, we can't easily share that list of heroes with other components and views. Second, we can't easily share that list of heroes with other components and views.
目前,`AppComponent`显示的是自定义的一个mock英雄数据。 目前,`AppComponent`显示的是我们自定义的一个mock英雄数据。
我们的改进点至少有两个1. 定义英雄数据不应该是这个组件的任务。2. 要想把这个英雄列表数据共享给其它组件和视图可不容易。 我们可改进的地方至少有两个:首先,定义英雄的数据不该是该组件的任务。其次,想把这份英雄列表的数据共享给其它组件和视图可不那么容易。
We can refactor this hero data acquisition business to a single service that provides heroes and We can refactor this hero data acquisition business to a single service that provides heroes and
share that service with all components that need heroes. share that service with all components that need heroes.
我们可以把提供英雄数据的业务移交给一个单一的服务,它将提供英雄数据,并且把这个服务在所有需要英雄数据的组件之间共享。 我们可以把提供英雄数据的任务重构为一个单独的服务,它将提供英雄数据,并且把这个服务在所有需要英雄数据的组件之间共享。
### Create the HeroService ### Create the HeroService
### 创建HeroService ### 创建HeroService
@ -111,8 +111,8 @@ code-example(format="." language="bash").
The `SpecialSuperHeroService` would be defined in the `special-super-hero.service.ts` file. The `SpecialSuperHeroService` would be defined in the `special-super-hero.service.ts` file.
我们遵循的文件命名约定是:服务名称的小写形式(基本名),加上`.service`后缀。 我们遵循的文件命名约定是:服务名称的小写形式(基本名),加上`.service`后缀。
如果服务名称包含多个字母,我们把基本名部分拼成中线形式dash-case也被称作烤串形式kebab-case 如果服务名称包含多个单词,我们就把基本名部分写成中线形式dash-case也被称作烤串形式kebab-case
于是`SpecialSuperHeroService`服务应该被定义在`special-super-hero.service.ts`文件中。 比如`SpecialSuperHeroService`服务应该被定义在`special-super-hero.service.ts`文件中。
:marked :marked
We name the class `HeroService` and export it for others to import. We name the class `HeroService` and export it for others to import.
@ -125,49 +125,50 @@ code-example(format="." language="bash").
### 可注入的服务 ### 可注入的服务
Notice that we imported the Angular `Injectable` function and applied that function as an `@Injectable()` decorator. Notice that we imported the Angular `Injectable` function and applied that function as an `@Injectable()` decorator.
注意我们引入了Angular的`Injectable`函数,并通过`@Injectable()`装饰器的形式使用这个函数。 注意我们引入了Angular的`Injectable`函数,并通过`@Injectable()`装饰器使用这个函数。
.callout.is-helpful .callout.is-helpful
:marked :marked
**Don't forget the parentheses!** Neglecting them leads to an error that's difficult to diagnose. **Don't forget the parentheses!** Neglecting them leads to an error that's difficult to diagnose.
**不要忘了写圆括号!** 如果忘了写,就会导致一个非常难以诊断的错误。
**不要忘了写圆括号!**如果忘了写,就会导致一个很难诊断的错误。
:marked :marked
TypeScript sees the `@Injectable()` decorator and emits metadata about our service, TypeScript sees the `@Injectable()` decorator and emits metadata about our service,
metadata that Angular may need to inject other dependencies into this service. metadata that Angular may need to inject other dependencies into this service.
TypeScript看到`@Injectable()`装饰器时,就会记下关于这个服务的元数据。当Angular需要往这个服务中注入其它依赖时就会用到这些元数据。 TypeScript看到`@Injectable()`装饰器时,就会记下本服务的元数据。如果Angular需要往这个服务中注入其它依赖就会使用这些元数据。
The `HeroService` doesn't have any dependencies *at the moment*. Add the decorator anyway. The `HeroService` doesn't have any dependencies *at the moment*. Add the decorator anyway.
It is a "best practice" to apply the `@Injectable()` decorator *from the start* It is a "best practice" to apply the `@Injectable()` decorator *from the start*
both for consistency and for future-proofing. both for consistency and for future-proofing.
此刻,`HeroService`还不能拥有任何依赖。添加上这个装饰器。无论是从提高统一性还是减少变更的目的出发,都应该从一开始就加上`@Injectable()`装饰器,这是最佳实践 此刻,`HeroService`还没有任何依赖,但我们还是得加上这个装饰器。作为一项最佳实践,无论是出于提高统一性还是减少变更的目的,都应该从一开始就加上`@Injectable()`装饰器。
:marked :marked
### Getting Heroes ### Getting Heroes
### 获取英雄 ### 获取英雄
Add a `getHeroes` method stub. Add a `getHeroes` method stub.
添加一个`getHeros`桩方法。 添加一个名叫`getHeros`桩方法。
+makeExample('toh-4/ts/app/hero.service.1.ts', 'getHeroes-stub', 'hero.service.ts (getHeroes桩方法)')(format=".") +makeExample('toh-4/ts/app/hero.service.1.ts', 'getHeroes-stub', 'hero.service.ts (getHeroes桩方法)')(format=".")
:marked :marked
We're holding back on the implementation for a moment to make an important point. We're holding back on the implementation for a moment to make an important point.
在这个实现上暂停一会儿,我们来讲一个重点。 在这个实现上暂停一下,我们先来讲一个重点。
The consumer of our service doesn't know how the service gets the data. The consumer of our service doesn't know how the service gets the data.
Our `HeroService` could get `Hero` data from anywhere. Our `HeroService` could get `Hero` data from anywhere.
It could get the data from a web service or local storage It could get the data from a web service or local storage
or from a mock data source. or from a mock data source.
我们这个服务的消费者并不知道它具体如何获取数据。 消费者并不知道本服务会如何获取数据。
我们的`HeroService`服务可以从任何地方获英雄的数据。 我们的`HeroService`服务可以从任何地方获英雄的数据。
它可能从网络服务器获取,可能从浏览器的局部存储区获取,也可能是写在源码中的mock数据。 它可以从网络服务器获取,可以从浏览器的局部存储区获取,也可以是直接写在源码中的mock数据。
That's the beauty of removing data access from the component. That's the beauty of removing data access from the component.
We can change our minds about the implementation as often as we like, We can change our minds about the implementation as often as we like,
for whatever reason, without touching any of the components that need heroes. for whatever reason, without touching any of the components that need heroes.
我们从组件中移除了数据访问代码,干得漂亮。 我们从组件中成功移除了数据访问代码,干得漂亮。
这下子,我们可以随时改变数据访问的实现方式了。 这下子,我们可以随时改变数据访问的实现方式了。
### Mock Heroes ### Mock Heroes
@ -175,23 +176,23 @@ code-example(format="." language="bash").
We already have mock `Hero` data sitting in the `AppComponent`. It doesn't belong there. It doesn't belong *here* either. We already have mock `Hero` data sitting in the `AppComponent`. It doesn't belong there. It doesn't belong *here* either.
We'll move the mock data to its own file. We'll move the mock data to its own file.
我们已经在`AppComponent`组件中实现过mock的英雄数据了。它不该在那个文件也不该在这个文件中我们要把mock数据移到它自己的文件中去。 我们曾在`AppComponent`组件中写过mock版的英雄数据。它不该在那里但也不该在*这里*!我们得把mock数据移到它自己的文件中去。
Cut the `HEROES` array from `app.component.ts` and paste it to a new file in the `app` folder named `mock-heroes.ts`. Cut the `HEROES` array from `app.component.ts` and paste it to a new file in the `app` folder named `mock-heroes.ts`.
We copy the `import {Hero} ...` statement as well because the heroes array uses the `Hero` class. We copy the `import {Hero} ...` statement as well because the heroes array uses the `Hero` class.
从`app.component.ts`文件中剪切`HEROS`数组,并把它粘贴到`app`目录下一个名叫`mock-heroes.ts`的文件中。 从`app.component.ts`文件中剪切`HEROS`数组,并把它粘贴到`app`目录下一个名叫`mock-heroes.ts`的文件中。
我们还要把`import {Hero}...`语句拷贝过来,因为我们的英雄数组用到了`Hero`类。 我们还要把`import {Hero}...`语句拷贝过来,因为我们的英雄数组用到了`Hero`类。
+makeExample('toh-4/ts/app/mock-heroes.ts', null, 'mock-heroes.ts (英雄数组)') +makeExample('toh-4/ts/app/mock-heroes.ts', null, 'mock-heroes.ts (英雄数组)')
:marked :marked
We export the `HEROES` constant so we can import it elsewhere — such as our `HeroService`. We export the `HEROES` constant so we can import it elsewhere — such as our `HeroService`.
我们导出`HEROES`常量,以便在其它地方导入它 —— 比如`HeroService`服务。 我们导出`HEROES`常量,以便在其它地方导入它 —— 比如`HeroService`服务。
Meanwhile, back in `app.component.ts` where we cut away the `HEROES` array, Meanwhile, back in `app.component.ts` where we cut away the `HEROES` array,
we leave behind an uninitialized `heroes` property: we leave behind an uninitialized `heroes` property:
与此同时,回到我们刚刚剪切出`HEROES`数组的`app.component.ts`文件,我们留下了一个尚未初始化的`heroes`属性: 同时,回到刚剪切出`HEROES`数组的`app.component.ts`文件,我们留下了一个尚未初始化的`heroes`属性:
+makeExample('toh-4/ts/app/app.component.1.ts', 'heroes-prop', 'app.component.ts (heroes属性)')(format=".") +makeExample('toh-4/ts/app/app.component.1.ts', 'heroes-prop', 'app.component.ts (heroes属性)')(format=".")
:marked :marked
### Return Mocked Heroes ### Return Mocked Heroes
@ -199,8 +200,8 @@ code-example(format="." language="bash").
Back in the `HeroService` we import the mock `HEROES` and return it from the `getHeroes` method. Back in the `HeroService` we import the mock `HEROES` and return it from the `getHeroes` method.
Our `HeroService` looks like this: Our `HeroService` looks like this:
回到`HeroService`,我们导入`HEROES`常量,并在`getHeroes`方法中返回它。 回到`HeroService`,我们导入`HEROES`常量,并在`getHeroes`方法中返回它。
我们的`HeroService`服务看起来像这样: 我们的`HeroService`服务现在看起来是这样:
+makeExample('toh-4/ts/app/hero.service.1.ts', null, 'hero.service.ts')(format=".") +makeExample('toh-4/ts/app/hero.service.1.ts', null, 'hero.service.ts')(format=".")
:marked :marked
### Use the Hero Service ### Use the Hero Service
@ -211,42 +212,42 @@ code-example(format="." language="bash").
We begin, as usual, by importing the thing we want to use, the `HeroService`. We begin, as usual, by importing the thing we want to use, the `HeroService`.
通常,我们会从导入要用的东西开始,比如`HeroService`。 通常,我们会从导入要用的东西开始,比如`HeroService`。
+makeExample('toh-4/ts/app/app.component.ts', 'hero-service-import', 'app.component.ts (导入HeroService)') +makeExample('toh-4/ts/app/app.component.ts', 'hero-service-import', 'app.component.ts (导入HeroService)')
:marked :marked
Importing the service allows us to *reference* it in our code. Importing the service allows us to *reference* it in our code.
How should the `AppComponent` acquire a runtime concrete `HeroService` instance? How should the `AppComponent` acquire a runtime concrete `HeroService` instance?
导入服务让我们可以在代码中引用它。 导入这个服务让我们可以在代码中引用它。
`AppComponent`该如何在运行中获得一个具体的`HeroService`实例呢? `AppComponent`该如何在运行中获得一个具体的`HeroService`实例呢?
### Do we *new* the *HeroService*? No way! ### Do we *new* the *HeroService*? No way!
### 我们要自己 *new* 出这个 *HeroService* 吗?没门! ### 我们要自己 *new* 出这个 *HeroService* 吗?没门!
We could create a new instance of the `HeroService` with "new" like this: We could create a new instance of the `HeroService` with "new" like this:
我们可以使用`new`关键字来创建`HeroService`的实例,就像这样: 固然,我们可以使用`new`关键字来创建`HeroService`的实例,就像这样:
+makeExample('toh-4/ts/app/app.component.1.ts', 'new-service')(format=".") +makeExample('toh-4/ts/app/app.component.1.ts', 'new-service')(format=".")
:marked :marked
That's a bad idea for several reasons including That's a bad idea for several reasons including
但这不是个好主意,理由很多,比如: 但这不是个好主意,有很多理由,比如:
* Our component has to know how to create a `HeroService`. * Our component has to know how to create a `HeroService`.
If we ever change the `HeroService` constructor, If we ever change the `HeroService` constructor,
we'll have to find every place we create the service and fix it. we'll have to find every place we create the service and fix it.
Running around patching code is error prone and adds to the test burden. Running around patching code is error prone and adds to the test burden.
* 我们的组件不得不了解如何取创建`HeroService`。 * 我们的组件将不得不懂得该如何创建`HeroService`。
如果有一天我们修改了`HeroService`的构造函数,我们不得不找出我们创建此服务的每一处代码,并修改它。 如果有一天我们修改了`HeroService`的构造函数,我们不得不找出创建此服务的每一处代码,并修改它。
给代码打补丁的行为容易导致错误,而且增加了测试的负担。 而给代码打补丁的行为容易导致错误,并增加了测试的负担。
* We create a new service each time we use "new". * We create a new service each time we use "new".
What if the service should cache heroes and share that cache with others? What if the service should cache heroes and share that cache with others?
We couldn't do that. We couldn't do that.
* 我们每次使用`new`都会创建一个新的服务。 * 我们每次使用`new`都会创建一个新的服务实例
如果这个服务需要缓存英雄列表,并且把这个缓存共享给别人,怎么办? 如果这个服务需要缓存英雄列表,并把这个缓存共享给别人呢?怎么办?
没办法,做不到。 没办法,做不到。
* We're locking the `AppComponent` into a specific implementation of the `HeroService`. * We're locking the `AppComponent` into a specific implementation of the `HeroService`.
It will be hard to switch implementations for different scenarios. It will be hard to switch implementations for different scenarios.
@ -254,24 +255,24 @@ code-example(format="." language="bash").
Will we need different mocked versions under test? Will we need different mocked versions under test?
Not easy. Not easy.
* 我们把`AppComponent`锁定到了`HeroService`的一个特定实现中。 * 我们把`AppComponent`锁死在`HeroService`的一个特定实现中。
我们很难在别的场景中把它更换为别的实现。 我们很难在别的场景中把它换成别的实现。
比如,能离线操作吗?能在测试时使用不同的模拟版本吗?这可不容易。 比如,能离线操作吗?能在测试时使用不同的模拟版本吗?这可不容易。
*What if ... what if ... Hey, we've got work to do!* *What if ... what if ... Hey, we've got work to do!*
* 如果……如果……嘿!我们有得忙了! * 如果……如果……嘿!这下我们有得忙了!
We get it. Really we do. We get it. Really we do.
But it is so ridiculously easy to avoid these problems that there is no excuse for doing it wrong. But it is so ridiculously easy to avoid these problems that there is no excuse for doing it wrong.
我们有办法了,真的!这个办法真是简单得不可思议,它能解决这些问题,你再也没有借口犯错误了。 有办法了,真的!这个办法真是简单得不可思议,它能解决这些问题,你再也没有犯错误的接口了。
### Inject the *HeroService* ### Inject the *HeroService*
### 注入 *HeroService* ### 注入 *HeroService*
Two lines replace the one line of *new*: Two lines replace the one line of *new*:
用两行代码代替用`new`时的一行: 两行代码代替用`new`时的一行:
1. we add a constructor. 1. we add a constructor.
1. 添加一个构造函数 1. 添加一个构造函数
1. we add to the component's `providers` metadata 1. we add to the component's `providers` metadata
@ -285,13 +286,13 @@ code-example(format="." language="bash").
The constructor itself does nothing. The parameter simultaneously The constructor itself does nothing. The parameter simultaneously
defines a private `_heroService` property and identifies it as a `HeroService` injection site. defines a private `_heroService` property and identifies it as a `HeroService` injection site.
构造函数自己什么也不用做,它的参数同时定义了一个私有的`_heroService`属性,并把它标记为注入`HeroService`的靶点。 构造函数自己什么也不用做,它在参数中定义了一个私有的`_heroService`属性,并把它标记为注入`HeroService`的靶点。
.l-sub-section .l-sub-section
:marked :marked
We prefix private variables with an underscore (_) to warn readers of our code We prefix private variables with an underscore (_) to warn readers of our code
that this variable is not part of the component's public API. that this variable is not part of the component's public API.
我们给私有变量添加下划线_前缀用以警告阅读我们代码的人这个变量不是组件的公开API的一部分。 我们给私有变量添加下划线_前缀用以警告这些代码的读者这个变量不是组件的公开API的一部分。
:marked :marked
Now Angular will know to supply an instance of the `HeroService` when it creates a new `AppComponent`. Now Angular will know to supply an instance of the `HeroService` when it creates a new `AppComponent`.
@ -304,7 +305,7 @@ code-example(format="." language="bash").
Angular得想办法获得这个实例。这就是Angular *依赖注入器Dependency Injector* 扮演的角色。 Angular得想办法获得这个实例。这就是Angular *依赖注入器Dependency Injector* 扮演的角色。
这个 **注入器** 有一个包括以前创建过的所有服务的 **容器** 。 这个 **注入器** 有一个包括以前创建过的所有服务的 **容器** 。
它既可以从容器中查找并返回一个已存在的`HeroService`实例,可以创建一个新的实例把它添加到容器中然后把它返回给Angular。 它既可以从容器中查找并返回一个已存在的`HeroService`实例,可以创建一个新的实例把它添加到容器中然后把它返回给Angular。
.l-sub-section .l-sub-section
:marked :marked
@ -316,18 +317,18 @@ code-example(format="." language="bash").
The *injector* does not know yet how to create a `HeroService`. The *injector* does not know yet how to create a `HeroService`.
If we ran our code now, Angular would fail with an error: If we ran our code now, Angular would fail with an error:
*注入器* 还不知道该如何创建`HeroService`。 *注入器*还不知道该如何创建`HeroService`。
如果现在运行我们的代码Angular会失败并报错 如果现在运行我们的代码Angular会失败,并报错:
code-example(format="." language="html"). code-example(format="." language="html").
EXCEPTION: No provider for HeroService! (AppComponent -> HeroService) EXCEPTION: No provider for HeroService! (AppComponent -> HeroService)
EXCEPTION: No provider for HeroService! (AppComponent -> HeroService) —— 异常没有HeroService的供应商(provider)AppComponent -> HeroService EXCEPTION: No provider for HeroService! (AppComponent -> HeroService) —— 异常没有HeroService的提供者(provider)AppComponent -> HeroService
:marked :marked
We have to teach the *injector* how to make a `HeroService` by registering a `HeroService` **provider**. We have to teach the *injector* how to make a `HeroService` by registering a `HeroService` **provider**.
Do that by adding the following `providers` array property to the bottom of the component metadata Do that by adding the following `providers` array property to the bottom of the component metadata
in the `@Component` call. in the `@Component` call.
我们还得注册一个`HeroService` **供应商** ,以便教 *注入器* 如何创建`HeroService`。 我们还得注册一个`HeroService`**Provider**,来告诉*注入器*如何创建`HeroService`。
要做到这一点,我们应该在`@Component`组件的元数据底部添加`providers`数组属性如下: 要做到这一点,我们应该在`@Component`组件的元数据底部添加`providers`数组属性如下:
+makeExample('toh-4/ts/app/app.component.1.ts', 'providers', 'app.component.ts (提供HeroService)') +makeExample('toh-4/ts/app/app.component.1.ts', 'providers', 'app.component.ts (提供HeroService)')
:marked :marked
@ -335,7 +336,7 @@ code-example(format="." language="html").
The `AppComponent` can use that service to get heroes and so can every child component of its component tree. The `AppComponent` can use that service to get heroes and so can every child component of its component tree.
`providers`数组告诉Angular在创建一个新的`AppComponent`时,也要创建一个`HeroService`的新鲜实例。 `providers`数组告诉Angular在创建一个新的`AppComponent`时,也要创建一个`HeroService`的新鲜实例。
`AppComponent`可以通过这个服务来获取英雄列表,在它组件树中的每一个子组件也可以这样做。 `AppComponent`可以用此服务来获取英雄列表,它组件树中的每一个子组件也可以这么做。
<a id="child-component"></a> <a id="child-component"></a>
.l-sub-section .l-sub-section
:marked :marked
@ -350,7 +351,7 @@ code-example(format="." language="html").
If the `HeroDetailComponent` needed its parent component's `HeroService`, If the `HeroDetailComponent` needed its parent component's `HeroService`,
it would ask Angular to inject the service into its constructor which would look just like the one for `AppComponent`: it would ask Angular to inject the service into its constructor which would look just like the one for `AppComponent`:
如果`HeroDetailComponent`需要访问来自它父组件的`HeroService`服务它可以要求Angular把这个服务注入到自己的构造函数中 —— 就像`AppComponent`中的做法一样。 如果`HeroDetailComponent`需要访问来自它父组件的`HeroService`服务它可以要求Angular把这个服务注入到自己的构造函数中 —— 就像`AppComponent`中的做法一样。
+makeExample('toh-4/ts/app/app.component.1.ts', 'ctor', 'hero-detail.component.ts (构造函数)') +makeExample('toh-4/ts/app/app.component.1.ts', 'ctor', 'hero-detail.component.ts (构造函数)')
:marked :marked
@ -361,17 +362,17 @@ code-example(format="." language="html").
The `AppComponent` is the top level component of our application. The `AppComponent` is the top level component of our application.
There should be only one instance of that component and only one instance of the `HeroService` in our entire app. There should be only one instance of that component and only one instance of the `HeroService` in our entire app.
`AppComponent`是我们应用的顶层组件。在我们的整个应用中,应该有唯一的一个顶层组件,应该唯一的一个`HeroService`实例。 `AppComponent`是我们应用的顶层组件。在我们的整个应用中,应该有唯一的一个顶层组件,应该唯一的一个`HeroService`实例。
:marked :marked
### *getHeroes* in the *AppComponent* ### *getHeroes* in the *AppComponent*
### *AppComponent* 中的 *getHeroes* ### *AppComponent* 中的 *getHeroes*
We've got the service in a `_heroService` private variable. Let's use it. We've got the service in a `_heroService` private variable. Let's use it.
我们已经获得了此服务,并把它存入了私有变量`_heroService`中。我们这就开始使用它。 我们已经获得了此服务,并把它存入了私有变量`_heroService`中。我们这就开始使用它。
We pause to think. We can call the service and get the data in one line. We pause to think. We can call the service and get the data in one line.
停下来想一想。我们可以在同一行中调用服务并获得数据。 停下来想一想。我们可以在同一行内调用此服务并获得数据。
+makeExample('toh-4/ts/app/app.component.1.ts', 'get-heroes')(format=".") +makeExample('toh-4/ts/app/app.component.1.ts', 'get-heroes')(format=".")
:marked :marked
We don't really need a dedicated method to wrap one line. We write it anyway: We don't really need a dedicated method to wrap one line. We write it anyway:
@ -392,29 +393,29 @@ code-example(format="." language="html").
Years of experience and bitter tears have taught us to keep complex logic out of the constructor, Years of experience and bitter tears have taught us to keep complex logic out of the constructor,
especially anything that might call a server as a data access method is sure to do. especially anything that might call a server as a data access method is sure to do.
多年的经验和惨痛的教训教育我们,把复杂的逻辑扔到构造函数外面去,特别是那些需要从服务器获取数据的逻辑更是如此。 多年的经验和惨痛的教训教育我们,应该把复杂的逻辑扔到构造函数外面去,特别是那些需要从服务器获取数据的逻辑更是如此。
The constructor is for simple initializations like wiring constructor parameters to properties. The constructor is for simple initializations like wiring constructor parameters to properties.
It's not for heavy lifting. We should be able to create a component in a test and not worry that it It's not for heavy lifting. We should be able to create a component in a test and not worry that it
might do real work &mdash; like calling a server! &mdash; before we tell it to do so. might do real work &mdash; like calling a server! &mdash; before we tell it to do so.
构造函数是为了初始化而设计的,比如把构造函数的参数赋值给属性。 构造函数是为了简单的初始化工作而设计的,比如把构造函数的参数赋值给属性。
它的负担不应该过于沉重。我们应该能在测试中创建一个测试,而不用担心它会完成实际的工作 —— 比如和服务器通讯,我们将来才会主动要求它做这些。 它的负担不应该过于沉重。我们应该能在测试中创建一个组件,而不用担心它会做实际的工作 —— 比如和服务器通讯,直到我们主动要求它做这些。
If not the constructor, something has to call `getHeroes`. If not the constructor, something has to call `getHeroes`.
如果不在构造函数中,总有地方调用`getHeroes`吧。 如果不在构造函数中,总有地方调用`getHeroes`吧。
Angular will call it if we implement the Angular **ngOnInit** *Lifecycle Hook*. Angular will call it if we implement the Angular **ngOnInit** *Lifecycle Hook*.
Angular offers a number of interfaces for tapping into critical moments in the component lifecycle: Angular offers a number of interfaces for tapping into critical moments in the component lifecycle:
at creation, after each change, and at its eventual destruction. at creation, after each change, and at its eventual destruction.
只要我们实现了Angular的 **ngOnInit** *生命周期钩子* Angular就会调用这个钩子。 这也不难。只要我们实现了Angular的 **ngOnInit** *生命周期钩子* Angular就会主动调用这个钩子。
Angular提供了一些接口用来介入组件生命周期的几个关键时间点刚创建时、每次变化时以及最终被销毁时。 Angular提供了一些接口用来介入组件生命周期的几个关键时间点刚创建时、每次变化时以及最终被销毁时。
Each interface has a single method. When the component implements that method, Angular calls it at the appropriate time. Each interface has a single method. When the component implements that method, Angular calls it at the appropriate time.
每个接口都是一个单独的方法。只要组件实现了这个方法Angular就会在合适的时间调用它。 每个接口都有唯一的一个方法。只要组件实现了这个方法Angular就会在合适的时机调用它。
.l-sub-section .l-sub-section
:marked :marked
Learn more about lifecycle hooks in the [Lifecycle Hooks](../guide/lifecycle-hooks.html) chapter. Learn more about lifecycle hooks in the [Lifecycle Hooks](../guide/lifecycle-hooks.html) chapter.
@ -429,7 +430,7 @@ code-example(format="." language="html").
We write an `ngOnInit` method with our initialization logic inside and leave it to Angular to call it We write an `ngOnInit` method with our initialization logic inside and leave it to Angular to call it
at the right time. In our case, we initialize by calling `getHeroes`. at the right time. In our case, we initialize by calling `getHeroes`.
我们写下带有初始化逻辑的`ngOnInit`方法然后留给Angular供其在正确的时调用。在这个例子中,我们通过调用`getHeroes`来完成初始化。 我们写下带有初始化逻辑的`ngOnInit`方法然后留给Angular供其在正确的时调用。在这个例子中,我们通过调用`getHeroes`来完成初始化。
+makeExample('toh-4/ts/app/app.component.1.ts', 'ng-on-init', 'app.component.ts (OnInit协议)')(format=".") +makeExample('toh-4/ts/app/app.component.1.ts', 'ng-on-init', 'app.component.ts (OnInit协议)')(format=".")
:marked :marked
Our application should be running as expected, showing a list of heroes and a hero detail view Our application should be running as expected, showing a list of heroes and a hero detail view
@ -439,7 +440,7 @@ code-example(format="." language="html").
We're getting closer. But something isn't quite right. We're getting closer. But something isn't quite right.
我们就快完成了,但还有点事不太对劲。 我们就快完成了,但还有点事不太对劲
## Async Services and Promises ## Async Services and Promises
## 异步服务与承诺Promise ## 异步服务与承诺Promise
@ -451,11 +452,11 @@ code-example(format="." language="html").
:marked :marked
Ask for heroes and they are there in the returned result. Ask for heroes and they are there in the returned result.
请求英雄数据,并直接在结果中返回它。 请求英雄数据,并直接在结果中返回它。
Someday we're going to get heroes from a remote server. We dont call http yet, but we aspire to in later chapters. Someday we're going to get heroes from a remote server. We dont call http yet, but we aspire to in later chapters.
将来我们会从远端服务器上获取英雄数据。我们还没调用http是我们希望在将来的章节中这么做。 将来我们会从远端服务器上获取英雄数据。我们还没调用http在未来的章节中我们会希望这么做。
When we do, we'll have to wait for the server to respond and we won't be able to block the UI while we wait, When we do, we'll have to wait for the server to respond and we won't be able to block the UI while we wait,
even if we want to (which we shouldn't) because the browser won't block. even if we want to (which we shouldn't) because the browser won't block.
@ -468,10 +469,10 @@ code-example(format="." language="html").
We'll use *promises*. We'll use *promises*.
我们将使用 *承诺* 。 我们将使用 *承诺Promise* 。
### The Hero Service makes a promise ### The Hero Service makes a promise
### `HeroService`生成一个承诺 ### `HeroService`生成一个承诺
A **promise** is ... well it's a promise to call us back later when the results are ready. A **promise** is ... well it's a promise to call us back later when the results are ready.
We ask an asynchronous service to do some work and give it a callback function. We ask an asynchronous service to do some work and give it a callback function.
@ -479,12 +480,12 @@ code-example(format="." language="html").
**承诺** 就是 …… 好吧,它就是一个承诺 —— 在有了结果时,它承诺会回调我们。 **承诺** 就是 …… 好吧,它就是一个承诺 —— 在有了结果时,它承诺会回调我们。
我们请求一个异步服务去做点什么,然后给它一个回调函数。 我们请求一个异步服务去做点什么,然后给它一个回调函数。
它会去做(无论用哪种方式),完成的时候,它调用我们的回调函数,并通过参数把工作成果或者错误信息传给我们。 它会去做(无论用哪种方式),一旦完成,它就会调用我们的回调函数,并通过参数把工作成果或者错误信息传给我们。
.l-sub-section .l-sub-section
:marked :marked
We are simplifying. Learn about ES2015 Promises [here](http://exploringjs.com/es6/ch_promises.html) and elsewhere on the web. We are simplifying. Learn about ES2015 Promises [here](http://exploringjs.com/es6/ch_promises.html) and elsewhere on the web.
这里只是粗浅的说说,要了解更多,请参见[这里](http://exploringjs.com/es6/ch_promises.html)以及在Web上搜索其它学习资源。 这里只是粗浅的说说,要了解更多,请参见[这里](http://exploringjs.com/es6/ch_promises.html)在Web上搜索其它学习资源。
:marked :marked
Update the `HeroService` with this promise-returning `getHeroes` method: Update the `HeroService` with this promise-returning `getHeroes` method:
@ -494,27 +495,27 @@ code-example(format="." language="html").
We're still mocking the data. We're simulating the behavior of an ultra-fast, zero-latency server, We're still mocking the data. We're simulating the behavior of an ultra-fast, zero-latency server,
by returning an **immediately resolved promise** with our mock heroes as the result. by returning an **immediately resolved promise** with our mock heroes as the result.
我们继续使用模拟数据。我们通过返回一个 *立即完成的承诺* 的方式,模拟了一个超快、零延迟的超级服务器。 我们继续使用模拟数据。我们通过返回一个 *立即解决的承诺* 的方式,模拟了一个超快、零延迟的超级服务器。
### Act on the Promise ### Act on the Promise
### 基于承诺的行动 ### 基于承诺的行动
Returning to the `AppComponent` and its `getHeroes` method, we see that it still looks like this: Returning to the `AppComponent` and its `getHeroes` method, we see that it still looks like this:
回到`AppComponent`和它的`getHeroes`方法,我们看到它看起来还是这样的: 回到`AppComponent`和它的`getHeroes`方法,我们看到它看起来还是这样的:
+makeExample('toh-4/ts/app/app.component.1.ts', 'getHeroes', 'app.component.ts (getHeroes - 老的)')(format=".") +makeExample('toh-4/ts/app/app.component.1.ts', 'getHeroes', 'app.component.ts (getHeroes - 老的)')(format=".")
:marked :marked
As a result of our change to `HeroService`, we're now setting `this.heroes` to a promise rather than an array of heroes. As a result of our change to `HeroService`, we're now setting `this.heroes` to a promise rather than an array of heroes.
作为我们修改`HeroService`的结果,我们正在把`this.heroes`替换为一个承诺,而不再是一个英雄数组。 在修改了`HeroService`之后,我们还要把`this.heroes`替换为一个承诺,而不再是一个英雄数组。
We have to change our implementation to *act on the promise when it resolves*. We have to change our implementation to *act on the promise when it resolves*.
When the promise resolves successfully, *then* we will have heroes to display. When the promise resolves successfully, *then* we will have heroes to display.
我们得修改我们的实现,把它变成 *基于承诺* 的,并且在承诺完成时再行动。 我们得修改这个实现,把它变成*基于承诺*的,并在承诺的事情被解决时再行动。
一旦承诺成功的完成了,我们就会显示英雄数据。 一旦承诺的事情被成功解决,我们就会显示英雄数据。
We pass our callback function as an argument to the promise's **then** method: We pass our callback function as an argument to the promise's **then** method:
我们把回调函数作为参数传给承诺对象的 **then** 函数: 我们把回调函数作为参数传给承诺对象的**then**函数:
+makeExample('toh-4/ts/app/app.component.ts', 'get-heroes', 'app.component.ts (getHeroes - 修改版)')(format=".") +makeExample('toh-4/ts/app/app.component.ts', 'get-heroes', 'app.component.ts (getHeroes - 修改版)')(format=".")
.l-sub-section .l-sub-section
:marked :marked
@ -522,7 +523,7 @@ code-example(format="." language="html").
in the callback is more succinct than the equivalent function expression and gracefully handles *this*. in the callback is more succinct than the equivalent function expression and gracefully handles *this*.
回调中所用的[ES2015箭头函数](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Functions/Arrow_functions) 回调中所用的[ES2015箭头函数](https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Functions/Arrow_functions)
能比等价的函数表达式更加简明、优雅的处理 *this* 指针。 能比等价的函数表达式更加快速、优雅的处理*this*指针。
:marked :marked
Our callback sets the component's `heroes` property to the array of heroes returned by the service. That's all there is to it! Our callback sets the component's `heroes` property to the array of heroes returned by the service. That's all there is to it!
@ -536,10 +537,10 @@ code-example(format="." language="html").
:marked :marked
Checkout the "[Take it slow](#slow)" appendix to see what the app might be like with a poor connection. Checkout the "[Take it slow](#slow)" appendix to see what the app might be like with a poor connection.
查看附件中的“[慢一点儿](#slow)”一节,了解在较差的网络连接中这个应用会是什么样的。 查看附件中的“[慢一点儿](#slow)”一节,了解在较差的网络连接中这个应用会是什么样的。
:marked :marked
### Review the App Structure ### Review the App Structure
### 回顾应用结构 ### 回顾应用结构
Lets verify that we have the following structure after all of our good refactoring in this chapter: Lets verify that we have the following structure after all of our good refactoring in this chapter:
再检查下,经历了本章的所有重构之后,我们应该有了下列文件结构: 再检查下,经历了本章的所有重构之后,我们应该有了下列文件结构:
@ -589,9 +590,9 @@ code-example(format="." language="html").
* We defined our `HeroService` as a provider for our `AppComponent` * We defined our `HeroService` as a provider for our `AppComponent`
* 我们把`HeroService`定义为`AppComponent`的一个供应商provider * 我们把`HeroService`定义为`AppComponent`的一个供应商provider
* We created mock hero data and imported them into our service * We created mock hero data and imported them into our service
* 我们创建了一个模拟的英雄数据,并把它导入我们的服务中 * 我们创建了一个模拟的英雄数据,并把它导入我们的服务中
* We designed our service to return a promise and our component to get our data from the promise * We designed our service to return a promise and our component to get our data from the promise
* 我们把服务改造为返回承诺的,并让组件从承诺获取数据 * 我们把服务改造为返回承诺的,并让组件从承诺获取数据
[Run the live example for part 4](/resources/live-examples/toh-4/ts/plnkr.html) [Run the live example for part 4](/resources/live-examples/toh-4/ts/plnkr.html)
@ -603,9 +604,9 @@ code-example(format="." language="html").
We want to create a dashboard, add menu links that route between the views, and format data in a template. We want to create a dashboard, add menu links that route between the views, and format data in a template.
As our app evolves, well learn how to design it to make it easier to grow and maintain. As our app evolves, well learn how to design it to make it easier to grow and maintain.
通过使用共享组件和服务,我们的《英雄指南》具有了更高的可复用性 通过使用共享组件和服务,我们的《英雄指南》更有复用性了
我们还要创建一个仪表盘,要添加菜单链接,路由到各个视图,还要在模板中格式化数据。 我们还要创建一个仪表盘,要添加菜单链接,路由到各个视图,还要在模板中格式化数据。
随着我们应用的进化,我们还会学到如何进行设计,让它更易扩展和维护。 随着我们应用的进化,我们还会学到如何进行设计,让它更易扩展和维护。
We learn about Angular Component Router and navigation among the views in the [next tutorial](toh-pt5.html) chapter. We learn about Angular Component Router and navigation among the views in the [next tutorial](toh-pt5.html) chapter.
@ -646,18 +647,18 @@ code-example(format="." language="html").
We stated [earlier](#child-component) that if we injected the parent `AppComponent` `HeroService` We stated [earlier](#child-component) that if we injected the parent `AppComponent` `HeroService`
into the `HeroDetailComponent`, *we must not add a providers array* to the `HeroDetailComponent` metadata. into the `HeroDetailComponent`, *we must not add a providers array* to the `HeroDetailComponent` metadata.
我们在[前面](#child-component)说过,如果我们把父组件`AppComponent`中的`HeroService`服务注入到`HeroDetailComponent` 我们在[前面](#child-component)说过,如果在父组件`AppComponent`中把`HeroService`服务注入到`HeroDetailComponent`
*我们就不应该添加一个providers数组* 到`HeroDetailComponent`的元数据中 *我们就不应该在`HeroDetailComponent`的元数据中再添加一个providers数组*
Why? Because that tells Angular to create a new instance of the `HeroService` at the `HeroDetailComponent` level. Why? Because that tells Angular to create a new instance of the `HeroService` at the `HeroDetailComponent` level.
The `HeroDetailComponent` doesn't want its *own* service instance; it wants its *parent's* service instance. The `HeroDetailComponent` doesn't want its *own* service instance; it wants its *parent's* service instance.
Adding the `providers` array creates a new service instance that shadows the parent instance. Adding the `providers` array creates a new service instance that shadows the parent instance.
为什么呢因为那会告诉Angular在`HeroDetailComponent`这一层创建一个新的`HeroService`实例。 为什么呢因为那会告诉Angular在`HeroDetailComponent`这一层创建一个新的`HeroService`实例。
显然,在这个例子中,`HeroDetailComponent`不会希望创建 *自己的* 服务实例,它想要的就父组件的服务实例。 显然,在这个例子中,`HeroDetailComponent`不会希望创建*自己的*服务实例,它想要的就是来自父组件的服务实例。
给组件添加一个`providers`数组注册provider就会创建一个新的服务实例它会遮蔽父组件中的同名实例。 给组件添加一个`providers`数组注册provider就会创建一个新的服务实例它会遮蔽父组件中的同名实例。
Think carefully about where and when to register a provider. Think carefully about where and when to register a provider.
Understand the scope of that registration. Be careful not to create a new service instance at the wrong level. Understand the scope of that registration. Be careful not to create a new service instance at the wrong level.
要想清楚在哪里、在什么时候注册供应商provider。要理解注册的范围。小心不要在错误的级别创建新的服务实例。 务必想清楚在哪里、在什么时候注册Provider。要理解注册的有效范围。小心点不要在错误的级别上创建新的服务实例。

View File

@ -5,15 +5,15 @@ include ../_util-fns
# 应用中的路由 # 应用中的路由
We received new requirements for our Tour of Heroes application: We received new requirements for our Tour of Heroes application:
我们收到了关于《英雄指南》应用的新需求: 我们收到了一份《英雄指南》的新需求:
* add a *Dashboard* view. * add a *Dashboard* view.
* 添加一个 *仪表盘* 视图。 * 添加一个*仪表盘*视图。
* navigate between the *Heroes* and *Dashboard* views. * navigate between the *Heroes* and *Dashboard* views.
* 在 *英雄列表* *仪表盘* 视图之间导航。 * 在*英雄列表*和*仪表盘*视图之间导航。
* clicking on a hero in either view navigates to a detail view of the selected hero. * clicking on a hero in either view navigates to a detail view of the selected hero.
* 无论在哪个视图中点击一个英雄,都会导航到该英雄的详情页。 * 无论在哪个视图中点击一个英雄,都会导航到该英雄的详情页。
* clicking a *deep link* in an email opens the detail view for a particular hero; * clicking a *deep link* in an email opens the detail view for a particular hero;
* 在邮件中点击一个 *深链接* ,会直接打开一个特定英雄的详情视图。 * 在邮件中点击一个*深链接*,会直接打开一个特定英雄的详情视图。
When were done, users will be able to navigate the app like this: When were done, users will be able to navigate the app like this:
@ -23,7 +23,7 @@ figure.image-display
:marked :marked
We'll add Angulars *Component Router* to our app to satisfy these requirements. We'll add Angulars *Component Router* to our app to satisfy these requirements.
我们将把Angular *组件路由器* 加入应用中,以满足这些需求。(译注:硬件的路由器是用来让你找到另一台机器的,而这里的路由器用于帮你找到一个组件) 我们将把Angular*组件路由器*加入应用中,以满足这些需求。(译注:硬件领域中的路由器是用来帮你找到另一台网络设备的,而这里的路由器用于帮你找到一个组件)
.l-sub-section .l-sub-section
:marked :marked
The [Routing and Navigation](../guide/router.html) chapter covers the router in more detail The [Routing and Navigation](../guide/router.html) chapter covers the router in more detail
@ -48,7 +48,7 @@ figure.image-display
Before we continue with our Tour of Heroes, lets verify that we have the following structure after adding our hero service Before we continue with our Tour of Heroes, lets verify that we have the following structure after adding our hero service
and hero detail component. If not, well need to go back and follow the previous chapters. and hero detail component. If not, well need to go back and follow the previous chapters.
在继续《英雄指南》之前,我们先检查一下,在添加了英雄服务和英雄详情组件之后,你是否已经有了如下目录结构。如果没有,你得先回上一章,再照做一 在继续《英雄指南》之前,先检查一下,在添加了英雄服务和英雄详情组件之后,你是否已经有了如下目录结构。如果没有,你得先回上一章,再照做一
.filetree .filetree
.file angular2-tour-of-heroes .file angular2-tour-of-heroes
.children .children
@ -92,49 +92,49 @@ code-example(format="." language="bash").
下面是我们的计划 下面是我们的计划
* turn `AppComponent` into an application shell that only handles navigation. * turn `AppComponent` into an application shell that only handles navigation.
* 把`AppComponent`变成应用的一个“壳”,它只处理导航。 * 把`AppComponent`变成应用程序的“壳”,它只处理导航。
* relocate the *Heroes* concerns within the current `AppComponent` to a separate `HeroesComponent` * relocate the *Heroes* concerns within the current `AppComponent` to a separate `HeroesComponent`
* 把现在`AppComponent`所关注的这些事移到一个独立的`HeroesComponent`中 * 把现在由`AppComponent`关注的*英雄们*移到一个独立的`HeroesComponent`中
* add routing * add routing
* 添加路由 * 添加路由
* create a new `DashboardComponent` * create a new `DashboardComponent`
* 添加一个新的`DashboardComponent`组件 * 添加一个新的`DashboardComponent`组件
* tie the *Dashboard* into the navigation structure. * tie the *Dashboard* into the navigation structure.
* 把 *仪表盘* 加入导航结构中。 * 把*仪表盘*加入导航结构中。
.l-sub-section .l-sub-section
:marked :marked
*Routing* is another name for *navigation*. The *router* is the mechanism for navigating from view to view. *Routing* is another name for *navigation*. The *router* is the mechanism for navigating from view to view.
*路由* 是导航的另一个名字。 *路由器* 就是从一个视图导航到另一个视图的机制。
*路由*是导航的另一个名字。*路由器*就是从一个视图导航到另一个视图的机制。
.l-main-section .l-main-section
:marked :marked
## Splitting the *AppComponent* ## Splitting the *AppComponent*
## 拆分 *AppComponent* ## 拆分*AppComponent*
Our current app loads `AppComponent` and immediately displays the list of heroes. Our current app loads `AppComponent` and immediately displays the list of heroes.
现在的应用会加载`AppComponent`组件,并且立显示出英雄列表。 现在的应用会加载`AppComponent`组件,并且立显示出英雄列表。
Our revised app should present a shell with a choice of views (*Dashboard* and *Heroes*) and then default to one of them. Our revised app should present a shell with a choice of views (*Dashboard* and *Heroes*) and then default to one of them.
我们修改过的应用将提供一个壳,它会从 *仪表盘* 和 *英雄列表* 中选择一个视图,然后默认转到其中之一 我们修改后的应用将提供一个壳,它会选择*仪表盘*和*英雄列表*视图之一,然后默认显示它
The `AppComponent` should only handle navigation. The `AppComponent` should only handle navigation.
Let's move the display of *Heroes* out of `AppComponent` and into its own `HeroesComponent`. Let's move the display of *Heroes* out of `AppComponent` and into its own `HeroesComponent`.
`AppComponent`组件应该只处理导航。 `AppComponent`组件应该只处理导航。
我们来把 *英雄列表* 的显示职责,从`AppComponent`移到`HeroesComponent`组件中。 我们来把*英雄列表*的显示职责,从`AppComponent`移到`HeroesComponent`组件中。
### *HeroesComponent*
### *HeroesComponent* ### *HeroesComponent*
`AppComponent` is already dedicated to *Heroes*. `AppComponent` is already dedicated to *Heroes*.
Instead of moving anything out of `AppComponent`, we'll just rename it `HeroesComponent` Instead of moving anything out of `AppComponent`, we'll just rename it `HeroesComponent`
and create a new `AppComponent` shell separately. and create a new `AppComponent` shell separately.
`AppComponent`已经移交给`HeroesComponent`了。 `AppComponent`的职责已经移交给`HeroesComponent`了。
与其把`AppComponent`中的所有东西都移过去,不如把它改名为`HeroesComponent`,并且单独创建一个新的`AppComponent`壳。 与其把`AppComponent`中所有的东西都搬过去,不如索性把它改名为`HeroesComponent`,然后单独创建一个新的`AppComponent`壳。
The steps are: The steps are:
@ -151,12 +151,12 @@ code-example(format="." language="bash").
:marked :marked
## Create *AppComponent* ## Create *AppComponent*
## 创建 *AppComponent* ## 创建*AppComponent*
The new `AppComponent` will be the application shell. The new `AppComponent` will be the application shell.
It will have some navigation links at the top and a display area below for the pages we navigate to. It will have some navigation links at the top and a display area below for the pages we navigate to.
新的`AppComponent`将成为应用的“壳”。 新的`AppComponent`将成为应用的“壳”。
它将在顶部放一些导航链接,并且把我们要导航到的页面放在下面的区域 它将在顶部放一些导航链接,并且把我们要导航到的页面放在下面的显示区中
The initial steps are: The initial steps are:
@ -171,17 +171,17 @@ code-example(format="." language="bash").
* expose an application `title` property. * expose an application `title` property.
* 导出应用的`title`属性。 * 导出应用的`title`属性。
* add the `@Component` metadata decorator above the class with a `my-app` selector. * add the `@Component` metadata decorator above the class with a `my-app` selector.
* 在类的顶部添加`@Component`元数据装饰器,其中指定`my-app`选择器。 * 在类的上方添加`@Component`元数据装饰器,装饰器中带有`my-app`选择器。
* add a template with `<h1>` tags surrounding a binding to the `title` property. * add a template with `<h1>` tags surrounding a binding to the `title` property.
* 在模板中添加一个`<h1>`标签,其中是到`title`属性的绑定。 * 在模板中添加一个`<h1>`标签,包裹着到`title`属性的绑定。
* add the `<my-heroes>` tags to the template so we still see the heroes. * add the `<my-heroes>` tags to the template so we still see the heroes.
* 在模板中添加`<my-heroes>`标签,以便我们仍能看到英雄列表。 * 在模板中添加`<my-heroes>`标签,以便我们仍能看到英雄列表。
* add the `HeroesComponent` to the `directives` array so Angular recognizes the `<my-heroes>` tags. * add the `HeroesComponent` to the `directives` array so Angular recognizes the `<my-heroes>` tags.
* 添加`HeroesComponent`组件到`directives`数组中以便Angular能认识`<my-heroes>`标签。 * 添加`HeroesComponent`组件到`directives`数组中以便Angular能认识`<my-heroes>`标签。
* add the `HeroService` to the `providers` array because we'll need it in every other view. * add the `HeroService` to the `providers` array because we'll need it in every other view.
* 添加`HeroService`到`providers`数组中,因为我们的每一个视图都需要它。 * 添加`HeroService`到`providers`数组中,因为我们的每一个视图都需要它。
* add the supporting `import` statements. * add the supporting `import` statements.
* 添加支性的`import`语句。 * 添加支性的`import`语句。
Our first draft looks like this: Our first draft looks like this:
@ -190,23 +190,23 @@ code-example(format="." language="bash").
:marked :marked
.callout.is-critical .callout.is-critical
header Remove <i>HeroService</i> from the <i>HeroesComponent</i> providers header Remove <i>HeroService</i> from the <i>HeroesComponent</i> providers
header 从<i>HeroesComponent</i>的`providers`中移除<i>HeroService</i> header 从<i>HeroesComponent</i>的<code>providers</code>中移除<i>HeroService</i>
:marked :marked
Go back to the `HeroesComponent` and **remove the `HeroService`** from its `providers` array. Go back to the `HeroesComponent` and **remove the `HeroService`** from its `providers` array.
We are *promoting* this service from the `HeroesComponent` to the `AppComponent`. We are *promoting* this service from the `HeroesComponent` to the `AppComponent`.
We ***do not want two copies*** of this service at two different levels of our app. We ***do not want two copies*** of this service at two different levels of our app.
回到`HeroesComponent`,并从`providers`数组中 **移除`HeroService`** 回到`HeroesComponent`,并从`providers`数组中**移除`HeroService`**。
我们要把它从`AppComponent` *晋升* 到`HeroesComponent`中。 我们要把它从`HeroesComponent`*提升*到`AppComponent`中。
我们不希望在应用的两个不同层次上存在它的 ***两个副本*** 我们不希望在应用的两个不同层次上存在它的***两个副本***。
:marked :marked
The app still runs and still displays heroes. The app still runs and still displays heroes.
Our refactoring of `AppComponent` into a new `AppComponent` and a `HeroesComponent` worked! Our refactoring of `AppComponent` into a new `AppComponent` and a `HeroesComponent` worked!
We have done no harm. We have done no harm.
应用仍然在运行,并显示着英雄列表。 应用仍然在运行,并显示着英雄列表。
我们把`AppComponent`重构成了一个新的`AppComponent`和`HeroesComponent`,它们工作得很好! 我们把`AppComponent`重构成了一个新的`AppComponent`和`HeroesComponent`,它们工作得很好!
我们毫发无伤的完成了重构。 我们毫发无损的完成了这次重构。
:marked :marked
## Add Routing ## Add Routing
## 添加路由 ## 添加路由
@ -216,12 +216,12 @@ code-example(format="." language="bash").
In other words, we'd like to navigate to the list of heroes. In other words, we'd like to navigate to the list of heroes.
我们已准备好开始下一步。 我们已准备好开始下一步。
与其自动显示英雄列表,我们更希望在用户点击按钮之后显示它。 与其自动显示英雄列表,我们更希望在用户点击按钮之后显示它。
换句话说,我们希望通过导航显示英雄列表。 换句话说,我们希望“导航”到英雄列表。
We'll need the Angular *Component Router*. We'll need the Angular *Component Router*.
我们需要Angular的 *组件路由器* 我们需要Angular的*组件路由器*。
### Set the base tag ### Set the base tag
### 设置base标签 ### 设置base标签
@ -235,14 +235,14 @@ code-example(format="." language="bash").
:marked :marked
See the *base href* section of the [Router](../guide/router.html#!#base-href) chapter to learn why this matters. See the *base href* section of the [Router](../guide/router.html#!#base-href) chapter to learn why this matters.
查看[路由器](../guide/router.html#!#base-href)一章的 *base href* 部分,了解为何如此。 查看[路由器](../guide/router.html#!#base-href)一章的*base href*部分,了解为何如此。
:marked :marked
### Make the router available. ### Make the router available.
### 让路由可用。 ### 让路由可用。
The *Component Router* is a service. Like any service, we have to import it and make it The *Component Router* is a service. Like any service, we have to import it and make it
available to the application by adding it to the `providers` array. available to the application by adding it to the `providers` array.
*组件路由器* 是一个服务。像所有服务一样,我们得导入它,并且通过把它加入`providers`数组来让它在应用中可用。 *组件路由器*是一个服务。像所有服务一样,我们得导入它,并且把它加入`providers`数组来让它在应用中可用。
The Angular router is a combination of multiple services (`ROUTER_PROVIDERS`), multiple directives (`ROUTER_DIRECTIVES`), The Angular router is a combination of multiple services (`ROUTER_PROVIDERS`), multiple directives (`ROUTER_DIRECTIVES`),
and a configuration decorator (`RouteConfig`). We'll import them all together: and a configuration decorator (`RouteConfig`). We'll import them all together:
@ -252,14 +252,14 @@ code-example(format="." language="bash").
:marked :marked
Next we update the `directives` and `providers` metadata arrays to *include* the router assets. Next we update the `directives` and `providers` metadata arrays to *include* the router assets.
接下来,我们我们更新`directives`和`providers`元数据数组,来包含这些路由器部件。 接下来,我们更新`directives`和`providers`元数据数组,来包含这些路由器部件。
+makeExample('toh-5/ts/app/app.component.2.ts', 'directives-and-providers', 'app.component.ts (directives和providers)')(format=".") +makeExample('toh-5/ts/app/app.component.2.ts', 'directives-and-providers', 'app.component.ts (directives和providers)')(format=".")
:marked :marked
Notice that we also removed the `HeroesComponent` from the `directives` array. Notice that we also removed the `HeroesComponent` from the `directives` array.
`AppComponent` no longer shows heroes; that will be the router's job. `AppComponent` no longer shows heroes; that will be the router's job.
We'll soon remove `<my-heroes>` from the template too. We'll soon remove `<my-heroes>` from the template too.
注意,我们已经从`directives`数组中移除了`HeroesComponent`。`AppComponent`不会再显示英雄,那是路由器的工作。 注意,我们已经从`directives`数组中移除了`HeroesComponent`。`AppComponent`不会再显示英雄,那是路由器的工作。
我们马上也会从模板中移除`<my-heroes>`。 我们马上也会从模板中移除`<my-heroes>`。
### Add and configure the router ### Add and configure the router
@ -268,12 +268,12 @@ code-example(format="." language="bash").
The `AppComponent` doesn't have a router yet. We'll use the `@RouteConfig` decorator to simultaneously The `AppComponent` doesn't have a router yet. We'll use the `@RouteConfig` decorator to simultaneously
(a) assign a router to the component and (b) configure that router with *routes*. (a) assign a router to the component and (b) configure that router with *routes*.
`AppComponent`还没有路由器。我们使用`@RouteConfig`装饰器来同时 (a)为组件指定一个路由器,并 (b) 通过 *routes* 来配置路由器。 `AppComponent`还没有路由器。我们使用`@RouteConfig`装饰器来同时 (a)为组件指定一个路由器,并 (b) 通过*路由*来配置路由器。
*Routes* tell the router which views to display when a user clicks a link or *Routes* tell the router which views to display when a user clicks a link or
pastes a URL into the browser address bar. pastes a URL into the browser address bar.
*routes* 告诉路由器当用户点击链接或者把URL粘贴到浏览器地址栏时应该显示哪个路由 *路由*告诉路由器当用户点击链接或者把URL粘贴到浏览器地址栏时应该显示哪个视图
Let's define our first route, a route to the `HeroesComponent`. Let's define our first route, a route to the `HeroesComponent`.
@ -283,8 +283,8 @@ code-example(format="." language="bash").
`@RouteConfig` takes an array of *route definitions*. `@RouteConfig` takes an array of *route definitions*.
We have only one route definition at the moment but rest assured, we'll add more. We have only one route definition at the moment but rest assured, we'll add more.
`@RouteConfig`有一个 *路由定义* 数组。 `@RouteConfig`有一个*路由定义*数组。
此刻我们只有一个路由定义,但别急,我们还会添加更多。 此刻我们只有一个路由定义,但别急,后面还会添加更多。
This *route definition* has three parts: This *route definition* has three parts:
@ -292,15 +292,15 @@ code-example(format="." language="bash").
* **path**: the router matches this route's path to the URL in the browser address bar (`/heroes`). * **path**: the router matches this route's path to the URL in the browser address bar (`/heroes`).
* **path**: 路由器会用它来匹配路由中的路径和浏览器地址栏中的路径,如`/heroes`。 * **path**: 路由器会用它来匹配路由中指定的路径和浏览器地址栏中的当前路径,如`/heroes`。
* **name**: the official name of the route; it *must* begin with a capital letter to avoid confusion with the *path* (`Heroes`). * **name**: the official name of the route; it *must* begin with a capital letter to avoid confusion with the *path* (`Heroes`).
* **name**: 路由的正式名字,它必须以大写字母开头儿,以免和 *path* 混淆,如`Heroes`。 * **name**: 路由的正式名字,它*必须*以大写字母开头儿,以免和*path*混淆,如`Heroes`。
* **component**: the component that the router should create when navigating to this route (`HeroesComponent`). * **component**: the component that the router should create when navigating to this route (`HeroesComponent`).
* **component**: 导航到此路由时,路由器需要创建的组件,如`HeroesComponent`。 * **component**: 导航到此路由时,路由器需要创建的组件,如`HeroesComponent`。
.l-sub-section .l-sub-section
:marked :marked
@ -315,23 +315,23 @@ code-example(format="." language="bash").
the router should match it to the `'Heroes'` route and display the `HeroesComponent`. the router should match it to the `'Heroes'` route and display the `HeroesComponent`.
But where? But where?
如果我们把路径`/heroes`粘贴到浏览器的地址栏,路由器会匹配到`'Heroes'`路由,并显示`HeroesComponent`组件。 如果我们把路径`/heroes`粘贴到浏览器的地址栏,路由器会匹配到`'Heroes'`路由,并显示`HeroesComponent`组件。
但问题是,把它显示在哪儿呢? 但问题是,把它显示在哪儿呢?
We have to ***tell it where*** by adding `<router-outlet>` marker tags to the bottom of the template. We have to ***tell it where*** by adding `<router-outlet>` marker tags to the bottom of the template.
`RouterOutlet` is one of the `ROUTER_DIRECTIVES`. `RouterOutlet` is one of the `ROUTER_DIRECTIVES`.
The router displays each component immediately below the `<router-outlet>` as we navigate through the application. The router displays each component immediately below the `<router-outlet>` as we navigate through the application.
我们必须 ***告诉它位置*** ,所以我们把`<router-outlet>`标签添加到模板的底部。 我们必须***告诉它位置***,所以我们把`<router-outlet>`标签添加到模板的底部。
`RouterOutlet`是`ROUTER_DIRECTIVES`常量中的一员。 `RouterOutlet`是`ROUTER_DIRECTIVES`常量中的一员。
当我们通过应用导航过来时,路由器立即把每个组件显示在`<router-outlet>`的位置 当我们在应用中导航时,路由器就把激活的组件显示在`<router-outlet>`里面
### Router Links ### Router Links
### 路由器链接 ### 路由器链接
We don't really expect users to paste a route URL into the address bar. We don't really expect users to paste a route URL into the address bar.
We add an anchor tag to the template which, when clicked, triggers navigation to the `HeroesComponent`. We add an anchor tag to the template which, when clicked, triggers navigation to the `HeroesComponent`.
我们不可能真等用户把路由的URL粘贴到地址栏中我们应该在模板中的什么地方添加一个A链接标签点击时就会触发导航到`HeroesComponent`组件的操作 我们当然不会真让用户往地址栏中粘贴路由的URL而应该在模板中的什么地方添加一个a链接标签。点击时就会导航到`HeroesComponent`组件
The revised template looks like this: The revised template looks like this:
@ -342,20 +342,20 @@ code-example(format="." language="bash").
We bind the `RouterLink` directive (another of the `ROUTER_DIRECTIVES`) to an array We bind the `RouterLink` directive (another of the `ROUTER_DIRECTIVES`) to an array
that tells the router where to navigate when the user clicks the link. that tells the router where to navigate when the user clicks the link.
注意,A标签中的`[routerLink]`绑定。我们把`RouterLink`指令(`ROUTER_DIRECTIVES`中的另一个指令)绑定到一个数组,它将告诉路由器,当用户点击这个链接时,应该导航到那里。 注意,a标签中的`[routerLink]`绑定。我们把`RouterLink`指令(`ROUTER_DIRECTIVES`中的另一个指令)绑定到一个数组。它将告诉路由器,当用户点击这个链接时,应该导航到哪里。
We define a *routing instruction* with a *link parameters array*. We define a *routing instruction* with a *link parameters array*.
The array only has one element in our little sample, the quoted ***name* of the route** to follow. The array only has one element in our little sample, the quoted ***name* of the route** to follow.
Looking back at the route configuration, we confirm that `'Heroes'` is the name of the route to the `HeroesComponent`. Looking back at the route configuration, we confirm that `'Heroes'` is the name of the route to the `HeroesComponent`.
我们通过一个 *链接参数数组* 定义了一个 *路由说明* 我们通过一个*链接参数数组*定义了一个*路由说明*。
我们这个小例子中,该数组只有一个元素,一个放在引号中的 **路由名称** 作路标。 在这个小例子中,该数组只有一个元素,一个放在引号中的**路由名称**,作路标。
回来看路由配置表,我们清楚地看到,这个名称 —— `'Heroes'` 就是指向`HeroesComponent`的那个路由的名称。 回来看路由配置表,我们清楚的看到,这个名称 —— `'Heroes'`就是指向`HeroesComponent`的那个路由的名称。
.l-sub-section .l-sub-section
:marked :marked
Learn about the *link parameters array* in the [Routing](../guide/router.html#link-parameters-array) chapter. Learn about the *link parameters array* in the [Routing](../guide/router.html#link-parameters-array) chapter.
学习关于 *接参数数组* 的更多知识,参见[路由](../guide/router.html#link-parameters-array)一章。 学习关于 *接参数数组* 的更多知识,参见[路由](../guide/router.html#link-parameters-array)一章。
:marked :marked
Refresh the browser. We see only the app title. We don't see the heroes list. Refresh the browser. We see only the app title. We don't see the heroes list.
@ -367,8 +367,9 @@ code-example(format="." language="bash").
We don't have a route that matches the path `/`, so there is nothing to show. We don't have a route that matches the path `/`, so there is nothing to show.
That's something we'll want to fix. That's something we'll want to fix.
浏览器的地址栏显示的是`/`。到`HeroesComponent`的路由中的路径是`/`,所以,自然没啥可显示的。 浏览器的地址栏显示的是`/`。而到`HeroesComponent`的路由中的路径是`/heroes`,不是`/`。
接下来我们要修复这个问题。 我们没有任何路由能匹配当前的路径`/`,所以,自然没啥可显示的。
接下来,我们就修复这个问题。
:marked :marked
We click the "Heroes" navigation link, the browser bar updates to `/heroes`, We click the "Heroes" navigation link, the browser bar updates to `/heroes`,
and now we see the list of heroes. We are navigating at last! and now we see the list of heroes. We are navigating at last!
@ -384,38 +385,38 @@ code-example(format="." language="bash").
For this reason and to distinguish it from other kinds of components, For this reason and to distinguish it from other kinds of components,
we call this type of component a *Router Component*. we call this type of component a *Router Component*.
*AppComponent* 现在有了一个路由器,并且能显示路由到的视图 *AppComponent*现在加上了路由器,并能显示路由到的视图了
因此,为了把它从其它种类的组件中区分出来,我们称这类组件为 *路由器组件*。 因此,为了把它从其它种类的组件中区分出来,我们称这类组件为*路由器组件*。
:marked :marked
## Add a *Dashboard* ## Add a *Dashboard*
## 添加一个 *仪表盘* ## 添加一个*仪表盘*
Routing only makes sense when we have multiple views. We need another view. Routing only makes sense when we have multiple views. We need another view.
只有在我们有多个视图的时候,路由才有意义。我们需要另一个视图。 我们有多个视图的时候,路由才有意义。所以我们需要另一个视图。
Create a placeholder `DashboardComponent` that gives us something to navigate to and from. Create a placeholder `DashboardComponent` that gives us something to navigate to and from.
先创建一个`DashboardComponent`的占位符,让我们可以导航到它或导航自它 先创建一个`DashboardComponent`的占位符,让我们可以导航到它或从它导航出来
+makeExample('toh-5/ts/app/dashboard.component.1.ts',null, 'app/dashboard.component.ts (v1)')(format=".") +makeExample('toh-5/ts/app/dashboard.component.1.ts',null, 'app/dashboard.component.ts (v1)')(format=".")
:marked :marked
Well come back and make it more useful later. Well come back and make it more useful later.
我们先不实现它,稍后,我们再回来,让这个组件更有用 我们先不实现它,稍后,我们再回来,给这个组件一些实际用途
### Configure the dashboard route ### Configure the dashboard route
### 配置仪表盘路由 ### 配置仪表盘路由
Go back to `app.component.ts` and teach it to navigate to the dashboard. Go back to `app.component.ts` and teach it to navigate to the dashboard.
回到`app.component.ts`文件,教它如何导航到这个仪表盘。 回到`app.component.ts`文件,我们得教它如何导航到这个仪表盘。
Import the `DashboardComponent` so we can reference it in the dashboard route definition. Import the `DashboardComponent` so we can reference it in the dashboard route definition.
导入`DashboardComponent`类,以便我们可以在仪表盘的路由定义中引用它 先导入`DashboardComponent`类,这样我们就可以在仪表盘的路由定义中引用它了
Add the following `'Dashboard'` route definition to the `@RouteConfig` array of definitions. Add the following `'Dashboard'` route definition to the `@RouteConfig` array of definitions.
把下列`'Dashboard'`路由的定义添加到`@RouteConfig`数组中 然后把下列`'Dashboard'`路由的定义添加到`@RouteConfig`数组中。
+makeExample('toh-5/ts/app/app.component.ts','dashboard-route', 'app.component.ts (仪表盘路由)')(format=".") +makeExample('toh-5/ts/app/app.component.ts','dashboard-route', 'app.component.ts (仪表盘路由)')(format=".")
.l-sub-section .l-sub-section
:marked :marked
@ -426,17 +427,17 @@ code-example(format="." language="bash").
Remember that the browser launches with `/` in the address bar. Remember that the browser launches with `/` in the address bar.
We don't have a route for that path and we'd rather not create one. We don't have a route for that path and we'd rather not create one.
我们希望应用在启动的时候就显示仪表盘,并且我们希望在浏览器的地址栏看到一个好看的URL比如`/dashboard`。 我们希望在应用启动的时候就显示仪表盘,而且我们希望在浏览器的地址栏看到一个好看的URL比如`/dashboard`。
记住,浏览器启动时,在地址栏中使用的路径是`/`。我们没有指向这个路径的路由,也不想创建它。 记住,浏览器启动时,在地址栏中使用的路径是`/`。我们没有指向这个路径的路由,也不想单独创建它。
Fortunately we can add the `useAsDefault: true` property to the *route definition* and the Fortunately we can add the `useAsDefault: true` property to the *route definition* and the
router will display the dashboard when the browser URL doesn't match an existing route. router will display the dashboard when the browser URL doesn't match an existing route.
幸运的是,我们可以把`useAsDefault: true`属性添加到 *路由定义* 上。这样如果浏览器中的URL匹配不上任何一个已知路由,那么路由器将显示这个仪表盘组件。 幸运的是,我们可以把`useAsDefault: true`属性添加到*路由定义*上。这样如果浏览器中的URL匹配不上任何已知的路由,路由器就会显示这个仪表盘组件。
:marked :marked
Finally, add a dashboard navigation link to the template, just above the *Heroes* link. Finally, add a dashboard navigation link to the template, just above the *Heroes* link.
最后,在模板上添加一个到仪表盘的导航链接,就在 *英雄(Heroes)* 链接的上方。 最后,在模板上添加一个到仪表盘的导航链接,就在*英雄(Heroes)*链接的上方。
+makeExample('toh-5/ts/app/app.component.ts','template', 'app.component.ts (模板)')(format=".") +makeExample('toh-5/ts/app/app.component.ts','template', 'app.component.ts (模板)')(format=".")
.l-sub-section .l-sub-section
@ -450,12 +451,12 @@ code-example(format="." language="bash").
Refresh the browser. The app displays the dashboard and Refresh the browser. The app displays the dashboard and
we can navigate between the dashboard and the heroes. we can navigate between the dashboard and the heroes.
刷新浏览器。应用显示出了仪表盘,并且我们可以在仪表盘和英雄列表之间导航了。 刷新浏览器。应用显示出了仪表盘,并可以在仪表盘和英雄列表之间导航了。
## Dashboard Top Heroes ## Dashboard Top Heroes
## 仪表盘上的顶英雄 ## 仪表盘上的顶英雄
Lets spice up the dashboard by displaying the top four heroes at a glance. Lets spice up the dashboard by displaying the top four heroes at a glance.
我们要让仪表盘更有趣,比如:一眼就能看到四个顶尖英雄。 我们想让仪表盘更有趣,比如:一眼就能看到四个顶级英雄。
Replace the `template` metadata with a `templateUrl` property that points to a new Replace the `template` metadata with a `templateUrl` property that points to a new
template file. template file.
@ -477,50 +478,50 @@ code-example(format="." language="bash").
We use `*ngFor` once again to iterate over a list of heroes and display their names. We use `*ngFor` once again to iterate over a list of heroes and display their names.
We added extra `<div>` elements to help with styling later in this chapter. We added extra `<div>` elements to help with styling later in this chapter.
我们又一次使用`*ngFor`来在英雄列表上迭代,并显示他们的名字。 我们又一次使用`*ngFor`来在英雄列表上迭代,并显示他们的名字。
我们添加了一个额外的`<div>`元素,来帮助稍后的美化工作。 添加了一个额外的`<div>`元素,来帮助稍后的美化工作。
There's a `(click)` binding to a `gotoDetail` method we haven't written yet and There's a `(click)` binding to a `gotoDetail` method we haven't written yet and
we're displaying a list of heroes that we don't have. we're displaying a list of heroes that we don't have.
We have work to do, starting with those heroes. We have work to do, starting with those heroes.
这里的`(click)`绑定到了`gotoDetail`方法,但我们还没实现它。并且我们也没有要显示的英雄列表数据。 这里的`(click)`绑定到了`gotoDetail`方法,但我们还没实现它。而且我们也还没有要显示的英雄列表数据。
我们有活儿干了,就从些英雄列表开始吧。 我们有活儿干了,就从些英雄列表开始吧。
### Share the *HeroService* ### Share the *HeroService*
### 共享 *HeroService* ### 共享*HeroService*
We'd like to re-use the `HeroService` to populate the component's `heroes` array. We'd like to re-use the `HeroService` to populate the component's `heroes` array.
我们想要用`HeroService`来存放组件的`heroes`数组。 我们想要用`HeroService`来存放组件的`heroes`数组。
Recall earlier in the chapter that we removed the `HeroService` from the `providers` array of the `HeroesComponent` Recall earlier in the chapter that we removed the `HeroService` from the `providers` array of the `HeroesComponent`
and added it to the `providers` array of the top level `AppComponent`. and added it to the `providers` array of the top level `AppComponent`.
回忆一下,在前面的章节中,我们从`HeroesComponent`的`providers`数组中移除了`HeroService`服务,并把它添加到顶级组件`AppComponent`的`providers`数组中。 回忆一下,在前面的章节中,我们从`HeroesComponent`的`providers`数组中移除了`HeroService`服务,并把它添加到顶级组件`AppComponent`的`providers`数组中。
That move created a singleton `HeroService` instance, available to *all* components of the application. That move created a singleton `HeroService` instance, available to *all* components of the application.
We'll inject and use it here in the `DashboardComponent` . We'll inject and use it here in the `DashboardComponent` .
这个改动创建了一个`HeroService`的单例对象,并且对应用中的 *所有* 组件都有效。 这个改动创建了一个`HeroService`的单例对象,它对应用中的*所有*组件都有效。
在`DashboardComponent`组件中,我们把它注入进来,并使用它。 在`DashboardComponent`组件中,我们把它注入进来,并使用它。
### Get heroes ### Get heroes
### 获取英雄数组 ### 获取英雄数组
Open the `dashboard.component.ts` and add the requisite `import` statements. Open the `dashboard.component.ts` and add the requisite `import` statements.
打开`dashboard.component.ts`文件,并把必备的`import`语句加进去。 打开`dashboard.component.ts`文件,并把必备的`import`语句加进去。
+makeExample('toh-5/ts/app/dashboard.component.2.ts','imports', 'app/dashboard.component.ts (导入)')(format=".") +makeExample('toh-5/ts/app/dashboard.component.2.ts','imports', 'app/dashboard.component.ts (导入)')(format=".")
:marked :marked
We need `OnInit` interface because we'll initialize the heroes in the `ngOnInit` method as we've done before. We need `OnInit` interface because we'll initialize the heroes in the `ngOnInit` method as we've done before.
We need the `Hero` and `HeroService` symbols in order to reference those types. We need the `Hero` and `HeroService` symbols in order to reference those types.
我们得实现`OnInit`接口,因为我们要在`ngOnInit`方法中初始化英雄数组 —— 就像上次一样。 我们得实现`OnInit`接口,因为我们要在`ngOnInit`方法中初始化英雄数组 —— 就像上次一样。
我们需要导入`Hero`和`HeroService`类来引用它们的数据类型。 我们需要导入`Hero`和`HeroService`类来引用它们的数据类型。
Now implement the `DashboardComponent` class like this: Now implement the `DashboardComponent` class like this:
现在就实现`DashboardComponent`类,像这样: 我们现在就实现`DashboardComponent`类,像这样:
+makeExample('toh-5/ts/app/dashboard.component.2.ts','component', 'app/dashboard.component.ts (类)') +makeExample('toh-5/ts/app/dashboard.component.2.ts','component', 'app/dashboard.component.ts (类)')
:marked :marked
We saw this kind of logic before in the `HeroesComponent`. We saw this kind of logic before in the `HeroesComponent`.
@ -536,11 +537,11 @@ code-example(format="." language="bash").
The noteworthy differences: we cherry-pick four heroes (2nd, 3rd, 4th, and 5th) with *slice* The noteworthy differences: we cherry-pick four heroes (2nd, 3rd, 4th, and 5th) with *slice*
and stub the `gotoDetail` method until we're ready to implement it. and stub the `gotoDetail` method until we're ready to implement it.
不同之处在于:我们使用 *slice* 函数挑选四个英雄第2、3、4、5个且先为`gotoDetail`方法提供桩实现 —— 直到我们准备实现它。 不同之处在于:我们使用*slice*函数挑选四个英雄第2、3、4、5个暂时为`gotoDetail`方法提供桩实现 —— 直到我们准备实现它。
Refresh the browser and see four heroes in the new dashboard. Refresh the browser and see four heroes in the new dashboard.
刷新浏览器,并且在新的仪表盘中看到了四个英雄。 刷新浏览器,在这个新的仪表盘中看到了四个英雄。
.l-main-section .l-main-section
:marked :marked
@ -550,13 +551,13 @@ code-example(format="." language="bash").
Although we display the details of a selected hero at the bottom of the `HeroesComponent`, Although we display the details of a selected hero at the bottom of the `HeroesComponent`,
we don't yet *navigate* to the `HeroDetailComponent` in the three ways specified in our requirements: we don't yet *navigate* to the `HeroDetailComponent` in the three ways specified in our requirements:
虽然我们在`HeroesComponent`组件的底部显示了所选英雄的详情,但我们还从没有 *导航* 到`HeroDetailComponent`组件,不管用我们需求中指定的三种方式中的哪一种: 虽然我们在`HeroesComponent`组件的底部显示了所选英雄的详情,但我们还从没有*导航*到`HeroDetailComponent`组件过 —— 我们曾在需求中指定过三种:
1. from the *Dashboard* to a selected hero. 1. from the *Dashboard* to a selected hero.
1. 从 *仪表盘(Dashboard)* 导航到一个选定的英雄。 1. 从*仪表盘(Dashboard)*导航到一个选定的英雄。
1. from the *Heroes* list to a selected hero. 1. from the *Heroes* list to a selected hero.
1. 从 *英雄列表(Heroes)* 导航到一个选定的英雄。 1. 从*英雄列表(Heroes)*导航到一个选定的英雄。
1. from a "deep link" URL pasted into the browser address bar. 1. from a "deep link" URL pasted into the browser address bar.
1. 把一个指向他的"深链接"URL粘贴到浏览器的地址栏。 1. 把一个指向该英雄的“深链接”URL粘贴到浏览器的地址栏。
Adding a `'HeroDetail'` route seem an obvious place to start. Adding a `'HeroDetail'` route seem an obvious place to start.
@ -567,13 +568,13 @@ code-example(format="." language="bash").
We'll add a route to the `HeroDetailComponent` in the `AppComponent` where our other routes are configured. We'll add a route to the `HeroDetailComponent` in the `AppComponent` where our other routes are configured.
我们将在`AppComponent`中添加一个到`HeroDetailComponent`的路由,也是配置其它路由的地方。 我们将在`AppComponent`中添加一个到`HeroDetailComponent`的路由,也是配置其它路由的地方。
The new route is a bit unusual in that we must tell the `HeroDetailComponent` *which hero to show*. The new route is a bit unusual in that we must tell the `HeroDetailComponent` *which hero to show*.
We didn't have to tell the `HeroesComponent` or the `DashboardComponent` anything. We didn't have to tell the `HeroesComponent` or the `DashboardComponent` anything.
新路由的不寻常之处在于,我们必须告诉`HeroDetailComponent` *该显示哪个英雄* 新路由的不寻常之处在于,我们必须告诉`HeroDetailComponent`*该显示哪个英雄*。
以前的`HeroesComponent`组件和`DashboardComponent`组件都还不曾要求我们告诉它任何东西。 以前的`HeroesComponent`组件和`DashboardComponent`组件还从未要求我们告诉它任何东西。
At the moment the parent `HeroesComponent` sets the component's `hero` property to a hero object with a binding like this. At the moment the parent `HeroesComponent` sets the component's `hero` property to a hero object with a binding like this.
@ -586,67 +587,67 @@ code-example(format='').
Certainly not the last one; we can't embed an entire hero object in the URL! Nor would we want to. Certainly not the last one; we can't embed an entire hero object in the URL! Nor would we want to.
显然,在我们的任何一个路由场景中它都无法工作。 显然,在我们的任何一个路由场景中它都无法工作。
不仅如此我们也没法把一个完整的hero对象嵌入到URL中其实我们也不想这样。 不仅如此我们也没法把一个完整的hero对象嵌入到URL中不过我们本来也不想这样。
### Parameterized route ### Parameterized route
### 参数化路由 ### 参数化路由
We *can* add the hero's `id` to the URL. When routing to the hero whose `id` is 11, we could expect to see an URL such as this: We *can* add the hero's `id` to the URL. When routing to the hero whose `id` is 11, we could expect to see an URL such as this:
我们 *可以* 把英雄的`id`添加到URL中。当导航到一个`id`为11的英雄时我们期望的URL是这样的 我们*可以*把英雄的`id`添加到URL中。当导航到一个`id`为11的英雄时我们期望的URL是这样的
code-example(format=''). code-example(format='').
/detail/11 /detail/11
:marked :marked
The `/detail/` part of that URL is constant. The trailing numeric `id` part changes from hero to hero. The `/detail/` part of that URL is constant. The trailing numeric `id` part changes from hero to hero.
We need to represent that variable part of the route with a *parameter* (or *token*) that stands for the hero's `id`. We need to represent that variable part of the route with a *parameter* (or *token*) that stands for the hero's `id`.
URL中的`/detail/`部分是不变的。结尾的数字`id`部分会随着英雄的不同而变化。 URL中的`/detail/`部分是固定不变的,但结尾的数字`id`部分会随着英雄的不同而变化。
我们要把路由中可变的那部分表示成一个 *参数(parameter)* 或 *标识(token)* ,用以获取英雄的`id`。 我们要把路由中可变的那部分表示成一个*参数(parameter)*或*Token*,用以获取英雄的`id`。
### Configure a Route with a Parameter ### Configure a Route with a Parameter
### 通过参数来配置路由 ### 配置带参数的路由
Here's the *route definition* we'll use. Here's the *route definition* we'll use.
下面是我们将使用的 *路由定义* 下面是我们将使用的*路由定义*
+makeExample('toh-5/ts/app/app.component.ts','hero-detail-route', 'app/app.component.ts (到HeroDetailComponent的路由)')(format=".") +makeExample('toh-5/ts/app/app.component.ts','hero-detail-route', 'app/app.component.ts (到HeroDetailComponent的路由)')(format=".")
:marked :marked
The colon (:) in the path indicates that `:id` is a placeholder to be filled with a specific hero `id` The colon (:) in the path indicates that `:id` is a placeholder to be filled with a specific hero `id`
when navigating to the `HeroDetailComponent`. when navigating to the `HeroDetailComponent`.
路径中的冒号(:标记`:id`是一个占位符,当导航到这个`HeroDetailComponent`组件时,它将被填成一个特定英雄的`id`。 路径中的冒号(:表示`:id`是一个占位符,当导航到这个`HeroDetailComponent`组件时,它将被填入一个特定英雄的`id`。
.l-sub-section .l-sub-section
:marked :marked
Of course we have to import the `HeroDetailComponent` before we create this route: Of course we have to import the `HeroDetailComponent` before we create this route:
当然,在创建这个路由之前,我们必须导入`HeroDetailComponent`类 当然,在创建这个路由之前,我们必须导入`HeroDetailComponent`类
+makeExample('toh-5/ts/app/app.component.ts','hero-detail-import')(format=".") +makeExample('toh-5/ts/app/app.component.ts','hero-detail-import')(format=".")
:marked :marked
We're finished with the `AppComponent`. We're finished with the `AppComponent`.
我们正在完成`AppComponent`组件。 我们已完成了`AppComponent`组件。
We won't add a `'Hero Detail'` link to the template because users We won't add a `'Hero Detail'` link to the template because users
don't click a navigation *link* to view a particular hero. don't click a navigation *link* to view a particular hero.
They click a *hero* whether that hero is displayed on the dashboard or in the heroes list. They click a *hero* whether that hero is displayed on the dashboard or in the heroes list.
我们没有往模板中添加一个`'英雄详情'`,这是因为用户不会直接点击导航栏中的链接去查看一个特定的英雄。 我们没有往模板中添加一个`'英雄详情'`,这是因为用户不会直接点击导航栏中的链接去查看一个特定的英雄。
他们只会从英雄列表或者仪表盘中点击一个英雄。 他们只会通过在英雄列表或者仪表盘中点击来显示一个英雄。
We'll get to those *hero* clicks later in the chapter. We'll get to those *hero* clicks later in the chapter.
There's no point in working on them until the `HeroDetailComponent` There's no point in working on them until the `HeroDetailComponent`
is ready to be navigated *to*. is ready to be navigated *to*.
稍后我们会响应这些 *英雄* 的点击事件。 稍后我们会响应这些*英雄*的点击事件。
现在对它们做什么都没有意义 —— 除非`HeroDetailComponent`已经做好了,并且能够被导航过去。 现在对它们做什么都没有意义 —— 除非`HeroDetailComponent`已经做好了,并且能够被导航过去。
That will require an `HeroDetailComponent` overhaul. That will require an `HeroDetailComponent` overhaul.
那需要对`HeroDetailComponent`做一次大修。 需要对`HeroDetailComponent`做一次大修。
.l-main-section .l-main-section
:marked :marked
## Revise the *HeroDetailComponent* ## Revise the *HeroDetailComponent*
## 修改 *HeroDetailComponent* ## 修改*HeroDetailComponent*
Before we rewrite the `HeroDetailComponent`, let's remember what it looks like now: Before we rewrite the `HeroDetailComponent`, let's remember what it looks like now:
@ -655,14 +656,14 @@ code-example(format='').
:marked :marked
The template won't change. We'll display a hero the same way. The big changes are driven by how we get the hero. The template won't change. We'll display a hero the same way. The big changes are driven by how we get the hero.
模板不用修改。我们会用跟以前一样的方式显示英雄。导致这次大修的原因是如何获得这个英雄的数据。 模板不用修改,我们会用原来的方式显示英雄。导致这次大修的原因是如何获得这个英雄的数据。
We will no longer receive the hero in a parent component property binding. We will no longer receive the hero in a parent component property binding.
The new `HeroDetailComponent` should take the `id` parameter from the router's `RouteParams` service The new `HeroDetailComponent` should take the `id` parameter from the router's `RouteParams` service
and use the `HeroService` to fetch the hero with that `id` from storage. and use the `HeroService` to fetch the hero with that `id` from storage.
我们不会再从父组件的属性绑定中取得英雄数据。 我们不会再从父组件的属性绑定中取得英雄数据。
新的`HeroDetailComponent`应该从路由器的`RouteParams`服务取得`id`参数,并通过`HeroService`服务获取具有这个指定`id`的英雄数据。 新的`HeroDetailComponent`应该从路由器的`RouteParams`服务取得`id`参数,并通过`HeroService`服务获取具有这个指定`id`的英雄数据。
We need an import statement to reference the `RouteParams`. We need an import statement to reference the `RouteParams`.
@ -671,7 +672,7 @@ code-example(format='').
:marked :marked
We import the `HeroService`so we can fetch a hero`. We import the `HeroService`so we can fetch a hero`.
我们导入了`HeroService`,现在我们能获取一个英雄了。 我们导入了`HeroService`,现在能获取一个英雄了。
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'import-hero-service')(format=".") +makeExample('toh-5/ts/app/hero-detail.component.ts', 'import-hero-service')(format=".")
:marked :marked
We import the `OnInit` interface because we'll call the `HeroService` inside the `ngOnInit` component lifecycle hook. We import the `OnInit` interface because we'll call the `HeroService` inside the `ngOnInit` component lifecycle hook.
@ -704,19 +705,19 @@ code-example(format='').
The hero `id` is a number. Route parameters are *always strings*. The hero `id` is a number. Route parameters are *always strings*.
So we convert the route parameter value to a number with the JavaScript (+) operator. So we convert the route parameter value to a number with the JavaScript (+) operator.
英雄的`id`是数字。路由参数 *永远是字符串* 英雄的`id`是数字,而路由参数的值*总是字符串*
所以我们需要通过JavaScript的(+)操作符把路由参数的值转成数字。 所以我们需要通过JavaScript的(+)操作符把路由参数的值转成数字。
### Add *HeroService.getHero* ### Add *HeroService.getHero*
### 添加 *HeroService.getHero* ### 添加*HeroService.getHero*
The problem with this bit of code is that `HeroService` doesn't have a `getHero` method! The problem with this bit of code is that `HeroService` doesn't have a `getHero` method!
We better fix that quickly before someone notices that we broke the app. We better fix that quickly before someone notices that we broke the app.
些代码的问题在于`HeroService`并没有一个叫`getHero`的方法,我们最好在有人通知我们应用坏了之前赶快修复它。 段代码的问题在于`HeroService`并没有一个叫`getHero`的方法,我们最好在别人报告应用出问题之前赶快修复它。
Open `HeroService` and add the `getHero` method. It's trivial given that we're still faking data access: Open `HeroService` and add the `getHero` method. It's trivial given that we're still faking data access:
打开`HeroService`,并添加`getHero`方法进去。对于我们的假数据存取逻辑来说,这点修改是微不足道的: 打开`HeroService`,并添加一个`getHero`方法。对于我们的假数据存取逻辑来说,这点修改是微不足道的:
+makeExample('toh-5/ts/app/hero.service.ts', 'get-hero', 'app/hero.service.ts (getHero)')(format=".") +makeExample('toh-5/ts/app/hero.service.ts', 'get-hero', 'app/hero.service.ts (getHero)')(format=".")
:marked :marked
Return to the `HeroDetailComponent` to clean up loose ends. Return to the `HeroDetailComponent` to clean up loose ends.
@ -729,14 +730,14 @@ code-example(format='').
We can navigate *to* the `HeroDetailComponent` in several ways. We can navigate *to* the `HeroDetailComponent` in several ways.
How do we navigate somewhere else when we're done? How do we navigate somewhere else when we're done?
我们能用多种方式导航 *到* `HeroDetailComponent`。 我们能用多种方式导航*到*`HeroDetailComponent`。
但当我们完工时,我们该导航到那里呢? 但当我们完工时,我们该导航到那里呢?
The user could click one of the two links in the `AppComponent`. Or click the browser's back button. The user could click one of the two links in the `AppComponent`. Or click the browser's back button.
We'll add a third option, a `goBack` method that navigates backward one step in the browser's history stack We'll add a third option, a `goBack` method that navigates backward one step in the browser's history stack
用户可以点击`AppComponent`中的两个链接,或点击浏览器的“后退”按钮。 现在用户可以点击`AppComponent`中的两个链接,或点击浏览器的“后退”按钮。
我们来添加第三个选项:一个`goBack`方法,来根据浏览器的历史堆栈,往回退一步。 我们来添加第三个选项:一个`goBack`方法,来根据浏览器的历史堆栈,退一步。
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'go-back', 'app/hero-detail.component.ts (goBack)')(format=".") +makeExample('toh-5/ts/app/hero-detail.component.ts', 'go-back', 'app/hero-detail.component.ts (goBack)')(format=".")
.l-sub-section .l-sub-section
@ -745,13 +746,13 @@ code-example(format='').
That's acceptable in a demo. We'd guard against it in a real application, That's acceptable in a demo. We'd guard against it in a real application,
perhaps with the [*routerCanDeactivate* hook](../api/router/CanDeactivate-interface.html). perhaps with the [*routerCanDeactivate* hook](../api/router/CanDeactivate-interface.html).
回退太多步儿会离开我们的应用。 回退太多步儿会跑出我们的应用。
在Demo中这算不上问题。但在真实的应用中我们需要对此进行防范。 在Demo中这算不上问题。但在真实的应用中我们需要对此进行防范。
或许你该用[*routerCanDeactivate* 钩子](/docs/ts/latest/api/router/CanDeactivate-interface.html)。 也许你该用[*routerCanDeactivate*钩子](/docs/ts/latest/api/router/CanDeactivate-interface.html)。
:marked :marked
Then we wire this method with an event binding to a *Back* button that we add to the bottom of the component template. Then we wire this method with an event binding to a *Back* button that we add to the bottom of the component template.
然后,我们通过一个事件绑定把此方法绑定到模板底部的 *后退(Back)* 按钮上。 然后,我们通过一个事件绑定把此方法绑定到模板底部的*后退(Back)*按钮上。
+makeExample('toh-5/ts/app/hero-detail.component.html', 'back-button')(format=".") +makeExample('toh-5/ts/app/hero-detail.component.html', 'back-button')(format=".")
:marked :marked
Modifing the template to add this button spurs us to take one more incremental improvement and migrate the template to its own file Modifing the template to add this button spurs us to take one more incremental improvement and migrate the template to its own file
@ -775,10 +776,10 @@ code-example(format='').
.l-main-section .l-main-section
:marked :marked
## Select a *Dashboard* Hero ## Select a *Dashboard* Hero
## 选择一个 *仪表盘* 中的英雄 ## 选择一个*仪表盘*中的英雄
When a user selects a hero in the dashboard, the app should navigate to the `HeroDetailComponent` to view and edit the selected hero.. When a user selects a hero in the dashboard, the app should navigate to the `HeroDetailComponent` to view and edit the selected hero..
当用户从仪表盘中选择了一位英雄时,应用应该导航到`HeroDetailComponent`以查看和编辑所选的英雄。 当用户从仪表盘中选择了一位英雄时,本应用要导航到`HeroDetailComponent`以查看和编辑所选的英雄。
In the dashboard template we bound each hero's click event to the `gotoDetail` method, passing along the selected `hero` entity. In the dashboard template we bound each hero's click event to the `gotoDetail` method, passing along the selected `hero` entity.
@ -803,16 +804,16 @@ code-example(format='').
We wrote *link parameters arrays* in the `AppComponent` for the navigation links. We wrote *link parameters arrays* in the `AppComponent` for the navigation links.
Those arrays had only one element, the name of the destination route. Those arrays had only one element, the name of the destination route.
我们当初在`AppComponent`中生成导航链接的时候曾写过 *链接参数数组* 我们当初在`AppComponent`中生成导航链接的时候曾写过*链接参数数组*。
This array has two elements, the ***name*** of the destination route and a ***route parameter object*** This array has two elements, the ***name*** of the destination route and a ***route parameter object***
with an `id` field set to the value of the selected hero's `id`. with an `id` field set to the value of the selected hero's `id`.
这个数组有两个元素,目标路由的 ***名称(name)*** ,和一个 ***路由参数对象*** ,其中包括一个`id`字段,它的取值是所选英雄的`id`。 这个数组有两个元素,目标路由的***名称(name)***,和一个***路由参数对象***,其中包括一个`id`字段,它的取值是所选英雄的`id`。
The two array items align with the ***name*** and ***:id*** token in the parameterized `HeroDetail` route configuration we added to `AppComponent` earlier in the chapter. The two array items align with the ***name*** and ***:id*** token in the parameterized `HeroDetail` route configuration we added to `AppComponent` earlier in the chapter.
当初我们在`AppComponent`中添加路由的时候,这两个数组元素以 ***name*** 和 ***:id*** 为代号被参数化在路由配置对象`HeroDetail`中。 当初我们在`AppComponent`中添加路由的时候,这两个数组元素以***name***和***:id***为代号被参数化的`HeroDetail`路由配置对象中。
+makeExample('toh-5/ts/app/app.component.ts','hero-detail-route', 'app/app.component.ts (英雄详情路由)')(format=".") +makeExample('toh-5/ts/app/app.component.ts','hero-detail-route', 'app/app.component.ts (英雄详情路由)')(format=".")
:marked :marked
The `DashboardComponent` doesn't have the router yet. We obtain it in the usual way: The `DashboardComponent` doesn't have the router yet. We obtain it in the usual way:
@ -826,20 +827,20 @@ code-example(format='').
:marked :marked
Refresh the browser and select a hero from the dashboard; the app should navigate directly to that heros details. Refresh the browser and select a hero from the dashboard; the app should navigate directly to that heros details.
刷新浏览器,并且从仪表盘中选择一位英雄;应用就会直接导航到英雄的详情。 刷新浏览器,并从仪表盘中选择一位英雄,应用就会直接导航到英雄的详情。
.l-main-section .l-main-section
:marked :marked
## Select a Hero in the *HeroesComponent* ## Select a Hero in the *HeroesComponent*
## 在 *HeroesComponent* 中选择一位英雄 ## 在*HeroesComponent*中选择一位英雄
We'll do something similar in the `HeroesComponent`. We'll do something similar in the `HeroesComponent`.
我们要做的事和`HeroesComponent`中很像。 我们现在要做的事和`HeroesComponent`中很像。
That component's current template exhibits a "master/detail" style with the list of heroes That component's current template exhibits a "master/detail" style with the list of heroes
at the top and details of the selected hero below. at the top and details of the selected hero below.
那个组件现在的模板展示了一个主从风格的界面:上方是英雄列表,底下是所选英雄的详情。 那个组件的当前模板展示了一个主从风格的界面:上方是英雄列表,底下是所选英雄的详情。
+makeExample('toh-4/ts/app/app.component.ts','template', 'app/heroes.component.ts (当前的模板)')(format=".") +makeExample('toh-4/ts/app/app.component.ts','template', 'app/heroes.component.ts (当前的模板)')(format=".")
:marked :marked
Delete the last line of the template with the `<my-hero-detail>` tags. Delete the last line of the template with the `<my-hero-detail>` tags.
@ -850,17 +851,17 @@ code-example(format='').
We're going to display the hero detail on its own page and route to it as we did in the dashboard. We're going to display the hero detail on its own page and route to it as we did in the dashboard.
这里我们不再展示完整的`HeroDetailComponent`了。 这里我们不再展示完整的`HeroDetailComponent`了。
我们要在它自己的页面中显示英雄详情,并像我们在仪表盘中所做的那样路由到它。 我们要在它自己的页面中显示英雄详情,并像我们在仪表盘中所做的那样路由到它。
But we'll throw in a small twist for variety. But we'll throw in a small twist for variety.
When the user selects a hero from the list, we *won't* go to the detail page. When the user selects a hero from the list, we *won't* go to the detail page.
We'll show a *mini-detail* on *this* page instead and make the user click a button to navigate to the *full detail *page. We'll show a *mini-detail* on *this* page instead and make the user click a button to navigate to the *full detail *page.
但是,我们要做一点小小的改动。 但是,我们要做一点小小的改动。
当用户从这个列表中选择一个英雄时,我们 *不会* 再跳转到详情页。 当用户从这个列表中选择一个英雄时,我们*不会*再跳转到详情页。
而是在本页中显示一个 *Mini版英雄详情* ,并且让用户点击一个按钮,来导航到 *完整版英雄详情* 页。 而是在本页中显示一个*Mini版英雄详情*,并等用户点击一个按钮时,才导航到*完整版英雄详情*页。
### Add the *mini-detail* ### Add the *mini-detail*
### 添加 *Mini版英雄详情* ### 添加*Mini版英雄详情*
Add the following HTML fragment at the bottom of the template where the `<my-hero-detail>` used to be: Add the following HTML fragment at the bottom of the template where the `<my-hero-detail>` used to be:
在模板底部原来放`<my-hero-detail>`的地方添加下列HTML片段 在模板底部原来放`<my-hero-detail>`的地方添加下列HTML片段
@ -874,7 +875,7 @@ figure.image-display
img(src='/resources/images/devguide/toh/mini-hero-detail.png' alt="Mini版英雄详情" height="70") img(src='/resources/images/devguide/toh/mini-hero-detail.png' alt="Mini版英雄详情" height="70")
:marked :marked
### Format with the *UpperCasePipe* ### Format with the *UpperCasePipe*
### 使用 *UpperCasePipe* 格式化 ### 使用*UpperCasePipe*格式化
Notice that the hero's name is displayed in CAPITAL LETTERS. That's the effect of the `UpperCasePipe` Notice that the hero's name is displayed in CAPITAL LETTERS. That's the effect of the `UpperCasePipe`
that we slipped into the interpolation binding. Look for it right after the pipe operator, ( | ). that we slipped into the interpolation binding. Look for it right after the pipe operator, ( | ).
@ -886,7 +887,7 @@ figure.image-display
Angular ships with several pipes and we can write our own. Angular ships with several pipes and we can write our own.
管道擅长做下列工作:格式化字符串、金额、日期和其它显示数据。 管道擅长做下列工作:格式化字符串、金额、日期和其它显示数据。
Angular自带了好几个管道且我们还可以写自己的管道。 Angular自带了好几个管道且我们还可以写自己的管道。
.l-sub-section .l-sub-section
:marked :marked
Learn about pipes in the [Pipes](../guide/pipes.html) chapter. Learn about pipes in the [Pipes](../guide/pipes.html) chapter.
@ -898,23 +899,23 @@ figure.image-display
We are not done. We still have to update the component class to support navigation to the We are not done. We still have to update the component class to support navigation to the
`HeroDetailComponent` when the user clicks the *View Details* button. `HeroDetailComponent` when the user clicks the *View Details* button.
还没完呢。在用户点击 *查看详情* 按钮时,要让他能导航到 `HeroDetailComponent`,我们仍然不得不修改组件类。 这还没完。当用户点击*查看详情*按钮时,要让他能导航到`HeroDetailComponent`,我们仍然不得不修改组件类。
This component file is really big. Most of it is either template or CSS styles. This component file is really big. Most of it is either template or CSS styles.
It's difficult to find the component logic amidst the noise of HTML and CSS. It's difficult to find the component logic amidst the noise of HTML and CSS.
这个组件文件太大了。它大部分都是模板或css样式。 这个组件文件太大了。它大部分都是模板或css样式。
要想在HTML和CSS的噪音中找出组件的工作逻辑太难了。 要想在HTML和CSS的噪音中看清组件的工作逻辑太难了。
Let's migrate the template and the styles to their own files before we make any more changes: Let's migrate the template and the styles to their own files before we make any more changes:
在做更多修改之前,我们先把模板和样式移到它们自己的文件中去: 在做更多修改之前,我们先把模板和样式移到它们自己的文件中去:
1. *Cut-and-paste* the template contents into a new `heroes.component.html` file. 1. *Cut-and-paste* the template contents into a new `heroes.component.html` file.
1. *剪切并粘贴* 模板内容到新的`heroes.component.html`文件。 1. 把模板内容*剪切并粘贴*到新的`heroes.component.html`文件。
1. *Cut-and-paste* the styles contents into a new `heroes.component.css` file. 1. *Cut-and-paste* the styles contents into a new `heroes.component.css` file.
1. *剪切并粘贴* 样式内容到新的`heroes.component.css`文件。 1. 把样式内容*剪切并粘贴*到新的`heroes.component.css`文件。
1. *Set* the component metadata's `templateUrl` and `styleUrls` properties to refer to both files. 1. *Set* the component metadata's `templateUrl` and `styleUrls` properties to refer to both files.
1. *设置* 组件元数据的`templateUrl`和`styleUrls`属性,来分别引用这两个文件。 1. *设置*组件元数据的`templateUrl`和`styleUrls`属性,来分别引用这两个文件。
The revised component data looks like this: The revised component data looks like this:
@ -930,7 +931,7 @@ figure.image-display
1. 把`router`注入到构造函数中(就像`HeroService`那样) 1. 把`router`注入到构造函数中(就像`HeroService`那样)
1. Implement the `gotoDetail` method by calling the `router.navigate` method 1. Implement the `gotoDetail` method by calling the `router.navigate` method
with a two-part 'HeroDetail' *link parameters array*. with a two-part 'HeroDetail' *link parameters array*.
1. 实现`gotoDetail`方法:以`HeroDetail`和 *链接参数数组* 为参数调用`router.navigate`方法。 1. 实现`gotoDetail`方法:以`HeroDetail`和*链接参数数组*为参数调用`router.navigate`方法。
Here's the revised component class: Here's the revised component class:
@ -942,13 +943,13 @@ figure.image-display
for heroes list to the mini-detail to the hero details and back to the heroes again. for heroes list to the mini-detail to the hero details and back to the heroes again.
We can jump back and forth between the dashboard and the heroes. We can jump back and forth between the dashboard and the heroes.
刷新浏览器,并开始点击。 刷新浏览器,并开始点击。
我们能在应用中导航从仪表盘到英雄详情再回来从英雄列表到Mini版英雄详情到英雄详情然后再回到英雄列表。 我们能在应用中导航从仪表盘到英雄详情再回来从英雄列表到Mini版英雄详情到英雄详情再回到英雄列表。
我们可以在仪表盘和英雄列表之间跳来跳去。 我们可以在仪表盘和英雄列表之间跳来跳去。
We've met all of the navigational requirements that propelled this chapter. We've met all of the navigational requirements that propelled this chapter.
我们已经达成了本章最初设定的所有导航需求。 我们已经满足了在本章开头儿设定的所有导航需求。
.l-main-section .l-main-section
:marked :marked
@ -957,8 +958,8 @@ figure.image-display
The app is functional but pretty ugly. The app is functional but pretty ugly.
Our creative designer team provided some CSS files to make it look better. Our creative designer team provided some CSS files to make it look better.
应用的功能已经正常了,但还太丑。 应用在功能上已经正常了,但还太丑。
我们富有创意的设计师团队提供了一些CSS文件能让它好看一些。 我们富有创意的设计师团队提供了一些CSS文件能让它变得好看一些。
### A Dashboard with Style ### A Dashboard with Style
### 具有样式的仪表盘 ### 具有样式的仪表盘
@ -966,19 +967,19 @@ figure.image-display
They've given us ~60 lines of CSS for this purpose including some simple media queries for responsive design. They've given us ~60 lines of CSS for this purpose including some simple media queries for responsive design.
设计师认为我们应该把仪表盘的英雄们显示在一排方块中。 设计师认为我们应该把仪表盘的英雄们显示在一排方块中。
他们给了我们大约60行CSS来实现它包括一些简单的媒体查询语句实现响应式设计。 他们给了我们大约60行CSS来实现它包括一些简单的媒体查询语句实现响应式设计。
If we paste these ~60 lines into the component `styles` metadata, If we paste these ~60 lines into the component `styles` metadata,
they'll completely obscure the component logic. they'll completely obscure the component logic.
Let's not do that. It's easier to edit CSS in a separate `*.css` file anyway. Let's not do that. It's easier to edit CSS in a separate `*.css` file anyway.
如果我们把这60来行CSS粘贴到组件元数据的`styles`中,它们会完全淹没组件的工作逻辑。 如果我们把这60来行CSS粘贴到组件元数据的`styles`中,它们会完全淹没组件的工作逻辑。
不能么做。在一个独立的`*.css`文件中编辑CSS当然会更简单。 不能么做。在一个独立的`*.css`文件中编辑CSS当然会更简单。
Add a `dashboard.component.css` file to the `app` folder and reference Add a `dashboard.component.css` file to the `app` folder and reference
that file in the component metadata's `styleUrls` array property like this: that file in the component metadata's `styleUrls` array property like this:
把`dashboard.component.css`文件添加到`app`目录下,并在组件元数据的`styleUrls`数组属性中引用它。就像这样: 把`dashboard.component.css`文件添加到`app`目录下,并在组件元数据的`styleUrls`数组属性中引用它。就像这样:
+makeExample('toh-5/ts/app/dashboard.component.ts', 'css', 'app/dashboard.component.ts (styleUrls)')(format=".") +makeExample('toh-5/ts/app/dashboard.component.ts', 'css', 'app/dashboard.component.ts (styleUrls)')(format=".")
:marked :marked
.l-sub-section .l-sub-section
@ -989,7 +990,7 @@ figure.image-display
`styleUrls`属性是一个由样式文件的文件名(包括路径)组成的数组。 `styleUrls`属性是一个由样式文件的文件名(包括路径)组成的数组。
如果需要,我们还可以列出来自多个不同位置的样式文件。 如果需要,我们还可以列出来自多个不同位置的样式文件。
和`templateUrl`一样,我们必须指定 _相对于此应用根目录_ 的路径。 和`templateUrl`一样我们必须指定_相对于此应用根目录_的路径。
:marked :marked
### Stylish Hero Details ### Stylish Hero Details
### 美化英雄详情 ### 美化英雄详情
@ -1018,7 +1019,7 @@ figure.image-display
We cooperated by surrounding those links in `<nav>` tags. We cooperated by surrounding those links in `<nav>` tags.
设计师还给了我们一些CSS用于让`AppComponent`中的导航链接看起来更像可被选择的按钮。 设计师还给了我们一些CSS用于让`AppComponent`中的导航链接看起来更像可被选择的按钮。
想让它们协同工作,我们只需要把那些链接包含在`<nav>`标签中。 让它们协同工作,我们得把那些链接包含在`<nav>`标签中。
Add a `app.component.css` file to the `app` folder with the following content. Add a `app.component.css` file to the `app` folder with the following content.
@ -1033,8 +1034,8 @@ figure.image-display
The Angular Router adds the `router-link-active` class to the HTML navigation element The Angular Router adds the `router-link-active` class to the HTML navigation element
whose route matches the active route. All we have to do is define the style for it. Sweet! whose route matches the active route. All we have to do is define the style for it. Sweet!
Angular路由器会为与当前激活路由匹配的路由所在的导航元素加上`router-link-active`类。 Angular路由器会为当前活动路由所在的导航元素加上`router-link-active`类。
我们唯一要做的就是为它(`.router-link-active`类)定义样式。真好! 我们唯一要做的就是为它(`router-link-active`类)定义样式。真好!
:marked :marked
Set the `AppComponent`s `styleUrls` property to this CSS file. Set the `AppComponent`s `styleUrls` property to this CSS file.
@ -1048,16 +1049,16 @@ figure.image-display
It's pretty easy to package it all up and re-use the component somewhere else. It's pretty easy to package it all up and re-use the component somewhere else.
当我们把样式添加到组件中时,我们假定组件所需的一切 —— HTML、CSS、程序代码 —— 都在紧邻的地方。 当我们把样式添加到组件中时,我们假定组件所需的一切 —— HTML、CSS、程序代码 —— 都在紧邻的地方。
这样,把它们打包在一起并在别的组件中复用它都会很容易。 这样,无论是把它们打包在一起还是在别的组件中复用它都会很容易。
We can also create styles at the *application level* outside of any component. We can also create styles at the *application level* outside of any component.
我们也可以在应用中所有组件之外的地方创建样式。 我们也可以在所有组件之外创建*应用级*样式。
Our designers provided some basic styles to apply to elements across the entire app. Our designers provided some basic styles to apply to elements across the entire app.
Add the following to a new file named `styles.css` in the root folder. Add the following to a new file named `styles.css` in the root folder.
我们的设计师提供了一组基础样式,这些样式适用的元素横跨整个应用。 我们的设计师提供了一组基础样式,这些样式应用到的元素横跨整个应用。
把下面这个名叫`styles.css`的新文件添加到根目录中: 把下面这个名叫`styles.css`的新文件添加到根目录中:
+makeExample('toh-5/ts/styles.1.css', '', 'styles.css (应用级样式)')(format=".") +makeExample('toh-5/ts/styles.1.css', '', 'styles.css (应用级样式)')(format=".")
@ -1121,9 +1122,10 @@ figure.image-display
### The Road Behind ### The Road Behind
### 走过的路 ### 走过的路
We travelled a great distance in this chapter. We travelled a great distance in this chapter.
在本章中,我们往前走了很远: 在本章中,我们往前走了很远:
- We added the Angular *Component Router* to navigate among different components. - We added the Angular *Component Router* to navigate among different components.
- 我们添加了Angular *组件路由器* 在各个不同组件之间导航。 - 我们添加了Angular*组件路由器*在各个不同组件之间导航。
- We learned how to create router links to represent navigation menu items - We learned how to create router links to represent navigation menu items
- 我们学会了如何创建路由链接来表示导航栏的菜单项。 - 我们学会了如何创建路由链接来表示导航栏的菜单项。
- We used router parameters to navigate to the details of user selected hero - We used router parameters to navigate to the details of user selected hero
@ -1131,7 +1133,7 @@ figure.image-display
- We shared the `HeroService` among multiple components - We shared the `HeroService` among multiple components
- 我们在多个组件之间共享了`HeroService`服务。 - 我们在多个组件之间共享了`HeroService`服务。
- We moved HTML and CSS out of the component file and into their own files. - We moved HTML and CSS out of the component file and into their own files.
- 我们把HTML和CSS从组件中移出来放到它们自己的文件中 - 我们把HTML和CSS从组件中移出来放到它们自己的文件中。
- We added the `uppercase` pipe to format data - We added the `uppercase` pipe to format data
- 我们添加了一个`uppercase`管道,来格式化数据。 - 我们添加了一个`uppercase`管道,来格式化数据。
@ -1141,7 +1143,7 @@ figure.image-display
We're still missing a key piece: remote data access. We're still missing a key piece: remote data access.
我们有了很多用于构建应用的基石。 我们有了很多用于构建应用的基石。
但我们仍然缺少很关键的一部分:远程数据存取。 但我们仍然缺少很关键的一块儿:远程数据存取。
In a forthcoming tutorial chapter, In a forthcoming tutorial chapter,
well replace our mock data with data retrieved from a server using http. well replace our mock data with data retrieved from a server using http.