angular-cn/public/docs/ts/latest/tutorial/toh-pt5.jade

1151 lines
57 KiB
Plaintext
Raw Normal View History

include ../_util-fns
2015-12-23 12:42:57 -05:00
:marked
# Routing Around the App
# 应用中的路由
2015-12-23 12:42:57 -05:00
We received new requirements for our Tour of Heroes application:
我们收到了关于《英雄指南》应用的新需求:
2015-12-23 12:42:57 -05:00
* add a *Dashboard* view.
* 添加一个 *仪表盘* 视图。
2015-12-23 12:42:57 -05:00
* navigate between the *Heroes* and *Dashboard* views.
* 在 *英雄列表* 和 *仪表盘* 视图之间导航。
2015-12-23 12:42:57 -05:00
* clicking on a hero in either view navigates to a detail view of the selected hero.
* 无论在哪个视图中点击一个英雄,都会导航到该英雄的详情页。
2015-12-23 12:42:57 -05:00
* clicking a *deep link* in an email opens the detail view for a particular hero;
* 在邮件中点击一个 *深链接* ,会直接打开一个特定英雄的详情视图。
2015-12-23 12:42:57 -05:00
When were done, users will be able to navigate the app like this:
完成时,用户就能像这样浏览一个应用:
2015-12-23 12:42:57 -05:00
figure.image-display
img(src='/resources/images/devguide/toh/nav-diagram.png' alt="查看导航")
2015-12-23 12:42:57 -05:00
:marked
We'll add Angulars *Component Router* to our app to satisfy these requirements.
我们将把Angular *组件路由器* 加入应用中,以满足这些需求。(译注:硬件的路由器是用来让你找到另一台机器的,而这里的路由器用于帮你找到一个组件)
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
The [Routing and Navigation](../guide/router.html) chapter covers the router in more detail
than we will in this tour.
[路由与导航](../guide/router.html)一章覆盖了比该教程中更详细的路由知识。
2015-12-23 12:42:57 -05:00
:marked
[Run the live example](/resources/live-examples/toh-5/ts/plnkr.html).
[运行鲜活范例](/resources/live-examples/toh-5/ts/plnkr.html).
2015-12-23 12:42:57 -05:00
.l-sub-section
img(src='/resources/images/devguide/plunker-separate-window-button.png' alt="弹出窗口" align="right" style="margin-right:-20px")
2015-12-23 12:42:57 -05:00
:marked
To see the URL changes in the browser address bar,
pop out the preview window by clicking the blue 'X' button in the upper right corner:
2015-12-23 12:42:57 -05:00
注意看浏览器地址栏中的URL变化点击右上角的蓝色'X'按钮,弹出预览窗口。
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Where We Left Off
## 我们在哪儿
2015-12-23 12:42:57 -05:00
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.
在继续《英雄指南》之前,我们先检查一下,在添加了英雄服务和英雄详情组件之后,你是否已经有了如下目录结构。如果没有,你得先回上一章,再照做一次。
2015-12-23 12:42:57 -05:00
.filetree
.file angular2-tour-of-heroes
.children
.file app
.children
.file app.component.ts
.file hero.ts
.file hero-detail.component.ts
.file hero.service.ts
.file main.ts
.file mock-heroes.ts
.file node_modules ...
.file typings ...
2015-12-23 12:42:57 -05:00
.file index.html
.file package.json
2016-04-27 14:28:22 -04:00
.file styles.css
.file systemjs.config.js
2015-12-23 12:42:57 -05:00
.file tsconfig.json
.file typings.json
2015-12-23 12:42:57 -05:00
:marked
### Keep the app transpiling and running
### 让应用代码保持转译和运行
2015-12-23 12:42:57 -05:00
Open a terminal/console window and enter the following command to
start the TypeScript compiler, start the server, and watch for changes:
打开terminal/console窗口运行下列命令启动TypeScript编译器它会监视文件变更并启动开发服务器
2015-12-23 12:42:57 -05:00
code-example(format="." language="bash").
npm start
:marked
The application runs and updates automatically as we continue to build the Tour of Heroes.
我们继续构建《英雄指南》,应用也会保持运行并自动更新。
2015-12-23 12:42:57 -05:00
## Action plan
## 行动计划
2015-12-23 12:42:57 -05:00
Here's our plan
下面是我们的计划
2015-12-23 12:42:57 -05:00
* turn `AppComponent` into an application shell that only handles navigation.
* 把`AppComponent`变成应用的一个“壳层”,它只处理导航。
2015-12-23 12:42:57 -05:00
* relocate the *Heroes* concerns within the current `AppComponent` to a separate `HeroesComponent`
* 把现在`AppComponent`所关注的这些事移到一个独立的`HeroesComponent`中
2015-12-23 12:42:57 -05:00
* add routing
* 添加路由
2015-12-23 12:42:57 -05:00
* create a new `DashboardComponent`
* 添加一个新的`DashboardComponent`组件
2015-12-23 12:42:57 -05:00
* tie the *Dashboard* into the navigation structure.
* 把 *仪表盘* 加入导航结构中。
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
*Routing* is another name for *navigation*. The *router* is the mechanism for navigating from view to view.
*路由* 是导航的另一个名字。 *路由器* 就是从一个视图导航到另一个视图的机制。
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Splitting the *AppComponent*
## 拆分 *AppComponent*
2015-12-23 12:42:57 -05:00
Our current app loads `AppComponent` and immediately displays the list of heroes.
现在的应用会加载`AppComponent`组件,并且立即显示出英雄列表。
2015-12-23 12:42:57 -05:00
Our revised app should present a shell with a choice of views (*Dashboard* and *Heroes*) and then default to one of them.
我们修改过的应用将提供一个壳,它会从 *仪表盘* 和 *英雄列表* 中选择一个视图,然后默认转到其中之一。
2015-12-23 12:42:57 -05:00
The `AppComponent` should only handle navigation.
Let's move the display of *Heroes* out of `AppComponent` and into its own `HeroesComponent`.
`AppComponent`组件应该只处理导航。
我们来把 *英雄列表* 的显示职责,从`AppComponent`移到`HeroesComponent`组件中。
2015-12-23 12:42:57 -05:00
### *HeroesComponent*
### *HeroesComponent*
2015-12-23 12:42:57 -05:00
`AppComponent` is already dedicated to *Heroes*.
Instead of moving anything out of `AppComponent`, we'll just rename it `HeroesComponent`
and create a new `AppComponent` shell separately.
`AppComponent`已经移交给`HeroesComponent`了。
与其把`AppComponent`中的所有东西都移过去,不如把它改名为`HeroesComponent`,并且单独创建一个新的`AppComponent`壳。
2015-12-23 12:42:57 -05:00
The steps are:
步骤如下:
2015-12-23 12:42:57 -05:00
* rename `app.component.ts` file to `heroes.component.ts`.
* 把`app.component.ts`文件改名为`heroes.component.ts`。
2015-12-23 12:42:57 -05:00
* rename the `AppComponent` class to `HeroesComponent`.
* 把`AppComponent`类改名为`HeroesComponent`。
2015-12-23 12:42:57 -05:00
* rename the selector `my-app` to `my-heroes`.
* 把`my-app`选择器改名为`my-heroes`。
2015-12-23 12:42:57 -05:00
:marked
+makeExample('toh-5/ts/app/heroes.component.ts', 'heroes-component-renaming', 'app/heroes.component.ts (改名)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
## Create *AppComponent*
## 创建 *AppComponent*
2015-12-23 12:42:57 -05:00
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.
新的`AppComponent`将成为应用的“壳”。
它将在顶部放一些导航链接,并且把我们要导航到的页面放在下面的区域。
2015-12-23 12:42:57 -05:00
The initial steps are:
这些起始步骤是:
2015-12-23 12:42:57 -05:00
* create a new file named `app.component.ts`.
* 创建一个名叫`app.component.ts`的新文件。
2015-12-23 12:42:57 -05:00
* define an `AppComponent` class.
* 定义一个`AppComponent`类。
2015-12-23 12:42:57 -05:00
* `export` it so we can reference it during bootstrapping in `main.ts`.
* `export`它,以便我们能在`main.ts`的启动期间引用它。
2015-12-23 12:42:57 -05:00
* expose an application `title` property.
* 导出应用的`title`属性。
2015-12-23 12:42:57 -05:00
* add the `@Component` metadata decorator above the class with a `my-app` selector.
* 在类的顶部添加`@Component`元数据装饰器,其中指定`my-app`选择器。
2015-12-23 12:42:57 -05:00
* add a template with `<h1>` tags surrounding a binding to the `title` property.
* 在模板中添加一个`<h1>`标签,其中是到`title`属性的绑定。
2015-12-23 12:42:57 -05:00
* add the `<my-heroes>` tags to the template so we still see the heroes.
* 在模板中添加`<my-heroes>`标签,以便我们仍然能看到英雄列表。
2015-12-23 12:42:57 -05:00
* add the `HeroesComponent` to the `directives` array so Angular recognizes the `<my-heroes>` tags.
* 添加`HeroesComponent`组件到`directives`数组中以便Angular能认识`<my-heroes>`标签。
2015-12-23 12:42:57 -05:00
* add the `HeroService` to the `providers` array because we'll need it in every other view.
* 添加`HeroService`到`providers`数组中,因为我们的每一个视图都需要它。
2015-12-23 12:42:57 -05:00
* add the supporting `import` statements.
* 添加支援性的`import`语句。
2015-12-23 12:42:57 -05:00
Our first draft looks like this:
我们的第一个草稿就像这样:
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/app.component.1.ts', null, 'app/app.component.ts (v1)')
:marked
.callout.is-critical
header Remove <i>HeroService</i> from the <i>HeroesComponent</i> providers
header 从<i>HeroesComponent</i>的`providers`中移除<i>HeroService</i>
2015-12-23 12:42:57 -05:00
:marked
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 ***do not want two copies*** of this service at two different levels of our app.
回到`HeroesComponent`,并从`providers`数组中 **移除`HeroService`** 。
我们要把它从`AppComponent` *晋升* 到`HeroesComponent`中。
我们不希望在应用的两个不同层次上存在它的 ***两个副本*** 。
2015-12-23 12:42:57 -05:00
:marked
The app still runs and still displays heroes.
Our refactoring of `AppComponent` into a new `AppComponent` and a `HeroesComponent` worked!
We have done no harm.
应用仍然在运行,并且显示着英雄列表。
我们把`AppComponent`重构成了一个新的`AppComponent`和`HeroesComponent`,它们工作得很好!
我们毫发无伤的完成了重构。
2015-12-23 12:42:57 -05:00
:marked
## Add Routing
## 添加路由
2015-12-23 12:42:57 -05:00
We're ready to take the next step.
Instead of displaying heroes automatically, we'd like to show them *after* the user clicks a button.
In other words, we'd like to navigate to the list of heroes.
我们已准备好开始下一步。
与其自动显示英雄列表,我们更希望在用户点击按钮之后显示它。
换句话说,我们希望通过导航显示英雄列表。
2015-12-23 12:42:57 -05:00
We'll need the Angular *Component Router*.
我们需要Angular的 *组件路由器* 。
2016-04-27 14:28:22 -04:00
### Set the base tag
### 设置base标签
2016-04-27 14:28:22 -04:00
Open the `index.html` and add `<base href="/">` at the top of the `<head>` section.
打开`index.html`并且在`<head>`区的顶部添加`<base href="/">`语句。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/index.html', 'base-href', 'index.html (base href)')(format=".")
.callout.is-important
header base href is essential
2016-04-23 02:54:01 -04:00
header base href是不可或缺的
2015-12-23 12:42:57 -05:00
:marked
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* 部分,了解为何如此。
2015-12-23 12:42:57 -05:00
:marked
### Make the router available.
### 让路由可用。
2015-12-23 12:42:57 -05:00
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.
*组件路由器* 是一个服务。像所有服务一样,我们得导入它,并且通过把它加入`providers`数组来让它在应用中可用。
2015-12-23 12:42:57 -05:00
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:
Angular路由器是由多个服务(`ROUTER_PROVIDERS`)和多个指令(`ROUTER_DIRECTIVES`)以及一个配置装饰器(`RouteConfig`)组成的。我们一次性导入它们。
+makeExample('toh-5/ts/app/app.component.2.ts', 'import-router', 'app.component.ts (导入router)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Next we update the `directives` and `providers` metadata arrays to *include* the router assets.
接下来,我们我们更新`directives`和`providers`元数据数组,来包含这些路由器部件。
+makeExample('toh-5/ts/app/app.component.2.ts', 'directives-and-providers', 'app.component.ts (directives和providers)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Notice that we also removed the `HeroesComponent` from the `directives` array.
`AppComponent` no longer shows heroes; that will be the router's job.
We'll soon remove `<my-heroes>` from the template too.
注意,我们已经从`directives`数组中移除了`HeroesComponent`。`AppComponent`不会再显示英雄,那是路由器的工作。
我们马上也会从模板中移除`<my-heroes>`。
2015-12-23 12:42:57 -05:00
### Add and configure the router
### 添加与配置路由器
2015-12-23 12:42:57 -05:00
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*.
`AppComponent`还没有路由器。我们使用`@RouteConfig`装饰器来同时 (a)为组件指定一个路由器,并 (b) 通过 *routes* 来配置路由器。
2015-12-23 12:42:57 -05:00
*Routes* tell the router which views to display when a user clicks a link or
pastes a URL into the browser address bar.
*routes* 告诉路由器当用户点击链接或者把URL粘贴到浏览器地址栏时应该显示哪个路由。
2015-12-23 12:42:57 -05:00
Let's define our first route, a route to the `HeroesComponent`.
我们来定义第一个路由 —— 到`HeroesComponent`的路由。
+makeExample('toh-5/ts/app/app.component.2.ts', 'route-config', 'app.component.ts (英雄列表的RouteConfig)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
`@RouteConfig` takes an array of *route definitions*.
We have only one route definition at the moment but rest assured, we'll add more.
`@RouteConfig`有一个 *路由定义* 数组。
此刻我们只有一个路由定义,但别急,我们还会添加更多。
2015-12-23 12:42:57 -05:00
This *route definition* has three parts:
“路由定义”包括三个部分:
2015-12-23 12:42:57 -05:00
* **path**: the router matches this route's path to the URL in the browser address bar (`/heroes`).
* **path**: 路由器会用它来匹配路由中的路径和浏览器地址栏中的路径,如`/heroes`。
2015-12-23 12:42:57 -05:00
* **name**: the official name of the route; it *must* begin with a capital letter to avoid confusion with the *path* (`Heroes`).
* **name**: 路由的正式名字,它必须以大写字母开头儿,以免和 *path* 混淆,如`Heroes`。
2015-12-23 12:42:57 -05:00
* **component**: the component that the router should create when navigating to this route (`HeroesComponent`).
* **component**: 当导航到此路由时,路由器需要创建的组件,如`HeroesComponent`。
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
Learn more about defining routes with @RouteConfig in the [Routing](../guide/router.html) chapter.
要学习更多使用`@RouteConfig`定义路由的知识,请参见[路由](../guide/router.html)一章。
2015-12-23 12:42:57 -05:00
:marked
### Router Outlet
### 路由插座Outlet
2015-12-23 12:42:57 -05:00
If we paste the path, `/heroes`, into the browser address bar,
the router should match it to the `'Heroes'` route and display the `HeroesComponent`.
But where?
如果我们把路径`/heroes`粘贴到浏览器的地址栏中,路由器会匹配到`'Heroes'`路由,并显示`HeroesComponent`组件。
但问题是,把它显示在哪儿呢?
2015-12-23 12:42:57 -05:00
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`.
The router displays each component immediately below the `<router-outlet>` as we navigate through the application.
我们必须 ***告诉它位置*** ,所以我们把`<router-outlet>`标签添加到模板的底部。
`RouterOutlet`是`ROUTER_DIRECTIVES`常量中的一员。
当我们通过应用导航过来时,路由器立即把每个组件显示在`<router-outlet>`的位置。
2015-12-23 12:42:57 -05:00
### Router Links
### 路由器链接
2015-12-23 12:42:57 -05:00
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`.
我们不可能真等用户把路由的URL粘贴到地址栏中我们应该在模板中的什么地方添加一个A链接标签点击时就会触发导航到`HeroesComponent`组件的操作。
2015-12-23 12:42:57 -05:00
The revised template looks like this:
修改过的模板是这样的:
+makeExample('toh-5/ts/app/app.component.2.ts', 'template', 'app.component.ts (英雄列表模板)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Notice the `[routerLink]` binding in the anchor tag.
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.
注意A标签中的`[routerLink]`绑定。我们把`RouterLink`指令(`ROUTER_DIRECTIVES`中的另一个指令)绑定到一个数组,它将告诉路由器,当用户点击这个链接时,应该导航到那里。
2015-12-23 12:42:57 -05:00
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.
Looking back at the route configuration, we confirm that `'Heroes'` is the name of the route to the `HeroesComponent`.
我们通过一个 *链接参数数组* 定义了一个 *路由说明* 。
在我们这个小例子中,该数组只有一个元素,一个放在引号中的 **路由名称** ,用作路标。
回来看路由配置表,我们清楚地看到,这个名称 —— `'Heroes'` 就是指向`HeroesComponent`的那个路由的名称。
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
Learn about the *link parameters array* in the [Routing](../guide/router.html#link-parameters-array) chapter.
学习关于 *连接参数数组* 的更多知识,参见[路由](../guide/router.html#link-parameters-array)一章。
2015-12-23 12:42:57 -05:00
:marked
Refresh the browser. We see only the app title. We don't see the heroes list.
刷新浏览器。我们只看到了应用标题。英雄列表到哪里去了?
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
The browser's address bar shows `/`.
The route path to `HeroesComponent` is `/heroes`, not `/`.
We don't have a route that matches the path `/`, so there is nothing to show.
That's something we'll want to fix.
浏览器的地址栏显示的是`/`。到`HeroesComponent`的路由中的路径是`/`,所以,自然没啥可显示的。
接下来我们要修复这个问题。
2015-12-23 12:42:57 -05:00
:marked
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!
我们点击“英雄列表Heroes”导航链接浏览器地址栏更新为`/heroes`,并且看到了英雄列表。我们终于导航过去了!
2015-12-23 12:42:57 -05:00
At this stage, our `AppComponent` looks like this.
在这个阶段,`AppComponent`看起来是这样的:
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/app.component.2.ts',null, 'app/app.component.ts (v2)')
:marked
The *AppComponent* is now attached to a router and displaying routed views.
For this reason and to distinguish it from other kinds of components,
we call this type of component a *Router Component*.
*AppComponent* 现在有了一个路由器,并且能显示路由到的视图。
因此,为了把它从其他种类的组件中区分出来,我们称这类组件为 *路由器组件*。
2015-12-23 12:42:57 -05:00
:marked
## Add a *Dashboard*
## 添加一个 *仪表盘*
2015-12-23 12:42:57 -05:00
Routing only makes sense when we have multiple views. We need another view.
只有在我们有多个视图的时候,路由才有意义。我们需要另一个视图。
2015-12-23 12:42:57 -05:00
Create a placeholder `DashboardComponent` that gives us something to navigate to and from.
先创建一个`DashboardComponent`的占位符,让我们可以导航到它或导航自它。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/dashboard.component.1.ts',null, 'app/dashboard.component.ts (v1)')(format=".")
:marked
Well come back and make it more useful later.
我们先不实现它,稍后,我们再回来,让这个组件更有用。
2015-12-23 12:42:57 -05:00
### Configure the dashboard route
### 配置仪表盘路由
2015-12-23 12:42:57 -05:00
Go back to `app.component.ts` and teach it to navigate to the dashboard.
回到`app.component.ts`文件,教它如何导航到这个仪表盘。
2015-12-23 12:42:57 -05:00
Import the `DashboardComponent` so we can reference it in the dashboard route definition.
导入`DashboardComponent`类,以便我们可以在仪表盘的路由定义中引用它。
2015-12-23 12:42:57 -05:00
Add the following `'Dashboard'` route definition to the `@RouteConfig` array of definitions.
把下列`'Dashboard'`路由的定义添加到`@RouteConfig`数组中去。
+makeExample('toh-5/ts/app/app.component.ts','dashboard-route', 'app.component.ts (仪表盘路由)')(format=".")
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
**useAsDefault**
We want the app to show the dashboard when it starts and
we want to see a nice URL in the browser address bar that says `/dashboard`.
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.
我们希望应用在启动的时候就显示仪表盘并且我们希望在浏览器的地址栏看到一个好看的URL比如`/dashboard`。
记住,浏览器启动时,在地址栏中使用的路径是`/`。我们没有指向这个路径的路由,也不想创建它。
2015-12-23 12:42:57 -05:00
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.
幸运的是,我们可以把`useAsDefault: true`属性添加到 *路由定义* 上。这样如果浏览器中的URL匹配不上任何一个已知路由那么路由器将显示这个仪表盘组件。
2015-12-23 12:42:57 -05:00
:marked
Finally, add a dashboard navigation link to the template, just above the *Heroes* link.
最后,在模板上添加一个到仪表盘的导航链接,就在 *英雄(Heroes)* 链接的上方。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/app.component.ts','template', 'app.component.ts (模板)')(format=".")
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
We nestled the two links within `<nav>` tags.
They don't do anything yet but they'll be convenient when we style the links a little later in the chapter.
我们在`<nav>`标签中放了两个链接。
它们现在还没有作用,但稍后,当我们对这些链接添加样式时,会显得比较方便。
2015-12-23 12:42:57 -05:00
:marked
Refresh the browser. The app displays the dashboard and
we can navigate between the dashboard and the heroes.
刷新浏览器。应用显示出了仪表盘,并且我们可以在仪表盘和英雄列表之间导航了。
2015-12-23 12:42:57 -05:00
## Dashboard Top Heroes
## 仪表盘上的顶尖英雄
2015-12-23 12:42:57 -05:00
Lets spice up the dashboard by displaying the top four heroes at a glance.
我们要让仪表盘更有趣,比如:一眼就能看到四个顶尖英雄。
2015-12-23 12:42:57 -05:00
Replace the `template` metadata with a `templateUrl` property that points to a new
template file.
2015-12-23 12:42:57 -05:00
把元数据中的`template`属性替换为`templateUrl`属性,它将指向一个新的模板文件。
+makeExample('toh-5/ts/app/dashboard.component.ts', 'template-url', 'app/dashboard.component.ts (templateUrl)')(format=".")
.l-sub-section
:marked
We specify the path _all the way back to the application root_. Angular doesn't support module-relative paths.
我们指定的所有路径都是相对于该应用的根目录的。Angular不支持使用相对于当前模块的路径。
:marked
2015-12-23 12:42:57 -05:00
Create that file with these contents:
使用下列内容创建文件:
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/dashboard.component.html', null, 'dashboard.component.html')(format=".")
:marked
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.
我们又一次使用`*ngFor`来在英雄列表上迭代,并且显示他们的名字。
我们添加了一个额外的`<div>`元素,来帮助稍后的美化工作。
2015-12-23 12:42:57 -05:00
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 have work to do, starting with those heroes.
这里的`(click)`绑定到了`gotoDetail`方法,但我们还没实现它。并且我们也没有要显示的英雄列表数据。
我们有活儿干了,就从那些英雄列表开始吧。
2015-12-23 12:42:57 -05:00
### Share the *HeroService*
### 共享 *HeroService*
2015-12-23 12:42:57 -05:00
We'd like to re-use the `HeroService` to populate the component's `heroes` array.
我们想要重用`HeroService`来存放组件的`heroes`数组。
2015-12-23 12:42:57 -05:00
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`.
回忆一下,在前面的章节中,我们从`HeroesComponent`的`providers`数组中移除了`HeroService`服务,并且把它添加到了顶级组件`AppComponent`的`providers`数组中。
2015-12-23 12:42:57 -05:00
That move created a singleton `HeroService` instance, available to *all* components of the application.
We'll inject and use it here in the `DashboardComponent` .
这个改动创建了一个`HeroService`的单例对象,并且对应用中的 *所有* 组件都有效。
在`DashboardComponent`组件中,我们将把它注入进来,并使用它。
2015-12-23 12:42:57 -05:00
### Get heroes
### 获取英雄数组
2015-12-23 12:42:57 -05:00
Open the `dashboard.component.ts` and add the requisite `import` statements.
打开`dashboard.component.ts`文件,并且把必备的`import`语句加进去。
+makeExample('toh-5/ts/app/dashboard.component.2.ts','imports', 'app/dashboard.component.ts (导入)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
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.
我们得实现`OnInit`接口,因为我们要在`ngOnInit`方法中初始化英雄数组 —— 就像上次一样。
我们需要导入`Hero`和`HeroService`类来引用它们的数据类型。
2015-12-23 12:42:57 -05:00
Now implement the `DashboardComponent` class like this:
现在就来实现`DashboardComponent`类,像这样:
+makeExample('toh-5/ts/app/dashboard.component.2.ts','component', 'app/dashboard.component.ts (类)')
2015-12-23 12:42:57 -05:00
:marked
We saw this kind of logic before in the `HeroesComponent`.
* create a `heroes` array property
* inject the `HeroService` in the constructor and hold it in a private `_heroService` field.
* call the service to get heroes inside the Angular `ngOnInit` lifecycle hook.
在`HeroesComponent`之前,我们也看到过类似的逻辑:
* 创建一个`heroes`数组属性
* 把`HeroService`注入构造函数中,并且把它保存在一个私有的`_heroService`字段中。
* 在Angular的`ngOnInit`生命周期钩子中调用这个服务,并且取得英雄列表。
2015-12-23 12:42:57 -05:00
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.
不同之处在于:我们使用 *slice* 函数挑选四个英雄第2、3、4、5个并且先为`gotoDetail`方法提供桩实现 —— 直到我们准备实现它。
2015-12-23 12:42:57 -05:00
Refresh the browser and see four heroes in the new dashboard.
刷新浏览器,并且在新的仪表盘中看到了四个英雄。
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Navigate to Hero Details
## 导航到英雄详情
2015-12-23 12:42:57 -05:00
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:
虽然我们在`HeroesComponent`组件的底部显示了所选英雄的详情,但我们还从没有 *导航* 到`HeroDetailComponent`组件,不管用我们需求中指定的三种方式中的哪一种:
1. from the *Dashboard* to a selected hero.
1. 从 *仪表盘(Dashboard)* 导航到一个选定的英雄。
1. from the *Heroes* list to a selected hero.
1. 从 *英雄列表(Heroes)* 导航到一个选定的英雄。
1. from a "deep link" URL pasted into the browser address bar.
1. 把一个指向他的"深链接"URL粘贴到浏览器的地址栏。
2015-12-23 12:42:57 -05:00
Adding a `'HeroDetail'` route seem an obvious place to start.
添加`'HeroDetail'`路由,是一个显而易见的起点。
2015-12-23 12:42:57 -05:00
### Routing to a hero detail
### 路由到一个英雄详情
2015-12-23 12:42:57 -05:00
We'll add a route to the `HeroDetailComponent` in the `AppComponent` where our other routes are configured.
我们将在`AppComponent`中添加一个到`HeroDetailComponent`的路由,那也是配置其它路由的地方。
2015-12-23 12:42:57 -05:00
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.
新路由的不寻常之处在于,我们必须告诉`HeroDetailComponent` *该显示哪个英雄* 。
以前的`HeroesComponent`组件和`DashboardComponent`组件都还不曾要求我们告诉它任何东西。
2015-12-23 12:42:57 -05:00
At the moment the parent `HeroesComponent` sets the component's `hero` property to a hero object with a binding like this.
现在,父组件`HeroesComponent`通过数据绑定来把一个英雄对象设置为组件的`hero`属性。就像这样:
2015-12-23 12:42:57 -05:00
code-example(format='').
&lt;my-hero-detail [hero]="selectedHero">&lt;/my-hero-detail>
:marked
That clearly won't work in any of our routing scenarios.
Certainly not the last one; we can't embed an entire hero object in the URL! Nor would we want to.
显然,在我们的任何一个路由场景中它都无法工作。
不仅如此我们也没法把一个完整的hero对象嵌入到URL中其实我们也不想这样。
2015-12-23 12:42:57 -05:00
### Parameterized route
### 参数化路由
2015-12-23 12:42:57 -05:00
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是这样的
2015-12-23 12:42:57 -05:00
code-example(format='').
/detail/11
:marked
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`.
URL中的`/detail/`部分是不变的。结尾的数字`id`部分会随着英雄的不同而变化。
我们要把路由中可变的那部分表示成一个 *参数(parameter)* 或 *标识(token)* ,用以获取英雄的`id`。
2015-12-23 12:42:57 -05:00
### Configure a Route with a Parameter
### 通过参数来配置路由
2015-12-23 12:42:57 -05:00
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=".")
2015-12-23 12:42:57 -05:00
:marked
The colon (:) in the path indicates that `:id` is a placeholder to be filled with a specific hero `id`
when navigating to the `HeroDetailComponent`.
路径中的冒号(:)标记`:id`是一个占位符,当导航到这个`HeroDetailComponent`组件时,它将被填成一个特定英雄的`id`。
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
Of course we have to import the `HeroDetailComponent` before we create this route:
当然,在创建这个路由之前,我们必须导入`HeroDetailComponent`类。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/app.component.ts','hero-detail-import')(format=".")
:marked
We're finished with the `AppComponent`.
我们正在完成`AppComponent`组件。
2015-12-23 12:42:57 -05:00
We won't add a `'Hero Detail'` link to the template because users
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.
我们没有往模板中添加一个`'英雄详情'`,这是因为用户不会直接点击导航栏中的链接去查看一个特定的英雄。
他们只会从英雄列表或者仪表盘中点击一个英雄。
2015-12-23 12:42:57 -05:00
We'll get to those *hero* clicks later in the chapter.
There's no point in working on them until the `HeroDetailComponent`
is ready to be navigated *to*.
稍后我们会响应这些 *英雄* 的点击事件。
现在对它们做什么都没有意义 —— 除非`HeroDetailComponent`已经做好了,并且能够被导航过去。
2015-12-23 12:42:57 -05:00
That will require an `HeroDetailComponent` overhaul.
那需要对`HeroDetailComponent`做一次大修。
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Revise the *HeroDetailComponent*
## 修改 *HeroDetailComponent*
2015-12-23 12:42:57 -05:00
Before we rewrite the `HeroDetailComponent`, let's remember what it looks like now:
在重写`HeroDetailComponent`之前,我们先记住它现在的样子:
+makeExample('toh-4/ts/app/hero-detail.component.ts', null, 'app/hero-detail.component.ts (当前)')
2015-12-23 12:42:57 -05:00
: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.
模板不用修改。我们会用跟以前一样的方式显示英雄。导致这次大修的原因是如何获得这个英雄的数据。
2015-12-23 12:42:57 -05:00
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
and use the `HeroService` to fetch the hero with that `id` from storage.
我们不会再从父组件的属性绑定中取得英雄数据。
新的`HeroDetailComponent`应该从路由器的`RouteParams`服务取得`id`参数,并通过`HeroService`服务获取具有这个指定`id`的英雄数据。
2015-12-23 12:42:57 -05:00
We need an import statement to reference the `RouteParams`.
我们需要一个import语句来引用`RouteParams`。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'import-route-params')(format=".")
:marked
We import the `HeroService`so we can fetch a hero`.
我们导入了`HeroService`,现在我们能获取一个英雄了。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'import-hero-service')(format=".")
:marked
We import the `OnInit` interface because we'll call the `HeroService` inside the `ngOnInit` component lifecycle hook.
我们要导入`OnInit`接口,是因为我们需要在组件的`ngOnInit`生命周期钩子中调用`HeroService`。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'import-oninit')(format=".")
:marked
We inject the both the `RouteParams` service and the `HeroService` into the constructor as we've done before,
making private variables for both:
像以前一样,我们把`RouteParams`服务和`HeroService`服务注入到构造函数中,让它们成为私有变量。
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'ctor', 'app/hero-detail.component.ts (构造函数)')(format=".")
:marked
We tell the class that we want to implement the `OnInit` interface.
我们告诉这个类,我们要实现`OnInit`接口。
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'implement')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Inside the `ngOnInit` lifecycle hook, extract the `id` parameter value from the `RouteParams` service
and use the `HeroService` to fetch the hero with that `id`.
在`ngOnInit`生命周期钩子中,从`RouteParams`服务中提取`id`参数,并且使用`HeroService`来获得具有这个`id`的英雄数据。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'ng-oninit', 'app/hero-detail.component.ts (ngOnInit)')(format=".")
:marked
Notice how we extract the `id` by calling the `RouteParams.get` method.
注意我们是如何通过调用`RouteParams.get`方法来获取`id`的。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'get-id')(format=".")
:marked
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.
英雄的`id`是数字。路由参数 *永远是字符串* 。
所以我们需要通过JavaScript的(+)操作符把路由参数的值转成数字。
2015-12-23 12:42:57 -05:00
### Add *HeroService.getHero*
### 添加 *HeroService.getHero*
2015-12-23 12:42:57 -05:00
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.
这些代码的问题在于`HeroService`并没有一个叫`getHero`的方法,我们最好在有人通知我们应用坏了之前赶快修复它。
2015-12-23 12:42:57 -05:00
Open `HeroService` and add the `getHero` method. It's trivial given that we're still faking data access:
打开`HeroService`,并且添加`getHero`方法进去。对于我们的假数据存取逻辑来说,这点修改是微不足道的:
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero.service.ts', 'get-hero', 'app/hero.service.ts (getHero)')(format=".")
:marked
Return to the `HeroDetailComponent` to clean up loose ends.
回到`HeroDetailComponent`来完成收尾工作。
2015-12-23 12:42:57 -05:00
### Find our way back
### 回到原路
2015-12-23 12:42:57 -05:00
We can navigate *to* the `HeroDetailComponent` in several ways.
How do we navigate somewhere else when we're done?
我们能用多种方式导航 *到* `HeroDetailComponent`。
但当我们完工时,我们该导航到那里呢?
2015-12-23 12:42:57 -05:00
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
用户可以点击`AppComponent`中的两个链接,或者点击浏览器的“后退”按钮。
我们来添加第三个选项:一个`goBack`方法,来根据浏览器的历史堆栈,往回退一步。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'go-back', 'app/hero-detail.component.ts (goBack)')(format=".")
.l-sub-section
:marked
Going back too far could take us out of the 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).
2016-04-18 10:59:24 -04:00
回退太多步儿会离开我们的应用。
在Demo中这算不上问题。但在真实的应用中我们需要对此进行防范。
或许你该用[*routerCanDeactivate* 钩子](/docs/ts/latest/api/router/CanDeactivate-interface.html)。
2015-12-23 12:42:57 -05:00
:marked
Then we wire this method with an event binding to a *Back* button that we add to the bottom of the component template.
2016-04-18 10:59:24 -04:00
然后,我们通过一个事件绑定把此方法绑定到模板底部的 *后退(Back)* 按钮上。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.html', 'back-button')(format=".")
:marked
Modifing the template to add this button spurs us to take one more incremental improvement and migrate the template to its own file
called `hero-detail.component.html`
2016-04-18 10:59:24 -04:00
修改模板,添加这个按钮以提醒我们还要做更多的改进,并把模板移到独立的`hero-detail.component.html`文件中去。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.html', '', 'app/hero-detail.component.html')(format=".")
:marked
We update the component metadata with a `templateUrl` pointing to the template file that we just created.
2016-04-18 10:59:24 -04:00
然后更新组件的元数据,用一个`templateUrl`属性指向我们刚刚创建的模板文件。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'template-url', 'app/hero-detail.component.ts (templateUrl)')(format=".")
:marked
Here's the (nearly) finished `HeroDetailComponent`:
2016-04-18 10:59:24 -04:00
下面是(几乎)完成的`HeroDetailComponent`
+makeExample('toh-5/ts/app/hero-detail.component.ts', 'v2', 'app/hero-detail.component.ts (最新版)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
.l-main-section
:marked
## Select a *Dashboard* Hero
2016-04-18 10:59:24 -04:00
## 选择一个 *仪表盘* 中的英雄
2015-12-23 12:42:57 -05:00
When a user selects a hero in the dashboard, the app should navigate to the `HeroDetailComponent` to view and edit the selected hero..
2016-04-18 10:59:24 -04:00
当用户从仪表盘中选择了一位英雄时,应用应该导航到`HeroDetailComponent`以查看和编辑所选的英雄。
2015-12-23 12:42:57 -05:00
In the dashboard template we bound each hero's click event to the `gotoDetail` method, passing along the selected `hero` entity.
2016-04-18 10:59:24 -04:00
在仪表盘模板中我们把每个英雄的click事件都绑定成`gotoDetail`方法,并且传入选中的这个`hero`实体对象。
+makeExample('toh-5/ts/app/dashboard.component.html','click', 'app/dashboard.component.html (click绑定)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
We stubbed the `gotoDetail` method when we rewrote the `DashboardComponent`.
Now we give it a real implementation.
2016-04-18 10:59:24 -04:00
当初我们重写`DashboardComponent`的时候,`gotoDetail`还是一个“桩方法”。
现在,我们给它一个真正的实现。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/dashboard.component.ts','goto-detail', 'app/dashboard.component.ts (gotoDetail)')(format=".")
:marked
The `gotoDetail` method navigates in two steps:
2016-04-18 10:59:24 -04:00
`gotoDetail`方法分两步完成导航:
1. set a route *link parameters array*
2016-04-18 10:59:24 -04:00
1. 生成路由的 *链接参数数组*
1. pass the array to the router's navigate method.
2016-04-18 10:59:24 -04:00
1. 把这个数组传给路由器的navigate方法。
2015-12-23 12:42:57 -05:00
We wrote *link parameters arrays* in the `AppComponent` for the navigation links.
Those arrays had only one element, the name of the destination route.
2016-04-18 10:59:24 -04:00
我们当初在`AppComponent`中生成导航链接的时候曾经写过 *链接参数数组* 。
2015-12-23 12:42:57 -05:00
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`.
2016-04-18 10:59:24 -04:00
这个数组有两个元素,目标路由的 ***名称(name)*** ,和一个 ***路由参数对象*** ,其中包括一个`id`字段,它的取值是所选英雄的`id`。
2015-12-23 12:42:57 -05:00
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.
2016-04-18 10:59:24 -04:00
当初我们在`AppComponent`中添加路由的时候,这两个数组元素以 ***name*** 和 ***:id*** 为代号被参数化在路由配置对象`HeroDetail`中。
+makeExample('toh-5/ts/app/app.component.ts','hero-detail-route', 'app/app.component.ts (英雄详情路由)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
The `DashboardComponent` doesn't have the router yet. We obtain it in the usual way:
`import` the `router` reference and inject it in the constructor (along with the `HeroService`):
2016-04-18 10:59:24 -04:00
`DashboardComponent`还没有路由器。我们使用常规的方式为它加上路由:
`import` `router`对象的引用,并且把它注入到构造函数中(就像`HeroService`那样):
+makeExample('toh-5/ts/app/dashboard.component.ts','import-router', 'app/dashboard.component.ts (节选)')(format=".")
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/dashboard.component.ts','ctor')(format=".")
:marked
Refresh the browser and select a hero from the dashboard; the app should navigate directly to that heros details.
2016-04-18 10:59:24 -04:00
刷新浏览器,并且从仪表盘中选择一位英雄;应用就会直接导航到英雄的详情。
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Select a Hero in the *HeroesComponent*
2016-04-18 10:59:24 -04:00
## 在 *HeroesComponent* 中选择一位英雄
2015-12-23 12:42:57 -05:00
We'll do something similar in the `HeroesComponent`.
2016-04-18 10:59:24 -04:00
我们要做的事和`HeroesComponent`中很像。
2015-12-23 12:42:57 -05:00
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.
2016-04-18 10:59:24 -04:00
那个组件现在的模板展示了一个主从风格的界面:上方是英雄列表,底下是所选英雄的详情。
+makeExample('toh-4/ts/app/app.component.ts','template', 'app/heroes.component.ts (当前的模板)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Delete the last line of the template with the `<my-hero-detail>` tags.
2016-04-18 10:59:24 -04:00
删除模板最后带有`<my-hero-detail>`标签的那一行。
2015-12-23 12:42:57 -05:00
We'll no longer show the full `HeroDetailComponent` here.
We're going to display the hero detail on its own page and route to it as we did in the dashboard.
2016-04-18 10:59:24 -04:00
这里我们不再展示完整的`HeroDetailComponent`了。
我们要在它自己的页面中显示英雄详情,并且像我们在仪表盘中所做的那样路由到它。
2015-12-23 12:42:57 -05:00
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.
We'll show a *mini-detail* on *this* page instead and make the user click a button to navigate to the *full detail *page.
2016-04-18 10:59:24 -04:00
但是,我们要做一点小小的改动。
当用户从这个列表中选择一个英雄时,我们 *不会* 再跳转到详情页。
而是在本页中显示一个 *Mini版英雄详情* ,并且让用户点击一个按钮,来导航到 *完整版英雄详情* 页。
2015-12-23 12:42:57 -05:00
### Add the *mini-detail*
2016-04-18 10:59:24 -04:00
### 添加 *Mini版英雄详情*
2015-12-23 12:42:57 -05:00
Add the following HTML fragment at the bottom of the template where the `<my-hero-detail>` used to be:
2016-04-18 10:59:24 -04:00
在模板底部原来放`<my-hero-detail>`的地方添加下列HTML片段
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/heroes.component.html','mini-detail')(format=".")
:marked
After clicking a hero, the user should see something like this below the hero list:
2016-04-18 10:59:24 -04:00
点击一个英雄,用户将会在英雄列表的下方看到这些:
2015-12-23 12:42:57 -05:00
figure.image-display
2016-04-18 10:59:24 -04:00
img(src='/resources/images/devguide/toh/mini-hero-detail.png' alt="Mini版英雄详情" height="70")
2015-12-23 12:42:57 -05:00
:marked
### Format with the *UpperCasePipe*
2016-04-18 10:59:24 -04:00
### 使用 *UpperCasePipe* 格式化
2015-12-23 12:42:57 -05:00
2016-03-08 07:53:01 -05:00
Notice that the hero's name is displayed in CAPITAL LETTERS. That's the effect of the `UpperCasePipe`
2015-12-23 12:42:57 -05:00
that we slipped into the interpolation binding. Look for it right after the pipe operator, ( | ).
2016-04-18 10:59:24 -04:00
注意,英雄的名字全被显示成大写字母。那是 `UpperCasePipe`的效果,借助它,我们能插手“插值表达式绑定”的过程。去管道操作符 ( | ) 后面找它。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/heroes.component.html','pipe')(format=".")
:marked
Pipes are a good way to format strings, currency amounts, dates and other display data.
Angular ships with several pipes and we can write our own.
2016-04-18 10:59:24 -04:00
管道擅长做下列工作:格式化字符串、金额、日期和其它显示数据。
Angular自带了好几个管道并且我们还可以写自己的管道。
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
Learn about pipes in the [Pipes](../guide/pipes.html) chapter.
2016-04-18 10:59:24 -04:00
要学习关于管道的更多知识,参见[管道](../guide/pipes.html)一章。
2015-12-23 12:42:57 -05:00
:marked
### Move content out of the component file
2016-04-18 10:59:24 -04:00
### 把内容移出组件文件
2015-12-23 12:42:57 -05:00
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.
2016-04-18 10:59:24 -04:00
还没完呢。在用户点击 *查看详情* 按钮时,要让他能导航到 `HeroDetailComponent`,我们仍然不得不修改组件类。
2015-12-23 12:42:57 -05:00
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.
2016-04-18 10:59:24 -04:00
这个组件文件太大了。它大部分都是模板或css样式。
要想在HTML和CSS的噪音中找出组件的工作逻辑太难了。
2015-12-23 12:42:57 -05:00
Let's migrate the template and the styles to their own files before we make any more changes:
2016-04-18 10:59:24 -04:00
在做更多修改之前,我们先把模板和样式移到它们自己的文件中去:
1. *Cut-and-paste* the template contents into a new `heroes.component.html` file.
2016-04-18 10:59:24 -04:00
1. *剪切并粘贴* 模板内容到新的`heroes.component.html`文件。
1. *Cut-and-paste* the styles contents into a new `heroes.component.css` file.
2016-04-18 10:59:24 -04:00
1. *剪切并粘贴* 样式内容到新的`heroes.component.css`文件。
1. *Set* the component metadata's `templateUrl` and `styleUrls` properties to refer to both files.
2016-04-18 10:59:24 -04:00
1. *设置* 组件元数据的`templateUrl`和`styleUrls`属性,来分别引用这两个文件。
2015-12-23 12:42:57 -05:00
The revised component data looks like this:
2016-04-18 10:59:24 -04:00
修改过的组件数据是这样的:
+makeExample('toh-5/ts/app/heroes.component.ts', 'metadata', 'app/heroes.component.ts (修改过的元数据)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Now we can see what's going on as we update the component class along the same lines as the dashboard:
2016-04-18 10:59:24 -04:00
现在,我们一下就明白该怎么像仪表盘中那样更新组件类了:
1. Import the `router`
2016-04-18 10:59:24 -04:00
1. 导入`router`
1. Inject the `router` in the constructor (along with the `HeroService`)
2016-04-18 10:59:24 -04:00
1. 把`router`注入到构造函数中(就像`HeroService`那样)
1. Implement the `gotoDetail` method by calling the `router.navigate` method
with a two-part 'HeroDetail' *link parameters array*.
2016-04-18 10:59:24 -04:00
1. 实现`gotoDetail`方法:以`HeroDetail`和 *链接参数数组* 为参数调用`router.navigate`方法。
2015-12-23 12:42:57 -05:00
Here's the revised component class:
2016-04-18 10:59:24 -04:00
下面是修改过的组件类:
+makeExample('toh-5/ts/app/heroes.component.ts', 'class', 'app/heroes.component.ts (类)')
2015-12-23 12:42:57 -05:00
:marked
Refresh the browser and start clicking.
We can navigate around the app, from the dashboard to hero details and back,
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.
2016-04-18 10:59:24 -04:00
刷新浏览器,并且开始点击。
我们能在应用中导航从仪表盘到英雄详情再回来从英雄列表到Mini版英雄详情到英雄详情然后再回到英雄列表。
我们可以在仪表盘和英雄列表之间跳来跳去。
2015-12-23 12:42:57 -05:00
We've met all of the navigational requirements that propelled this chapter.
2016-04-18 10:59:24 -04:00
我们已经达成了本章最初设定的所有导航需求。
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Styling the App
2016-04-18 10:59:24 -04:00
## 美化本应用
2015-12-23 12:42:57 -05:00
The app is functional but pretty ugly.
Our creative designer team provided some CSS files to make it look better.
2016-04-18 10:59:24 -04:00
应用的功能已经正常了,但还太丑。
我们富有创意的设计师团队提供了一些CSS文件能让它更好看一些。
2015-12-23 12:42:57 -05:00
### A Dashboard with Style
2016-04-18 10:59:24 -04:00
### 具有样式的仪表盘
2015-12-23 12:42:57 -05:00
The designers think we should display the dashboard heroes in a row of rectangles.
They've given us ~60 lines of CSS for this purpose including some simple media queries for responsive design.
2016-04-18 10:59:24 -04:00
设计师认为我们应该把仪表盘的英雄们显示在一排方块中。
他们给了我们大约60行CSS来实现它包括一些简单的媒体查询语句以实现响应式设计。
2015-12-23 12:42:57 -05:00
If we paste these ~60 lines into the component `styles` metadata,
they'll completely obscure the component logic.
Let's not do that. It's easier to edit CSS in a separate `*.css` file anyway.
2016-04-18 10:59:24 -04:00
如果我们把这60来行CSS粘贴到组件元数据的`styles`中,它们会完全淹没组件的工作逻辑。
不能那么做。在一个独立的`*.css`文件中编辑CSS当然会更简单。
2015-12-23 12:42:57 -05:00
Add a `dashboard.component.css` file to the `app` folder and reference
that file in the component metadata's `styleUrls` array property like this:
2016-04-18 10:59:24 -04:00
把`dashboard.component.css`文件添加到`app`目录下,并且在组件元数据的`styleUrls`数组属性中引用它。就像这样:
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/dashboard.component.ts', 'css', 'app/dashboard.component.ts (styleUrls)')(format=".")
:marked
.l-sub-section
:marked
The `styleUrls` property is an array of style file names (with paths).
We could list multiple style files from different locations if we needed them.
As with `templateUrl`, we must specify the path _all the way back to the application root_.
2016-04-18 10:59:24 -04:00
`styleUrls`属性是一个由样式文件的文件名(包括路径)组成的数组。
如果需要,我们还可以列出来自多个不同位置的样式文件。
和`templateUrl`一样,我们必须指定 _相对于此应用根目录_ 的路径。
2015-12-23 12:42:57 -05:00
:marked
### Stylish Hero Details
2016-04-18 10:59:24 -04:00
### 美化英雄详情
2015-12-23 12:42:57 -05:00
The designers also gave us CSS styles specifically for the `HeroDetailComponent`.
2016-04-18 10:59:24 -04:00
设计师还给了我们`HeroDetailComponent`特有的CSS风格。
2015-12-23 12:42:57 -05:00
Add a `hero-detail.component.css` to the `app` folder and refer to that file inside
the `styleUrls` array as we did for `DashboardComponent`.
2016-04-18 10:59:24 -04:00
在`app`目录下添加`hero-detail.component.css`文件,并且在`styleUrls`数组中引用它 —— 就像当初在`DashboardComponent`中做过的那样。
2015-12-23 12:42:57 -05:00
Here's the content for the aforementioned component CSS files.
2016-04-18 10:59:24 -04:00
上述组件的CSS文件内容如下
2015-12-23 12:42:57 -05:00
+makeTabs(
`toh-5/ts/app/hero-detail.component.css,
toh-5/ts/app/dashboard.component.css`,
null,
`app/hero-detail.component.css,
app/dashboard.component.css`)
:marked
### Style the Navigation Links
2016-04-18 10:59:24 -04:00
### 美化导航链接
2015-12-23 12:42:57 -05:00
The designers gave us CSS to make the navigation links in our `AppComponent` look more like selectable buttons.
We cooperated by surrounding those links in `<nav>` tags.
2016-04-18 10:59:24 -04:00
设计师还给了我们一些CSS用于让`AppComponent`中的导航链接看起来更像可被选择的按钮。
要想让它们协同工作,我们只需要把那些链接包含在`<nav>`标签中。
2015-12-23 12:42:57 -05:00
Add a `app.component.css` file to the `app` folder with the following content.
2016-04-18 10:59:24 -04:00
在`app`目录下添加一个`app.component.css`文件,内容如下:
+makeExample('toh-5/ts/app/app.component.css', 'css', 'app/app.component.css (导航样式)')
2015-12-23 12:42:57 -05:00
.l-sub-section
:marked
**The *router-link-active* class**
2016-04-18 10:59:24 -04:00
** *router-link-active* 类**
2015-12-23 12:42:57 -05:00
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!
2016-04-18 10:59:24 -04:00
Angular路由器会为与当前激活路由匹配的路由所在的导航元素加上`router-link-active`类。
我们唯一要做的就是为它(`.router-link-active`类)定义样式。真好!
2015-12-23 12:42:57 -05:00
:marked
Set the `AppComponent`s `styleUrls` property to this CSS file.
2016-04-18 10:59:24 -04:00
设置`AppComponent`的`styleUrls`属性指向这个CSS文件。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/app/app.component.ts','style-urls', 'app/app.component.ts (styleUrls)')(format=".")
:marked
### Global application styles
2016-04-18 10:59:24 -04:00
### 应用的全局样式
2015-12-23 12:42:57 -05:00
When we add styles to a component, we're keeping everything a component needs
&mdash; HTML, the CSS, the code &mdash; together in one convenient place.
It's pretty easy to package it all up and re-use the component somewhere else.
2016-04-18 10:59:24 -04:00
当我们把样式添加到组件中时,我们假定组件所需的一切 —— HTML、CSS、程序代码 —— 都在紧邻的地方。
这样,把它们打包在一起并在别的组件中复用它都会很容易。
2015-12-23 12:42:57 -05:00
We can also create styles at the *application level* outside of any component.
2016-04-18 10:59:24 -04:00
我们也可以在应用中所有组件之外的地方创建样式。
2015-12-23 12:42:57 -05:00
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.
2016-04-18 10:59:24 -04:00
我们的设计师提供了一组基础样式,这些样式适用的元素横跨整个应用。
把下面这个名叫`styles.css`的新文件添加到根目录中:
+makeExample('toh-5/ts/styles.1.css', '', 'styles.css (应用级样式)')(format=".")
2015-12-23 12:42:57 -05:00
:marked
Reference this stylesheet within the `index.html` in the traditional manner.
2016-04-18 10:59:24 -04:00
用传统的方式,在`index.html`中引用这个样式表。
2015-12-23 12:42:57 -05:00
+makeExample('toh-5/ts/index.html','css', 'index.html (link ref)')(format=".")
:marked
Look at the app now. Our dashboard, heroes, and navigation links are styling!
2016-04-18 10:59:24 -04:00
看看现在的应用!我们的仪表盘、英雄列表和导航链接都漂亮多了!
2015-12-23 12:42:57 -05:00
figure.image-display
2016-04-18 10:59:24 -04:00
img(src='/resources/images/devguide/toh/dashboard-top-heroes.png' alt="查看导航栏")
2015-12-23 12:42:57 -05:00
.l-main-section
:marked
## Application structure and code
2016-04-18 10:59:24 -04:00
## 应用结构和代码
2015-12-23 12:42:57 -05:00
Review the sample source code [in the live example for this chapter](/resources/live-examples/toh-5/ts/plnkr.html).
Verify that we have the following structure:
2016-04-18 10:59:24 -04:00
在[本章的鲜活范例](/resources/live-examples/toh-5/ts/plnkr.html)中回顾范例代码。请验证一下我们已经有了下列结构:
2015-12-23 12:42:57 -05:00
.filetree
.file angular2-tour-of-heroes
.children
.file app
.children
.file app.component.ts
.file app.component.css
.file dashboard.component.css
.file dashboard.component.html
.file dashboard.component.ts
.file hero.ts
.file hero-detail.component.css
.file hero-detail.component.html
.file hero-detail.component.ts
.file hero.service.ts
.file heroes.component.css
.file heroes.component.html
.file heroes.component.ts
.file main.ts
.file mock-heroes.ts
.file node_modules ...
.file typings ...
2015-12-23 12:42:57 -05:00
.file index.html
.file package.json
.file styles.css
2015-12-23 12:42:57 -05:00
.file tsconfig.json
.file typings.json
2015-12-23 12:42:57 -05:00
:marked
.l-main-section
:marked
## Recap
2016-04-18 10:59:24 -04:00
## 总结
2015-12-23 12:42:57 -05:00
### The Road Behind
2016-04-18 10:59:24 -04:00
### 走过的路
We travelled a great distance in this chapter.
2016-04-18 10:59:24 -04:00
在本章中,我们往前走了很远:
- We added the Angular *Component Router* to navigate among different components.
2016-04-18 10:59:24 -04:00
- 我们添加了Angular *组件路由器* 在各个不同的组件之间导航。
- We learned how to create router links to represent navigation menu items
2016-04-18 10:59:24 -04:00
- 我们学会了如何创建路由链接来表示导航栏的菜单项。
- We used router parameters to navigate to the details of user selected hero
2016-04-18 10:59:24 -04:00
- 我们使用路由参数来导航到用户所选的英雄详情。
- We shared the `HeroService` among multiple components
2016-04-18 10:59:24 -04:00
- 我们在多个组件之间共享了`HeroService`服务。
- We moved HTML and CSS out of the component file and into their own files.
2016-04-18 10:59:24 -04:00
- 我们把HTML和CSS从组件中移出来放到它们自己的文件中去。
- We added the `uppercase` pipe to format data
2016-04-18 10:59:24 -04:00
- 我们添加了一个`uppercase`管道,来格式化数据。
2015-12-23 12:42:57 -05:00
### The Road Ahead
2016-04-18 10:59:24 -04:00
### 前方的路
2015-12-23 12:42:57 -05:00
We have much of the foundation we need to build an application.
We're still missing a key piece: remote data access.
2016-04-18 10:59:24 -04:00
我们有了很多用于构建应用的基石。
但我们仍然缺少很关键的一部分:远程数据存取。
2015-12-23 12:42:57 -05:00
In a forthcoming tutorial chapter,
well replace our mock data with data retrieved from a server using http.
2016-04-18 10:59:24 -04:00
在即将到来的章节中我们将从硬编码模拟数据改为使用http服务从服务器获取数据。