• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# 组件导航 (Navigation)(推荐)
2
3组件导航(Navigation)主要用于实现页面间以及组件内部的页面跳转,支持在不同组件间传递跳转参数,提供灵活的跳转栈操作,从而更便捷地实现对不同页面的访问和复用。本文将从组件导航(Navigation)的显示模式、路由操作、子页面管理、跨包跳转以及跳转动效等几个方面进行详细介绍。
4
5[Navigation](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md)是路由导航的根视图容器,一般作为页面(@Entry)的根容器,包括单栏(Stack)、分栏(Split)和自适应(Auto)三种显示模式。Navigation组件适用于模块内和跨模块的路由切换,通过组件级路由能力实现更加自然流畅的转场体验,并提供多种标题栏样式来呈现更好的标题和内容联动效果。一次开发,多端部署场景下,Navigation组件能够自动适配窗口显示大小,在窗口较大的场景下自动切换分栏展示效果。
6
7Navigation组件主要包含​导航页和子页。导航页由标题栏(包含菜单栏)、内容区和工具栏组成,可以通过[hideNavBar](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#hidenavbar9)属性进行隐藏,导航页不存在页面栈中,与子页,以及子页之间可以通过路由操作进行切换。
8
9在API Version 9上,Navigation需要配合[NavRouter](../reference/apis-arkui/arkui-ts/ts-basic-components-navrouter.md)组件实现页面路由。从API Version 10开始,更推荐使用[NavPathStack](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navpathstack10)实现页面路由。
10
11
12## 设置页面显示模式
13
14Navigation组件通过mode属性设置页面的显示模式。
15
16- 自适应模式
17
18  Navigation组件默认为自适应模式,此时mode属性为NavigationMode.Auto。自适应模式下,当页面宽度大于等于一定阈值( API version 9及以前:520vp,API version 10及以后:600vp )时,Navigation组件采用分栏模式,反之采用单栏模式。
19
20
21  ```
22  Navigation() {
23    // ...
24  }
25  .mode(NavigationMode.Auto)
26  ```
27
28- 单页面模式
29
30    **图1** 单页面布局示意图  
31
32  ![zh-cn_image_0000001511740532](figures/zh-cn_image_0000001511740532.png)
33
34  将mode属性设置为NavigationMode.Stack,Navigation组件即可设置为单页面显示模式。
35
36
37  ```ts
38  Navigation() {
39    // ...
40  }
41  .mode(NavigationMode.Stack)
42  ```
43
44  ![导航单栏模式](figures/导航单栏模式.jpg)
45
46- 分栏模式
47
48  **图2** 分栏布局示意图
49
50  ![zh-cn_image_0000001562820845](figures/zh-cn_image_0000001562820845.png)
51
52  将mode属性设置为NavigationMode.Split,Navigation组件即可设置为分栏显示模式。
53
54
55  ```ts
56  @Entry
57  @Component
58  struct NavigationExample {
59    @State TooTmp: ToolbarItem = {
60      'value': "func", 'icon': "./image/ic_public_highlights.svg", 'action': () => {
61      }
62    }
63    @Provide('pageInfos') pageInfos: NavPathStack = new NavPathStack()
64    private arr: number[] = [1, 2, 3];
65
66    @Builder
67    PageMap(name: string) {
68      if (name === "NavDestinationTitle1") {
69        pageOneTmp()
70      } else if (name === "NavDestinationTitle2") {
71        pageTwoTmp()
72      } else if (name === "NavDestinationTitle3") {
73        pageThreeTmp()
74      }
75    }
76
77    build() {
78      Column() {
79        Navigation(this.pageInfos) {
80          TextInput({ placeholder: 'search...' })
81            .width("90%")
82            .height(40)
83            .backgroundColor('#FFFFFF')
84
85          List({ space: 12 }) {
86            ForEach(this.arr, (item: number) => {
87              ListItem() {
88                Text("Page" + item)
89                  .width("100%")
90                  .height(72)
91                  .backgroundColor('#FFFFFF')
92                  .borderRadius(24)
93                  .fontSize(16)
94                  .fontWeight(500)
95                  .textAlign(TextAlign.Center)
96                  .onClick(() => {
97                    this.pageInfos.pushPath({ name: "NavDestinationTitle" + item })
98                  })
99              }
100            }, (item: number) => item.toString())
101          }
102          .width("90%")
103          .margin({ top: 12 })
104        }
105        .title("主标题")
106        .mode(NavigationMode.Split)
107        .navDestination(this.PageMap)
108        .menus([
109          {
110            value: "", icon: "./image/ic_public_search.svg", action: () => {
111            }
112          },
113          {
114            value: "", icon: "./image/ic_public_add.svg", action: () => {
115            }
116          },
117          {
118            value: "", icon: "./image/ic_public_add.svg", action: () => {
119            }
120          },
121          {
122            value: "", icon: "./image/ic_public_add.svg", action: () => {
123            }
124          },
125          {
126            value: "", icon: "./image/ic_public_add.svg", action: () => {
127            }
128          }
129        ])
130        .toolbarConfiguration([this.TooTmp, this.TooTmp, this.TooTmp])
131      }
132      .height('100%')
133      .width('100%')
134      .backgroundColor('#F1F3F5')
135    }
136  }
137
138  // PageOne.ets
139  @Component
140  export struct pageOneTmp {
141    @Consume('pageInfos') pageInfos: NavPathStack;
142
143    build() {
144      NavDestination() {
145        Column() {
146          Text("NavDestinationContent1")
147        }.width('100%').height('100%')
148      }.title("NavDestinationTitle1")
149      .onBackPressed(() => {
150        const popDestinationInfo = this.pageInfos.pop() // 弹出路由栈栈顶元素
151        console.log('pop' + '返回值' + JSON.stringify(popDestinationInfo))
152        return true
153      })
154    }
155  }
156
157  // PageTwo.ets
158  @Component
159  export struct pageTwoTmp {
160    @Consume('pageInfos') pageInfos: NavPathStack;
161
162    build() {
163      NavDestination() {
164        Column() {
165          Text("NavDestinationContent2")
166        }.width('100%').height('100%')
167      }.title("NavDestinationTitle2")
168      .onBackPressed(() => {
169        const popDestinationInfo = this.pageInfos.pop() // 弹出路由栈栈顶元素
170        console.log('pop' + '返回值' + JSON.stringify(popDestinationInfo))
171        return true
172      })
173    }
174  }
175
176  // PageThree.ets
177  @Component
178  export struct pageThreeTmp {
179    @Consume('pageInfos') pageInfos: NavPathStack;
180
181    build() {
182      NavDestination() {
183        Column() {
184          Text("NavDestinationContent3")
185        }.width('100%').height('100%')
186      }.title("NavDestinationTitle3")
187      .onBackPressed(() => {
188        const popDestinationInfo = this.pageInfos.pop() // 弹出路由栈栈顶元素
189        console.log('pop' + '返回值' + JSON.stringify(popDestinationInfo))
190        return true
191      })
192    }
193  }
194  ```
195
196  ![导航分栏模式](figures/导航分栏模式.jpg)
197
198
199## 设置标题栏模式
200
201标题栏在界面顶部,用于呈现界面名称和操作入口,Navigation组件通过titleMode属性设置标题栏模式。
202
203> **说明:**
204> Navigation或NavDestination未设置主副标题并且没有返回键时,不显示标题栏。
205
206- Mini模式
207
208  普通型标题栏,用于一级页面不需要突出标题的场景。
209
210  **图3** Mini模式标题栏  
211
212  ![mini](figures/mini.jpg)
213
214
215  ```ts
216  Navigation() {
217    // ...
218  }
219  .titleMode(NavigationTitleMode.Mini)
220  ```
221
222
223- Full模式
224
225  强调型标题栏,用于一级页面需要突出标题的场景。
226
227    **图4** Full模式标题栏  
228
229  ![free1](figures/free1.jpg)
230
231
232  ```ts
233  Navigation() {
234    // ...
235  }
236  .titleMode(NavigationTitleMode.Full)
237  ```
238
239
240## 设置菜单栏
241
242菜单栏位于Navigation组件的右上角,开发者可以通过menus属性进行设置。menus支持Array<[NavigationMenuItem](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navigationmenuitem)>和[CustomBuilder](../reference/apis-arkui/arkui-ts/ts-types.md#custombuilder8)两种参数类型。使用Array<NavigationMenuItem>类型时,竖屏最多支持显示3个图标,横屏最多支持显示5个图标,多余的图标会被放入自动生成的更多图标。
243
244**图5** 设置了3个图标的菜单栏  
245
246![菜单栏2](figures/菜单栏2.jpg)
247
248```ts
249let TooTmp: NavigationMenuItem = {'value': "", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
250Navigation() {
251  // ...
252}
253.menus([TooTmp,
254  TooTmp,
255  TooTmp])
256```
257
258图片也可以引用resources中的资源。
259
260```ts
261let TooTmp: NavigationMenuItem = {'value': "", 'icon': "resources/base/media/ic_public_highlights.svg", 'action': ()=> {}}
262Navigation() {
263  // ...
264}
265.menus([TooTmp,
266  TooTmp,
267  TooTmp])
268```
269
270**图6** 设置了4个图标的菜单栏  
271
272![菜单栏](figures/菜单栏.jpg)
273
274```ts
275let TooTmp: NavigationMenuItem = {'value': "", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
276Navigation() {
277  // ...
278}
279// 竖屏最多支持显示3个图标,多余的图标会被放入自动生成的更多图标。
280.menus([TooTmp,
281  TooTmp,
282  TooTmp,
283  TooTmp])
284```
285
286
287## 设置工具栏
288
289工具栏位于Navigation组件的底部,开发者可以通过[toolbarConfiguration](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#toolbarconfiguration10)属性进行设置。
290
291
292  **图7** 工具栏  
293
294![free3](figures/free3.jpg)
295
296```ts
297let TooTmp: ToolbarItem = {'value': "func", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
298let TooBar: ToolbarItem[] = [TooTmp,TooTmp,TooTmp]
299Navigation() {
300  // ...
301}
302.toolbarConfiguration(TooBar)
303```
304
305## 路由操作
306
307Navigation路由相关的操作都是基于页面栈[NavPathStack](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navpathstack10)提供的方法进行,每个Navigation都需要创建并传入一个NavPathStack对象,用于管理页面。主要涉及页面跳转、页面返回、页面替换、页面删除、参数获取、路由拦截等功能。
308
309从API version 12开始,页面栈允许被继承。开发者可以在派生类中自定义属性和方法,也可以重写父类的方法。派生类对象可以替代基类NavPathStack对象使用。具体示例代码参见:[页面栈继承示例代码](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#示例10定义路由栈派生类)。
310
311> **说明:**
312>
313> 不建议开发者通过监听生命周期的方式管理自己的页面栈。
314
315```ts
316@Entry
317@Component
318struct Index {
319  // 创建一个页面栈对象并传入Navigation
320  pageStack: NavPathStack = new NavPathStack()
321
322  build() {
323    Navigation(this.pageStack) {
324    }
325    .title('Main')
326  }
327}
328```
329
330### 页面跳转
331
332NavPathStack通过Push相关的接口去实现页面跳转的功能,主要分为以下三类:
333
3341. 普通跳转,通过页面的name去跳转,并可以携带param。
335
336    ```ts
337    this.pageStack.pushPath({ name: "PageOne", param: "PageOne Param" })
338    this.pageStack.pushPathByName("PageOne", "PageOne Param")
339    ```
340
3412. 带返回回调的跳转,跳转时添加onPop回调,能在页面出栈时获取返回信息,并进行处理。
342
343    ```ts
344    this.pageStack.pushPathByName('PageOne', "PageOne Param", (popInfo) => {
345      console.log('Pop page name is: ' + popInfo.info.name + ', result: ' + JSON.stringify(popInfo.result))
346    });
347    ```
348
3493. 带错误码的跳转,跳转结束会触发异步回调,返回错误码信息。
350
351    ```ts
352    this.pageStack.pushDestination({name: "PageOne", param: "PageOne Param"})
353      .catch((error: BusinessError) => {
354        console.error(`Push destination failed, error code = ${error.code}, error.message = ${error.message}.`);
355      }).then(() => {
356        console.info('Push destination succeed.');
357      });
358    this.pageStack.pushDestinationByName("PageOne", "PageOne Param")
359      .catch((error: BusinessError) => {
360        console.error(`Push destination failed, error code = ${error.code}, error.message = ${error.message}.`);
361      }).then(() => {
362        console.info('Push destination succeed.');
363      });
364    ```
365
366### 页面返回
367
368NavPathStack通过Pop相关接口去实现页面返回功能。
369
370```ts
371// 返回到上一页
372this.pageStack.pop()
373// 返回到上一个PageOne页面
374this.pageStack.popToName("PageOne")
375// 返回到索引为1的页面
376this.pageStack.popToIndex(1)
377// 返回到根首页(清除栈中所有页面)
378this.pageStack.clear()
379```
380
381### 页面替换
382
383NavPathStack通过Replace相关接口去实现页面替换功能。
384
385```ts
386// 将栈顶页面替换为PageOne
387this.pageStack.replacePath({ name: "PageOne", param: "PageOne Param" })
388this.pageStack.replacePathByName("PageOne", "PageOne Param")
389// 带错误码的替换,跳转结束会触发异步回调,返回错误码信息
390this.pageStack.replaceDestination({name: "PageOne", param: "PageOne Param"})
391  .catch((error: BusinessError) => {
392    console.error(`Replace destination failed, error code = ${error.code}, error.message = ${error.message}.`);
393  }).then(() => {
394    console.info('Replace destination succeed.');
395  })
396```
397
398### 页面删除
399
400NavPathStack通过Remove相关接口去实现删除页面栈中特定页面的功能。
401
402```ts
403// 删除栈中name为PageOne的所有页面
404this.pageStack.removeByName("PageOne")
405// 删除指定索引的页面
406this.pageStack.removeByIndexes([1,3,5])
407// 删除指定id的页面
408this.pageStack.removeByNavDestinationId("1");
409```
410
411### 移动页面
412
413NavPathStack通过Move相关接口去实现移动页面栈中特定页面到栈顶的功能。
414
415```ts
416// 移动栈中name为PageOne的页面到栈顶
417this.pageStack.moveToTop("PageOne");
418// 移动栈中索引为1的页面到栈顶
419this.pageStack.moveIndexToTop(1);
420```
421
422### 参数获取
423
424NavPathStack通过Get相关接口去获取页面的一些参数。
425
426```ts
427// 获取栈中所有页面name集合
428this.pageStack.getAllPathName()
429// 获取索引为1的页面参数
430this.pageStack.getParamByIndex(1)
431// 获取PageOne页面的参数
432this.pageStack.getParamByName("PageOne")
433// 获取PageOne页面的索引集合
434this.pageStack.getIndexByName("PageOne")
435```
436
437### 路由拦截
438
439NavPathStack提供了[setInterception](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#setinterception12)方法,用于设置Navigation页面跳转拦截回调。该方法需要传入一个NavigationInterception对象,该对象包含三个回调函数:
440
441| 名称       | 描述                                                 |
442| ------------ | ------------------------------------------------------ |
443| willShow   | 页面跳转前回调,允许操作栈,在当前跳转生效。       |
444| didShow    | 页面跳转后回调,在该回调中操作栈会在下一次跳转生效。 |
445| modeChange | Navigation单双栏显示状态发生变更时触发该回调。  |
446
447> **说明:**
448>
449> 无论是哪个回调,在进入回调时页面栈都已经发生了变化。
450
451开发者可以在willShow回调中通过修改路由栈来实现路由拦截重定向的能力。
452
453```ts
454this.pageStack.setInterception({
455  willShow: (from: NavDestinationContext | "navBar", to: NavDestinationContext | "navBar",
456    operation: NavigationOperation, animated: boolean) => {
457    if (typeof to === "string") {
458      console.log("target page is navigation home page.");
459      return;
460    }
461    // 将跳转到PageTwo的路由重定向到PageOne
462    let target: NavDestinationContext = to as NavDestinationContext;
463    if (target.pathInfo.name === 'PageTwo') {
464      target.pathStack.pop();
465      target.pathStack.pushPathByName('PageOne', null);
466    }
467  }
468})
469```
470
471### 单例跳转
472
473通过设置[LaunchMode](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#launchmode12枚举说明)为LaunchMode.MOVE_TO_TOP_SINGLETONLaunchMode.POP_TO_SINGLETON,可以实现Navigation路由栈的单实例跳转。单实例跳转的规则如下:
474
4751. 当指定为LaunchMode.MOVE_TO_TOP_SINGLETON时,系统会从栈底到栈顶查找具有指定名称的NavDestination。找到后,该页面将被移动到栈顶(replace操作会用指定的NavDestination替换当前栈顶)。
4762. 若指定为LaunchMode.POP_TO_SINGLETON,系统同样会从栈底到栈顶查找具有指定名称的NavDestination。找到后,便会移除该NavDestination上方的所有页面(replace操作会用指定的NavDestination替换当前栈顶)。
477
478有关单实例路由栈操作的示例代码,可以参考[Navigation单例跳转示例](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#示例2使用路由栈方法)。
479
480## 子页面
481
482[NavDestination](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md)是Navigation子页面的根容器,用于承载子页面的一些特殊属性以及生命周期等。NavDestination可以设置独立的标题栏和菜单栏等属性,使用方法与Navigation相同。NavDestination也可以通过mode属性设置不同的显示类型,用于满足不同页面的诉求。
483
484### 页面显示类型
485
486- 标准类型
487
488  NavDestination组件默认为标准类型,此时mode属性为NavDestinationMode.STANDARD。标准类型的NavDestination的生命周期跟随其在NavPathStack页面栈中的位置变化而改变。
489
490- 弹窗类型
491
492  NavDestination设置mode为NavDestinationMode.DIALOG弹窗类型,此时整个NavDestination默认透明显示。弹窗类型的NavDestination显示和消失时不会影响下层标准类型的NavDestination的显示和生命周期,两者可以同时显示。
493
494  ```ts
495  // Dialog NavDestination
496  @Entry
497  @Component
498   struct Index {
499     @Provide('NavPathStack') pageStack: NavPathStack = new NavPathStack()
500
501     @Builder
502     PagesMap(name: string) {
503       if (name == 'DialogPage') {
504         DialogPage()
505       }
506     }
507
508     build() {
509       Navigation(this.pageStack) {
510         Button('Push DialogPage')
511           .margin(20)
512           .width('80%')
513           .onClick(() => {
514             this.pageStack.pushPathByName('DialogPage', '');
515           })
516       }
517       .mode(NavigationMode.Stack)
518       .title('Main')
519       .navDestination(this.PagesMap)
520     }
521   }
522
523   @Component
524   export struct DialogPage {
525     @Consume('NavPathStack') pageStack: NavPathStack;
526
527     build() {
528       NavDestination() {
529         Stack({ alignContent: Alignment.Center }) {
530           Column() {
531             Text("Dialog NavDestination")
532               .fontSize(20)
533               .margin({ bottom: 100 })
534             Button("Close").onClick(() => {
535               this.pageStack.pop()
536             }).width('30%')
537           }
538           .justifyContent(FlexAlign.Center)
539           .backgroundColor(Color.White)
540           .borderRadius(10)
541           .height('30%')
542           .width('80%')
543         }.height("100%").width('100%')
544       }
545       .backgroundColor('rgba(0,0,0,0.5)')
546       .hideTitleBar(true)
547       .mode(NavDestinationMode.DIALOG)
548     }
549   }
550  ```
551  ![dialog_navdestination](figures/dialog_navdestination.png)
552
553### 页面生命周期
554
555Navigation作为路由容器,其生命周期承载在NavDestination组件上,以组件事件的形式开放。
556
557其生命周期大致可分为三类,自定义组件生命周期、通用组件生命周期和自有生命周期。其中,[aboutToAppear](../reference/apis-arkui/arkui-ts/ts-custom-component-lifecycle.md#abouttoappear)和[aboutToDisappear](../reference/apis-arkui/arkui-ts/ts-custom-component-lifecycle.md#abouttodisappear)是自定义组件的生命周期(NavDestination外层包含的自定义组件),[OnAppear](../reference/apis-arkui/arkui-ts/ts-universal-events-show-hide.md#onappear)和[OnDisappear](../reference/apis-arkui/arkui-ts/ts-universal-events-show-hide.md#ondisappear)是组件的通用生命周期。剩下的生命周期为NavDestination独有。
558
559生命周期时序如下图所示:
560
561![navigation_lifecycle](figures/navigation_lifecycle.png)
562
563- **aboutToAppear**:在创建自定义组件后,执行其build()函数之前执行(NavDestination创建之前),允许在该方法中改变状态变量,更改将在后续执行build()函数中生效。
564- **onWillAppear**:NavDestination创建后,挂载到组件树之前执行,在该方法中更改状态变量会在当前帧显示生效。
565- **onAppear**:通用生命周期事件,NavDestination组件挂载到组件树时执行。
566- **onWillShow**:NavDestination组件布局显示之前执行,此时页面不可见(应用切换到前台不会触发)。
567- **onShown**:NavDestination组件布局显示之后执行,此时页面已完成布局。
568- **onActive**: NavDestination处于激活态(处于栈顶可操作,且上层无特殊组件遮挡)触发。
569- **onWillHide**:NavDestination组件触发隐藏之前执行(应用切换到后台不会触发)。
570- **onInactive**: NavDestination组件处于非激活态(处于非栈顶不可操作,或处于栈顶时上层有特殊组件遮挡)触发。
571- **onHidden**:NavDestination组件触发隐藏后执行(非栈顶页面push进栈,栈顶页面pop出栈或应用切换到后台)。
572- **onWillDisappear**:NavDestination组件即将销毁之前执行,如果有转场动画,会在动画前触发(栈顶页面pop出栈)。
573- **onDisappear**:通用生命周期事件,NavDestination组件从组件树上卸载销毁时执行。
574- **aboutToDisappear**:自定义组件析构销毁之前执行,不允许在该方法中改变状态变量。
575
576### 页面监听和查询
577
578为了方便组件跟页面解耦,在NavDestination子页面内部的自定义组件可以通过全局方法监听或查询到页面的一些状态信息。
579
580- 页面信息查询
581
582  自定义组件提供[queryNavDestinationInfo](../reference/apis-arkui/arkui-ts/ts-custom-component-api.md#querynavdestinationinfo)方法,可以在NavDestination内部查询到当前所属页面的信息,返回值为[NavDestinationInfo](../reference/apis-arkui/js-apis-arkui-observer.md#navdestinationinfo),若查询不到则返回undefined。
583
584  ```ts
585   import { uiObserver } from '@kit.ArkUI';
586
587   // NavDestination内的自定义组件
588   @Component
589   struct MyComponent {
590     navDesInfo: uiObserver.NavDestinationInfo | undefined
591
592     aboutToAppear(): void {
593       this.navDesInfo = this.queryNavDestinationInfo();
594     }
595
596     build() {
597         Column() {
598           Text("所属页面Name: " + this.navDesInfo?.name)
599         }.width('100%').height('100%')
600     }
601   }
602  ```
603- 页面状态监听
604
605  通过[observer.on('navDestinationUpdate')](../reference/apis-arkui/js-apis-arkui-observer.md#observeronnavdestinationupdate)提供的注册接口可以注册NavDestination生命周期变化的监听,使用方式如下:
606
607  ```ts
608  uiObserver.on('navDestinationUpdate', (info) => {
609       console.info('NavDestination state update', JSON.stringify(info));
610   });
611  ```
612
613  也可以注册页面切换的状态回调,能在页面发生路由切换的时候拿到对应的页面信息[NavDestinationSwitchInfo](..//reference/apis-arkui/js-apis-arkui-observer.md#navdestinationswitchinfo12),并且提供了UIAbilityContext和UIContext不同范围的监听:
614
615  ```ts
616   // 在UIAbility中使用
617   import { UIContext, uiObserver } from '@kit.ArkUI';
618
619   // callBackFunc 是开发者定义的监听回调函数
620   function callBackFunc(info: uiObserver.NavDestinationSwitchInfo) {}
621   uiObserver.on('navDestinationSwitch', this.context, callBackFunc);
622
623   // 可以通过窗口的getUIContext()方法获取对应的UIContent
624   uiContext: UIContext | null = null;
625   uiObserver.on('navDestinationSwitch', this.uiContext, callBackFunc);
626  ```
627
628## 页面转场
629
630Navigation默认提供了页面切换的转场动画,通过页面栈操作时,会触发不同的转场效果(API version 13之前,Dialog类型的页面默认无转场动画。从API version13开始,Dialog类型的页面支持系统转场动画。),Navigation也提供了关闭系统转场、自定义转场以及共享元素转场的能力。
631
632### 关闭转场
633
634- 全局关闭
635
636  Navigation通过NavPathStack中提供的[disableAnimation](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#disableanimation11)方法可以在当前Navigation中关闭或打开所有转场动画。
637  ```ts
638  pageStack: NavPathStack = new NavPathStack()
639
640  aboutToAppear(): void {
641    this.pageStack.disableAnimation(true)
642  }
643  ```
644- 单次关闭
645
646  NavPathStack中提供的Push、Pop、Replace等接口中可以设置animated参数,默认为true表示有转场动画,需要单次关闭转场动画可以置为false,不影响下次转场动画。
647  ```ts
648  pageStack: NavPathStack = new NavPathStack()
649
650  this.pageStack.pushPath({ name: "PageOne" }, false)
651  this.pageStack.pop(false)
652  ```
653
654### 自定义转场
655
656- Navigation自定义转场
657
658  Navigation自定义转场动画能力通过[customNavContentTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#customnavcontenttransition11)事件提供,可以通过以下三步定义自定义转场动画:
659
660  1. 构建一个自定义转场动画工具类CustomNavigationUtils,通过一个Map管理各页面的自定义动画对象CustomTransition。页面在创建时注册其自定义转场动画对象,在销毁时取消注册。
661  2. 实现一个转场协议对象[NavigationAnimatedTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navigationanimatedtransition11)。其中,timeout属性表示转场结束的超时时间,默认为1000ms,transition属性为自定义的转场动画方法。开发者需在此实现自己的转场动画逻辑,系统在转场开始时会调用此方法,onTransitionEnd为转场结束时的回调。
662  3. 调用customNavContentTransition方法并返回实现的转场协议对象,若返回undefined,则使用系统默认转场。
663
664  具体示例代码可参考[Navigation自定义转场示例](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#示例3设置可交互转场动画)。
665
666- NavDestination自定义转场
667
668  NavDestination支持自定义转场动画,通过设置[customTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#customtransition15)属性即可实现单个页面的自定义转场效果。要实现这一功能,需完成以下步骤:
669
670  1. 实现[NavDestination的转场代理](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#navdestinationtransitiondelegate15),针对不同的堆栈操作类型返回自定义的转场协议对象[NavDestinationTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#navdestinationtransition15)。其中,event是必填参数,需在此处编写自定义转场动画的逻辑;而onTransitionEnd、duration、curve与delay为可选参数,分别对应动画结束后的回调、动画持续时间、动画曲线类型与开始前的延时。若在转场代理中返回多个转场协议对象,这些动画效果将逐层叠加。
671  2. 通过调用NavDestination组件的customTransition属性,并传入上述实现的转场代理,完成自定义转场的设置。
672
673  具体示例代码可以参考[NavDestination自定义转场示例](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#示例2设置navdestination自定义转场)。
674
675### 共享元素转场
676
677NavDestination之间切换时可以通过[geometryTransition](../reference/apis-arkui/arkui-ts/ts-transition-animation-geometrytransition.md#geometrytransition)实现共享元素转场。配置了共享元素转场的页面同时需要关闭系统默认的转场动画。
6781. 为需要实现共享元素转场的组件添加geometryTransition属性,id参数必须在两个NavDestination之间保持一致。
679
680    ```ts
681    // 起始页配置共享元素id
682    NavDestination() {
683      Column() {
684        // ...
685        Image($r('app.media.startIcon'))
686        .geometryTransition('sharedId')
687        .width(100)
688        .height(100)
689      }
690    }
691    .title('FromPage')
692
693    // 目的页配置共享元素id
694    NavDestination() {
695      Column() {
696        // ...
697        Image($r('app.media.startIcon'))
698        .geometryTransition('sharedId')
699        .width(200)
700        .height(200)
701      }
702    }
703    .title('ToPage')
704    ```
705
7062. 将页面路由的操作,放到animateTo动画闭包中,配置对应的动画参数以及关闭系统默认的转场。
707
708    ```ts
709    NavDestination() {
710      Column() {
711        Button('跳转目的页')
712        .width('80%')
713        .height(40)
714        .margin(20)
715        .onClick(() => {
716            this.getUIContext()?.animateTo({ duration: 1000 }, () => {
717              this.pageStack.pushPath({ name: 'ToPage' }, false)
718            })
719        })
720      }
721    }
722    .title('FromPage')
723    ```
724
725## 跨包动态路由
726
727通过静态import页面再进行路由跳转的方式会造成不同模块之间的依赖耦合,以及首页加载时间长等问题。
728
729动态路由设计的初衷旨在解决多个模块(HAR/HSP)能够复用相同的业务逻辑,实现各业务模块间的解耦,同时支持路由功能的扩展与整合。
730
731**动态路由的优势:**
732
733- 路由定义除了跳转的URL以外,可以丰富的配置扩展信息,如横竖屏默认模式,是否需要鉴权等等,做路由跳转时统一处理。
734- 给每个路由页面设置一个名字,按照名称进行跳转而不是文件路径。
735- 页面的加载可以使用动态import(按需加载),防止首个页面加载大量代码导致卡顿。
736
737动态路由提供[系统路由表](#系统路由表)和[自定义路由表](#自定义路由表)两种实现方式。
738
739- 系统路由表相对自定义路由表,使用更简单,只需要添加对应页面跳转配置项,即可实现页面跳转。
740
741- 自定义路由表使用起来更复杂,但是可以根据应用业务进行定制处理。
742
743支持自定义路由表和系统路由表混用。
744
745### 系统路由表
746
747系统路由表是动态路由的一种实现方式。从API version 12开始,Navigation支持使用系统路由表的方式进行动态路由。各业务模块([HSP](../quick-start/in-app-hsp.md)/[HAR](../quick-start/har-package.md))中需要独立配置route_map.json文件,在触发路由跳转时,应用只需要通过NavPathStack提供的路由方法,传入需要路由的页面配置名称,此时系统会自动完成路由模块的动态加载、页面组件构建,并完成路由跳转,从而实现了开发层面的模块解耦。系统路由表不支持预览器及模拟器。其主要步骤如下:
748
7491. 在跳转目标模块的配置文件module.json5添加路由表配置:
750
751   ```json
752     {
753       "module" : {
754         "routerMap": "$profile:route_map"
755       }
756     }
757   ```
7582. 添加完路由配置文件地址后,需要在工程resources/base/profile中创建route_map.json文件。添加如下配置信息:
759
760   ```json
761     {
762       "routerMap": [
763         {
764           "name": "PageOne",
765           "pageSourceFile": "src/main/ets/pages/PageOne.ets",
766           "buildFunction": "PageOneBuilder",
767           "data": {
768             "description" : "this is PageOne"
769           }
770         }
771       ]
772     }
773   ```
774
775    配置说明如下:
776
777   | 配置项 | 说明 |
778   |---|---|
779   | name | 跳转页面名称。|
780   | pageSourceFile | 跳转目标页在包内的路径,相对src目录的相对路径。|
781   | buildFunction | 跳转目标页的入口函数名称,必须以@Builder修饰。 |
782   | data | 应用自定义字段。可以通过配置项读取接口getConfigInRouteMap获取。|
783
7843. 在跳转目标页面中,需要配置入口Builder函数,函数名称需要和route_map.json配置文件中的buildFunction保持一致,否则在编译时会报错。
785
786   ```ts
787     // 跳转页面入口函数
788     @Builder
789     export function PageOneBuilder() {
790       PageOne()
791     }
792
793     @Component
794     struct PageOne {
795       pathStack: NavPathStack = new NavPathStack()
796
797       build() {
798         NavDestination() {
799         }
800         .title('PageOne')
801         .onReady((context: NavDestinationContext) => {
802            this.pathStack = context.pathStack
803         })
804       }
805     }
806   ```
8074. 通过pushPathByName等路由接口进行页面跳转。(注意:此时Navigation中可以不用配置navDestination属性。)
808
809   ```ts
810     @Entry
811     @Component
812     struct Index {
813       pageStack : NavPathStack = new NavPathStack();
814
815       build() {
816         Navigation(this.pageStack){
817         }.onAppear(() => {
818           this.pageStack.pushPathByName("PageOne", null, false);
819         })
820         .hideNavBar(true)
821       }
822     }
823   ```
824
825### 自定义路由表
826
827自定义路由表是动态路由的一种实现方式。开发者可以通过自定义路由表的方式来实现跨包动态路由,具体实现方法请参考<!--RP1-->[Navigation自定义动态路由](https://gitee.com/openharmony/applications_app_samples/tree/master/code/BasicFeature/ApplicationModels/DynamicRouter)<!--RP1End--> 示例。
828
829**实现方案:**
830
8311. 定义页面跳转配置项。
832   - 使用资源文件进行定义,通过资源管理[@ohos.resourceManager](../reference/apis-localization-kit/js-apis-resource-manager.md)在运行时对资源文件解析。
833   - 在ets文件中配置路由加载配置项,一般包括路由页面名称(即pushPath等接口中页面的别名),文件所在模块名称(hsp/har的模块名),加载页面在模块内的路径(相对src目录的路径)。
8342. 加载目标跳转页面,通过[动态import](../arkts-utils/arkts-dynamic-import.md)将跳转目标页面所在的模块在运行时加载,在模块加载完成后,调用模块中的方法,通过import在模块的方法中加载模块中显示的目标页面,并返回页面加载完成后定义的Builder函数。
8353. 触发页面跳转,在Navigation的[navDestination](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navdestination10)属性执行步骤2中加载的Builder函数,即可跳转到目标页面。
836<!--RP2--><!--RP2End-->