路由
In-app navigation
在第一部分结束时,这个在线商店应用会有一个基本的商品名录。该应用还没有任何可变的状态或导航。它只有一个 URL,该 URL 总是会显示“我的商店”页面,其中是商品列表及其描述。
At the end of part 1, the online store application has a basic product catalog. The app doesn't have any variable states or navigation. There is one URL, and that URL always displays the "My Store" page with a list of products and their descriptions.
本指南会告诉你如何使用 Angular 路由来让用户进行应用内导航。在单页面应用中,你不会重新加载新页面,而是根据用户所在的位置,向他显示不同的组件和数据。
This guide shows you how to use Angular routing to give the user in-app navigation. In a single-page app, instead of loading new pages, you show different components and data to the user based on where the user is in the application.
Angular 路由器能让你在不同的视图中显示产品的详情,每个产品都有自己的 URL。当用户执行应用任务时,路由器可以从一个视图导航到另一个视图(但在同一个页面)。比如:
The router lets you display full product details in separate views, each with its own URL. Routing enables navigation from one view to the next (within the same page) as users perform tasks such as the following:
在地址栏中输入一个 URL,导航到相应的页面。
Entering a URL in the address bar to navigate to a corresponding view.
点击页面上的链接,导航到新页面。
Clicking links on the page to navigate to a new view.
点击浏览器的后退和前进按钮,在浏览器的历史中前后导航。
Clicking the browser's back and forward buttons to navigate backward and forward through the browser history.
注册路由
Registering a route
该应用已经设置为使用 Angular Router
,并通过路由导航到之前修改过的商品列表组件。本节会向你展示如何定义一个可以显示单个商品详情的路由。
The app is already set up to use the Angular Router
and to use routing to navigate to the product list component you modified earlier. This section shows you how to define a route to show individual product details.
为商品详情生成一个新组件。把组件命名为
product-details
。Generate a new component for product details. Give the component the name
product-details
.提示:在文件列表框中,右键单击
app
文件夹,选择Angular Generator
和Component
。Reminder: In the file list, right-click the
app
folder, chooseAngular Generator
andComponent
.在
app.module.ts
中,添加一个商品详情路由,该路由的path
是products/:productId
,component
是ProductDetailsComponent
。In
app.module.ts
, add a route for product details, with apath
ofproducts/:productId
andProductDetailsComponent
for thecomponent
.src/app/app.module.ts @NgModule({ imports: [ BrowserModule, ReactiveFormsModule, RouterModule.forRoot([ { path: '', component: ProductListComponent }, { path: 'products/:productId', component: ProductDetailsComponent }, ]) ],
路由会将一个或多个 URL 路径与一个组件关联起来。
A route associates one or more URL paths with a component.
该指令配置组件的模板,以定义用户如何导航到路由或 URL。当用户点击商品名称时,应用就会显示那个商品的详情。
The directive configures the component template to define how the user navigates to the route or URL. When the user clicks a product name, the app displays the details for that product.
打开
product-list.component.html
。Open
product-list.component.html
.修改
*ngFor
指令,在遍历列表的过程中把products
数组中的每个索引赋值给productId
变量。Update the
*ngFor
directive to assign each index in theproducts
array to theproductId
variable when iterating over the list.修改商品名称的链接,使其包含
routerLink
。Modify the product name anchor to include a
routerLink
.
src/app/product-list/product-list.component.html <div *ngFor="let product of products; index as productId"> <h3> <a [title]="product.name + ' details'" [routerLink]="['/products', productId]"> {{ product.name }} </a> </h3> <!-- . . . --> </div>
RouterLink 指令让路由器控制了一个链接元素。在这种情况下,路由或 URL 包含一个固定的区段(
/products
),但其最后一个区段是变量,要插入当前商品的 id 属性。例如,id
为 1 的商品的 URL 类似于https://getting-started-myfork.stackblitz.io/products/1
。The RouterLink directive gives the router control over the anchor element. In this case, the route, or URL, contains one fixed segment,
/products
, while the final segment is variable, inserting the id property of the current product. For example, the URL for a product with anid
of 1 will be similar tohttps://getting-started-myfork.stackblitz.io/products/1
.通过单击商品名称来测试路由器。该应用会显示商品详情组件,该组件目前始终显示 “product-details works!”
Test the router by clicking a product name. The app displays the product details component, which currently always says "product-details works!"
注意预览窗口中的 URL 变化了。它的最后一段是
products/#
,这里的#
代表你点击的那个路由的编号。Notice that the URL in the preview window changes. The final segment is
products/#
where#
is the number of the route you clicked.
使用路由信息
Using route information
商品详情组件负责处理每个商品的显示。Angular 的路由器会根据浏览器的 URL 和你定义的这些路由来决定如何显示组件。本节会告诉你如何通过 Angular 的路由器来组合使用 products
数据和路由信息,以显示每个商品的详情。
The product details component handles the display of each product. The Angular Router displays components based on the browser's URL and your defined routes. This section shows you how to use the Angular Router to combine the products
data and route information to display the specific details for each product.
打开
product-details.component.ts
文件Open
product-details.component.ts
改用外部文件中的商品数据。
Arrange to use product data from an external file.
从
@angular/router
包导入ActivatedRoute
,从../products
文件导入products
数组。Import
ActivatedRoute
from the@angular/router
package, and theproducts
array from../products
.src/app/product-details/product-details.component.ts import { Component, OnInit } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; import { products } from '../products';
定义
product
属性,并将ActivatedRoute
作为参数添加到构造函数的括号中,以便把它注入到构造函数中。Define the
product
property and inject theActivatedRoute
into the constructor by adding it as an argument within the constructor's parentheses.src/app/product-details/product-details.component.ts export class ProductDetailsComponent implements OnInit { product; constructor( private route: ActivatedRoute, ) { } }
ActivatedRoute
专门用于由 Angular 路由器加载的每个路由组件。它包含关于该路由,路由参数以及与该路由关联的其它数据的信息。The
ActivatedRoute
is specific to each routed component that the Angular Router loads. It contains information about the route, its parameters, and additional data associated with the route.通过注入
ActivatedRoute
,你把该组件配置成了使用服务的。《快速上手》教程中的这部分只是简略使用了该语法,在管理数据部分深入讲解了服务的更多细节。By injecting the
ActivatedRoute
, you are configuring the component to use a service. The Managing Data page covers services in more detail.
在
ngOnInit()
方法中订阅了路由参数,并且根据productId
获取了该产品。In the
ngOnInit()
method, subscribe to route parameters and fetch the product based on theproductId
.src/app/product-details/product-details.component.ts ngOnInit() { this.route.paramMap.subscribe(params => { this.product = products[+params.get('productId')]; }); }
这个路由参数对应于你在路由中定义的路径变量。与该路由匹配的 URL 提供了
productId
。Angular 使用这个productId
来显示每个单独商品的详细信息。The route parameters correspond to the path variables you define in the route. The URL that matches the route provides the
productId
. Angular uses theproductId
to display the details for each unique product.修改模板,在
*ngIf
中显示商品详情。Update the template to display product details information inside an
*ngIf
.src/app/product-details/product-details.component.html <h2>Product Details</h2> <div *ngIf="product"> <h3>{{ product.name }}</h3> <h4>{{ product.price | currency }}</h4> <p>{{ product.description }}</p> </div>
现在,当用户点击商品列表中的某个名字时,路由器就会导航到商品的不同网址,用商品详情组件代替商品列表组件,并显示商品详情。
Now, when users click on a name in the product list, the router navigates them to the distinct URL for the product, swaps out the product list component for the product details component, and displays the product details.
要了解关于 Angular 路由器的更多信息,请参阅路由和导航。
For more information about the Angular Router, see Routing & Navigation.
下一步
Next steps
恭喜!你已经把路由集成到你的在线商店了。
Congratulations! You have integrated routing into your online store.
从商品列表页面链接到了单个商品。
Products are linked from the product list view to individual products.
用户可以点击列表中的某个商品名称来在新视图中查看其详细信息,并带有显著的 URL/路由。
Users can click on a product name from the list to see details in a new view, with a distinct URL/route.
要继续探索 Angular,请选择以下选项之一:
To continue exploring Angular, choose either of the following options:
继续浏览“管理数据”部分,以添加购物车功能,使用服务来管理购物车数据,并通过 HTTP 检索配送价格的外部数据。
Continue to the "Managing Data" section to add a shopping cart feature, use a service to manage the cart data and use HTTP to retrieve external data for shipping prices.
跳到部署部分,把你的应用部署到 Firebase 或转成本地开发。
Skip ahead to the Deployment section to deploy your app to Firebase or move to local development.