• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1# 组件导航(Navigation) (推荐)
2<!--Kit: ArkUI-->
3<!--Subsystem: ArkUI-->
4<!--Owner: @mayaolll-->
5<!--Designer: @jiangdayuan-->
6<!--Tester: @lxl007-->
7<!--Adviser: @HelloCrease-->
8
9组件导航(Navigation)主要用于实现Navigation页面(NavDestination)间的跳转,支持在不同Navigation页面间传递参数,提供灵活的跳转栈操作,从而更便捷地实现对不同页面的访问和复用。本文将从组件导航(Navigation)的显示模式、路由操作、子页面管理、跨包跳转以及跳转动效等几个方面进行详细介绍。
10
11[Navigation](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md)是路由导航的根视图容器,一般作为页面(@Entry)的根容器,包括单栏(Stack)、分栏(Split)和自适应(Auto)三种显示模式。Navigation组件适用于模块内和跨模块的路由切换,通过组件级路由能力实现更加自然流畅的转场体验,并提供多种标题栏样式来呈现更好的标题和内容联动效果。一次开发,多端部署场景下,Navigation组件能够自动适配窗口显示大小,在窗口较大的场景下自动切换分栏展示效果。
12
13Navigation组件主要包含​导航页和子页。导航页由标题栏(包含菜单栏)、内容区和工具栏组成,可以通过[hideNavBar](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#hidenavbar9)属性进行隐藏,导航页不存在[路由栈](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navpathstack10)中,与子页,以及子页之间可以通过路由操作进行切换。
14
15在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)实现页面路由。
16
17
18## 设置页面显示模式
19
20Navigation组件通过mode属性设置页面的显示模式。
21
22- 自适应模式
23
24  Navigation组件默认为自适应模式,此时mode属性为NavigationMode.Auto。自适应模式下,当页面宽度大于等于一定阈值( API version 9及以前:520vp,API version 10及以后:600vp )时,Navigation组件采用分栏模式,反之采用单栏模式。
25
26
27  ```
28  Navigation() {
29    // ...
30  }
31  .mode(NavigationMode.Auto)
32  ```
33
34- 单页面模式
35
36  单页面模式适用于窄屏设备,发生路由跳转时,整个页面都会被替换。
37
38    **图1** 单页面布局示意图  
39
40  ![zh-cn_image_0000001511740532](figures/zh-cn_image_0000001511740532.png)
41
42  将mode属性设置为NavigationMode.Stack,Navigation组件即可设置为单页面显示模式。
43
44
45  ```ts
46  Navigation() {
47    // ...
48  }
49  .mode(NavigationMode.Stack)
50  ```
51
52  ![导航单栏模式](figures/导航单栏模式.jpg)
53
54- 分栏模式
55
56  分栏模式适用于宽屏设备,分为左右两部分,发生路由跳转时,只有右边子页会被替换。
57
58  **图2** 分栏布局示意图
59
60  ![zh-cn_image_0000001562820845](figures/zh-cn_image_0000001562820845.png)
61
62  将mode属性设置为NavigationMode.Split,Navigation组件即可设置为分栏显示模式。
63
64
65  ```ts
66  @Entry
67  @Component
68  struct NavigationExample {
69    @State toolTmp: ToolbarItem = {
70      'value': "func",
71      'icon': "./image/ic_public_highlights.svg",  // 当前目录image文件夹下的图标资源
72      'action': () => {}
73    }
74    @Provide('pageInfos') pageInfos: NavPathStack = new NavPathStack()
75    private arr: number[] = [1, 2, 3];
76
77    @Builder
78    pageMap(name: string) {
79      if (name === "NavDestinationTitle1") {
80        pageOneTmp();
81      } else if (name === "NavDestinationTitle2") {
82        pageTwoTmp();
83      } else if (name === "NavDestinationTitle3") {
84        pageThreeTmp();
85      }
86    }
87
88    build() {
89      Column() {
90        Navigation(this.pageInfos) {
91          TextInput({ placeholder: 'search...' })
92            .width("90%")
93            .height(40)
94            .backgroundColor('#FFFFFF')
95
96          List({ space: 12 }) {
97            ForEach(this.arr, (item: number) => {
98              ListItem() {
99                Text("Page" + item)
100                  .width("100%")
101                  .height(72)
102                  .backgroundColor('#FFFFFF')
103                  .borderRadius(24)
104                  .fontSize(16)
105                  .fontWeight(500)
106                  .textAlign(TextAlign.Center)
107                  .onClick(() => {
108                    this.pageInfos.pushPath({ name: "NavDestinationTitle" + item });
109                  })
110              }
111            }, (item: number) => item.toString())
112          }
113          .width("90%")
114          .margin({ top: 12 })
115        }
116        .title("主标题")
117        .mode(NavigationMode.Split)
118        .navDestination(this.pageMap)
119        .menus([
120          {
121            value: "", icon: "./image/ic_public_search.svg", action: () => {
122            }
123          },
124          {
125            value: "", icon: "./image/ic_public_add.svg", action: () => {
126            }
127          },
128          {
129            value: "", icon: "./image/ic_public_add.svg", action: () => {
130            }
131          },
132          {
133            value: "", icon: "./image/ic_public_add.svg", action: () => {
134            }
135          },
136          {
137            value: "", icon: "./image/ic_public_add.svg", action: () => {
138            }
139          }
140        ])
141        .toolbarConfiguration([this.toolTmp, this.toolTmp, this.toolTmp])
142      }
143      .height('100%')
144      .width('100%')
145      .backgroundColor('#F1F3F5')
146    }
147  }
148
149  // PageOne.ets
150  @Component
151  export struct pageOneTmp {
152    @Consume('pageInfos') pageInfos: NavPathStack;
153
154    build() {
155      NavDestination() {
156        Column() {
157          Text("NavDestinationContent1")
158        }.width('100%').height('100%')
159      }.title("NavDestinationTitle1")
160      .onBackPressed(() => {
161        const popDestinationInfo = this.pageInfos.pop(); // 弹出路由栈栈顶元素
162        console.log('pop' + '返回值' + JSON.stringify(popDestinationInfo));
163        return true;
164      })
165    }
166  }
167
168  // PageTwo.ets
169  @Component
170  export struct pageTwoTmp {
171    @Consume('pageInfos') pageInfos: NavPathStack;
172
173    build() {
174      NavDestination() {
175        Column() {
176          Text("NavDestinationContent2")
177        }.width('100%').height('100%')
178      }.title("NavDestinationTitle2")
179      .onBackPressed(() => {
180        const popDestinationInfo = this.pageInfos.pop(); // 弹出路由栈栈顶元素
181        console.log('pop' + '返回值' + JSON.stringify(popDestinationInfo));
182        return true;
183      })
184    }
185  }
186
187  // PageThree.ets
188  @Component
189  export struct pageThreeTmp {
190    @Consume('pageInfos') pageInfos: NavPathStack;
191
192    build() {
193      NavDestination() {
194        Column() {
195          Text("NavDestinationContent3")
196        }.width('100%').height('100%')
197      }.title("NavDestinationTitle3")
198      .onBackPressed(() => {
199        const popDestinationInfo = this.pageInfos.pop(); // 弹出路由栈栈顶元素
200        console.log('pop' + '返回值' + JSON.stringify(popDestinationInfo));
201        return true;
202      })
203    }
204  }
205  ```
206
207  ![导航分栏模式](figures/导航分栏模式.jpg)
208
209
210## 设置标题栏模式
211
212标题栏在界面顶部,用于呈现界面名称和操作入口,Navigation组件通过titleMode属性设置标题栏模式。
213
214> **说明:**
215> Navigation或NavDestination未设置主副标题并且没有返回键时,不显示标题栏。
216
217- Mini模式
218
219  普通型标题栏,用于一级页面不需要突出标题的场景。
220
221  **图3** Mini模式标题栏  
222
223  ![mini](figures/mini.jpg)
224
225
226  ```ts
227  Navigation() {
228    // ...
229  }
230  .titleMode(NavigationTitleMode.Mini)
231  ```
232
233
234- Full模式
235
236  强调型标题栏,用于一级页面需要突出标题的场景。
237
238    **图4** Full模式标题栏  
239
240  ![free1](figures/free1.jpg)
241
242
243  ```ts
244  Navigation() {
245    // ...
246  }
247  .titleMode(NavigationTitleMode.Full)
248  ```
249
250
251## 设置菜单栏
252
253菜单栏位于Navigation组件的右上角,开发者可以通过menus属性进行设置。menus支持Array&lt;[NavigationMenuItem](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navigationmenuitem)&gt;和[CustomBuilder](../reference/apis-arkui/arkui-ts/ts-types.md#custombuilder8)两种参数类型。使用Array&lt;NavigationMenuItem&gt;类型时,竖屏最多支持显示3个图标,横屏最多支持显示5个图标,多余的图标会被放入自动生成的更多图标。
254
255**图5** 设置了3个图标的菜单栏  
256
257![菜单栏2](figures/菜单栏2.jpg)
258
259```ts
260let TooTmp: NavigationMenuItem = {'value': "", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
261Navigation() {
262  // ...
263}
264.menus([TooTmp,
265  TooTmp,
266  TooTmp])
267```
268
269图片也可以引用resources中的资源。
270
271```ts
272let TooTmp: NavigationMenuItem = {'value': "", 'icon': "resources/base/media/ic_public_highlights.svg", 'action': ()=> {}}
273Navigation() {
274  // ...
275}
276.menus([TooTmp,
277  TooTmp,
278  TooTmp])
279```
280
281**图6** 设置了4个图标的菜单栏  
282
283![菜单栏](figures/菜单栏.jpg)
284
285```ts
286let TooTmp: NavigationMenuItem = {'value': "", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
287Navigation() {
288  // ...
289}
290// 竖屏最多支持显示3个图标,多余的图标会被放入自动生成的更多图标。
291.menus([TooTmp,
292  TooTmp,
293  TooTmp,
294  TooTmp])
295```
296
297
298## 设置工具栏
299
300工具栏位于Navigation组件的底部,开发者可以通过[toolbarConfiguration](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#toolbarconfiguration10)属性进行设置。
301
302
303  **图7** 工具栏  
304
305![free3](figures/free3.jpg)
306
307```ts
308let TooTmp: ToolbarItem = {'value': "func", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}};
309let TooBar: ToolbarItem[] = [TooTmp,TooTmp,TooTmp];
310Navigation() {
311  // ...
312}
313.toolbarConfiguration(TooBar)
314```
315
316## 路由操作
317
318Navigation路由相关的操作都是基于导航控制器[NavPathStack](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navpathstack10)提供的方法进行,每个Navigation都需要创建并传入一个NavPathStack对象,用于管理页面。主要涉及页面跳转、页面返回、页面替换、页面删除、参数获取、路由拦截等功能。
319
320从API version 12开始,导航控制器允许被继承。开发者可以在派生类中自定义属性和方法,也可以重写父类的方法。派生类对象可以替代基类NavPathStack对象使用。Navigation中的NavDestination页面存在于NavPathStack中,以栈的结构管理,我们称为路由栈。具体示例代码参见:[导航控制器继承示例代码](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#示例10定义导航控制器派生类)。
321
322> **说明:**
323>
324> 1.不建议开发者通过监听生命周期的方式管理自己的路由栈。
325>
326> 2.在应用处于后台状态下,调用NavPathStack的栈操作方法,会在应用再次回到前台状态时触发刷新。
327
328```ts
329@Entry
330@Component
331struct Index {
332  // 创建一个导航控制器对象并传入Navigation
333  pageStack: NavPathStack = new NavPathStack();
334
335  build() {
336    Navigation(this.pageStack) {
337    }
338    .title('Main')
339  }
340}
341```
342
343### 页面跳转
344
345NavPathStack通过Push相关的接口去实现页面跳转的功能,主要分为以下三类:
346
3471. 普通跳转,通过页面的name去跳转,并可以携带param。
348
349    ```ts
350    this.pageStack.pushPath({ name: "PageOne", param: "PageOne Param" });
351    this.pageStack.pushPathByName("PageOne", "PageOne Param");
352    ```
353
3542. 带返回回调的跳转,跳转时添加onPop回调,能在页面出栈时获取返回信息,并进行处理。
355
356    ```ts
357    this.pageStack.pushPathByName('PageOne', "PageOne Param", (popInfo) => {
358      console.log('Pop page name is: ' + popInfo.info.name + ', result: ' + JSON.stringify(popInfo.result));
359    });
360    ```
361
3623. 带错误码的跳转,跳转结束会触发异步回调,返回错误码信息。
363
364    ```ts
365    this.pageStack.pushDestination({name: "PageOne", param: "PageOne Param"})
366      .catch((error: BusinessError) => {
367        console.error(`Push destination failed, error code = ${error.code}, error.message = ${error.message}.`);
368      }).then(() => {
369        console.info('Push destination succeed.');
370      });
371    this.pageStack.pushDestinationByName("PageOne", "PageOne Param")
372      .catch((error: BusinessError) => {
373        console.error(`Push destination failed, error code = ${error.code}, error.message = ${error.message}.`);
374      }).then(() => {
375        console.info('Push destination succeed.');
376      });
377    ```
378
379### 页面返回
380
381NavPathStack通过Pop相关接口去实现页面返回功能。
382
383```ts
384// 返回到上一页
385this.pageStack.pop();
386// 返回到上一个PageOne页面
387this.pageStack.popToName("PageOne");
388// 返回到索引为1的页面
389this.pageStack.popToIndex(1);
390// 返回到根首页(清除栈中所有页面)
391this.pageStack.clear();
392```
393
394### 页面替换
395
396NavPathStack通过Replace相关接口去实现页面替换功能。
397
398```ts
399// 将栈顶页面替换为PageOne
400this.pageStack.replacePath({ name: "PageOne", param: "PageOne Param" });
401this.pageStack.replacePathByName("PageOne", "PageOne Param");
402// 带错误码的替换,跳转结束会触发异步回调,返回错误码信息
403this.pageStack.replaceDestination({name: "PageOne", param: "PageOne Param"})
404  .catch((error: BusinessError) => {
405    console.error(`Replace destination failed, error code = ${error.code}, error.message = ${error.message}.`);
406  }).then(() => {
407    console.info('Replace destination succeed.');
408  })
409```
410
411### 页面删除
412
413NavPathStack通过Remove相关接口去实现删除路由栈中特定页面的功能。
414
415```ts
416// 删除栈中name为PageOne的所有页面
417this.pageStack.removeByName("PageOne");
418// 删除指定索引的页面
419this.pageStack.removeByIndexes([1, 3, 5]);
420// 删除指定id的页面
421this.pageStack.removeByNavDestinationId("1");
422```
423
424### 移动页面
425
426NavPathStack通过Move相关接口去实现移动路由栈中特定页面到栈顶的功能。
427
428```ts
429// 移动栈中name为PageOne的页面到栈顶
430this.pageStack.moveToTop("PageOne");
431// 移动栈中索引为1的页面到栈顶
432this.pageStack.moveIndexToTop(1);
433```
434
435### 参数获取
436
437NavDestination子页第一次创建时会触发[onReady](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#onready11)回调,可以获取此页面对应的参数。
438
439```ts
440@Component
441struct Page01 {
442  pathStack: NavPathStack | undefined = undefined;
443  pageParam: string = '';
444
445  build() {
446    NavDestination() {
447      // ...
448    }.title('Page01')
449    .onReady((context: NavDestinationContext) => {
450      this.pathStack = context.pathStack;
451      this.pageParam = context.pathInfo.param as string;
452    })
453  }
454}
455```
456
457NavDestination组件中可以通过设置[onResult](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#onresult15)接口,接收返回时传递的路由参数。
458
459```ts
460class NavParam {
461  desc: string = 'navigation-param'
462}
463
464@Component
465struct DemoNavDestination {
466  // ...
467  build() {
468    NavDestination() {
469      // ...
470    }
471    .onResult((param: Object) => {
472      if (param instanceof NavParam) {
473        console.log('TestTag', 'get NavParam, its desc: ' + (param as NavParam).desc);
474        return;
475      }
476      console.log('TestTag', 'param not instance of NavParam');
477    })
478  }
479}
480```
481
482其他业务场景,可以通过主动调用NavPathStack的Get相关接口去获取指定页面的参数。
483
484```ts
485// 获取栈中所有页面name集合
486this.pageStack.getAllPathName();
487// 获取索引为1的页面参数
488this.pageStack.getParamByIndex(1);
489// 获取PageOne页面的参数
490this.pageStack.getParamByName("PageOne");
491// 获取PageOne页面的索引集合
492this.pageStack.getIndexByName("PageOne");
493```
494
495### 路由拦截
496
497NavPathStack提供了[setInterception](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#setinterception12)方法,用于设置Navigation页面跳转拦截回调。该方法需要传入一个NavigationInterception对象,该对象包含三个回调函数:
498
499| 名称       | 描述                                                 |
500| ------------ | ------------------------------------------------------ |
501| willShow   | 页面跳转前回调,允许操作栈,在当前跳转生效。       |
502| didShow    | 页面跳转后回调,在该回调中操作栈会在下一次跳转生效。 |
503| modeChange | Navigation单双栏显示状态发生变更时触发该回调。  |
504
505> **说明:**
506>
507> 无论是哪个回调,在进入回调时路由栈都已经发生了变化。
508
509开发者可以在willShow回调中通过修改路由栈来实现路由拦截重定向的能力。
510
511```ts
512this.pageStack.setInterception({
513  willShow: (from: NavDestinationContext | "navBar", to: NavDestinationContext | "navBar",
514    operation: NavigationOperation, animated: boolean) => {
515    if (typeof to === "string") {
516      console.log("target page is navigation home page.");
517      return;
518    }
519    // 将跳转到PageTwo的路由重定向到PageOne
520    let target: NavDestinationContext = to as NavDestinationContext;
521    if (target.pathInfo.name === 'PageTwo') {
522      target.pathStack.pop();
523      target.pathStack.pushPathByName('PageOne', null);
524    }
525  }
526})
527```
528
529### 单例跳转
530
531通过设置[LaunchMode](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#launchmode12枚举说明)为LaunchMode.MOVE_TO_TOP_SINGLETONLaunchMode.POP_TO_SINGLETON,可以实现Navigation路由栈的单实例跳转。单实例跳转的规则如下:
532
5331. 当指定为LaunchMode.MOVE_TO_TOP_SINGLETON时,系统会从栈底到栈顶查找具有指定名称的NavDestination。找到后,该页面将被移动到栈顶(replace操作会用指定的NavDestination替换当前栈顶)。
5342. 若指定为LaunchMode.POP_TO_SINGLETON,系统同样会从栈底到栈顶查找具有指定名称的NavDestination。找到后,便会移除该NavDestination上方的所有页面(replace操作会用指定的NavDestination替换当前栈顶)。
535
536当栈中存在的NavDestination页面通过单实例方式移动到栈顶时,将触发[onNewParam](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#onnewparam19)回调。
537
538有关单实例跳转的示例代码,可以参考[Navigation单例跳转示例](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#示例2使用导航控制器方法)。
539
540## 子页面
541
542[NavDestination](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md)是Navigation子页面的根容器,用于承载子页面的一些特殊属性以及生命周期等。NavDestination可以设置独立的标题栏和菜单栏等属性,使用方法与Navigation相同。NavDestination也可以通过mode属性设置不同的显示类型,用于满足不同页面的诉求。
543
544### 页面显示类型
545
546- 标准类型
547
548  NavDestination组件默认为标准类型,此时mode属性为NavDestinationMode.STANDARD。标准类型的NavDestination的生命周期跟随其在NavPathStack路由栈中的位置变化而改变。
549
550- 弹窗类型
551
552  NavDestination设置mode为NavDestinationMode.DIALOG弹窗类型,此时整个NavDestination默认透明显示。弹窗类型的NavDestination显示和消失时不会影响下层标准类型的NavDestination的显示和生命周期,两者可以同时显示。
553
554  ```ts
555  // Dialog NavDestination
556  @Entry
557  @Component
558   struct Index {
559     @Provide('NavPathStack') pageStack: NavPathStack = new NavPathStack();
560
561     @Builder
562     PagesMap(name: string) {
563       if (name == 'DialogPage') {
564         DialogPage();
565       }
566     }
567
568     build() {
569       Navigation(this.pageStack) {
570         Button('Push DialogPage')
571           .margin(20)
572           .width('80%')
573           .onClick(() => {
574             this.pageStack.pushPathByName('DialogPage', '');
575           })
576       }
577       .mode(NavigationMode.Stack)
578       .title('Main')
579       .navDestination(this.PagesMap)
580     }
581   }
582
583   @Component
584   export struct DialogPage {
585     @Consume('NavPathStack') pageStack: NavPathStack;
586
587     build() {
588       NavDestination() {
589         Stack({ alignContent: Alignment.Center }) {
590           Column() {
591             Text("Dialog NavDestination")
592               .fontSize(20)
593               .margin({ bottom: 100 })
594             Button("Close").onClick(() => {
595               this.pageStack.pop();
596             }).width('30%')
597           }
598           .justifyContent(FlexAlign.Center)
599           .backgroundColor(Color.White)
600           .borderRadius(10)
601           .height('30%')
602           .width('80%')
603         }.height("100%").width('100%')
604       }
605       .backgroundColor('rgba(0,0,0,0.5)')
606       .hideTitleBar(true)
607       .mode(NavDestinationMode.DIALOG)
608     }
609   }
610  ```
611  ![dialog_navdestination](figures/dialog_navdestination.png)
612
613### 页面生命周期
614
615Navigation作为路由容器,其生命周期承载在NavDestination组件上,以组件事件的形式开放。
616
617其生命周期大致可分为三类,自定义组件生命周期、通用组件生命周期和自有生命周期。其中,[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独有。
618
619生命周期时序如下图所示:
620
621![navigation_lifecycle](figures/navigation_lifecycle.png)
622
623- **aboutToAppear**:在创建自定义组件后,执行其build()函数之前执行(NavDestination创建之前),允许在该方法中改变状态变量,更改将在后续执行build()函数中生效。
624- **onWillAppear**:NavDestination创建后,挂载到组件树之前执行,在该方法中更改状态变量会在当前帧显示生效。
625- **onAppear**:通用生命周期事件,NavDestination组件挂载到组件树时执行。
626- **onWillShow**:NavDestination组件布局显示之前执行,此时页面不可见(应用切换到前台不会触发)。
627- **onShown**:NavDestination组件布局显示之后执行,此时页面已完成布局。
628- **onActive**:NavDestination处于激活态(处于栈顶可操作,且上层无特殊组件遮挡)触发。
629- **onWillHide**:NavDestination组件触发隐藏之前执行(应用切换到后台不会触发)。
630- **onInactive**:NavDestination组件处于非激活态(处于非栈顶不可操作,或处于栈顶时上层有特殊组件遮挡)触发。
631- **onHidden**:NavDestination组件触发隐藏后执行(非栈顶页面push进栈,栈顶页面pop出栈或应用切换到后台)。
632- **onWillDisappear**:NavDestination组件即将销毁之前执行,如果有转场动画,会在动画前触发(栈顶页面pop出栈)。
633- **onDisappear**:通用生命周期事件,NavDestination组件从组件树上卸载销毁时执行。
634- **aboutToDisappear**:自定义组件析构销毁之前执行,不允许在该方法中改变状态变量。
635
636### 页面监听和查询
637
638为了方便组件跟页面解耦,在NavDestination子页面内部的自定义组件可以通过全局方法监听或查询到页面的一些状态信息。
639
640- 页面信息查询
641
642  自定义组件提供[queryNavDestinationInfo](../reference/apis-arkui/arkui-ts/ts-custom-component-api.md#querynavdestinationinfo)方法,可以在NavDestination内部查询到当前所属页面的信息,返回值为[NavDestinationInfo](../reference/apis-arkui/js-apis-arkui-observer.md#navdestinationinfo),若查询不到则返回undefined。
643
644  ```ts
645   import { uiObserver } from '@kit.ArkUI';
646
647   // NavDestination内的自定义组件
648   @Component
649   struct MyComponent {
650     navDesInfo: uiObserver.NavDestinationInfo | undefined;
651
652     aboutToAppear(): void {
653       this.navDesInfo = this.queryNavDestinationInfo();
654     }
655
656     build() {
657         Column() {
658           Text("所属页面Name: " + this.navDesInfo?.name)
659         }.width('100%').height('100%')
660     }
661   }
662  ```
663- 页面状态监听
664
665  通过[observer.on('navDestinationUpdate')](../reference/apis-arkui/js-apis-arkui-observer.md#uiobserveronnavdestinationupdate)提供的注册接口可以注册NavDestination生命周期变化的监听,使用方式如下:
666
667  ```ts
668  uiObserver.on('navDestinationUpdate', (info) => {
669       console.info('NavDestination state update', JSON.stringify(info));
670   });
671  ```
672
673  也可以注册页面切换的状态回调,能在页面发生路由切换的时候拿到对应的页面信息[NavDestinationSwitchInfo](..//reference/apis-arkui/js-apis-arkui-observer.md#navdestinationswitchinfo12),并且提供了UIAbilityContext和UIContext不同范围的监听:
674
675  ```ts
676   // 在UIAbility中使用
677   import { UIContext, uiObserver } from '@kit.ArkUI';
678
679   // callbackFunc是开发者定义的监听回调函数
680   function callbackFunc(info: uiObserver.NavDestinationSwitchInfo) {}
681   uiObserver.on('navDestinationSwitch', this.context, callbackFunc);
682
683   // 可以通过窗口的getUIContext()方法获取对应的UIContent
684   uiContext: UIContext | null = null;
685   uiObserver.on('navDestinationSwitch', this.uiContext, callbackFunc);
686  ```
687
688## 页面转场
689
690Navigation默认提供了页面切换的转场动画,通过导航控制器操作时,会触发不同的转场效果(API version 13之前,Dialog类型的页面默认无转场动画。从API version13开始,Dialog类型的页面支持系统转场动画。),Navigation也提供了关闭系统转场、自定义转场以及共享元素转场的能力。
691
692### 关闭转场
693
694- 全局关闭
695
696  Navigation通过NavPathStack中提供的[disableAnimation](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#disableanimation11)方法可以在当前Navigation中关闭或打开所有转场动画。
697  ```ts
698  pageStack: NavPathStack = new NavPathStack();
699
700  aboutToAppear(): void {
701    this.pageStack.disableAnimation(true);
702  }
703  ```
704- 单次关闭
705
706  NavPathStack中提供的Push、Pop、Replace等接口中可以设置animated参数,默认为true表示有转场动画,需要单次关闭转场动画可以置为false,不影响下次转场动画。
707  ```ts
708  pageStack: NavPathStack = new NavPathStack();
709
710  this.pageStack.pushPath({ name: "PageOne" }, false);
711  this.pageStack.pop(false);
712  ```
713
714### 自定义转场
715
716- Navigation自定义转场
717
718  Navigation自定义转场动画能力通过[customNavContentTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#customnavcontenttransition11)事件提供,可以通过以下三步定义自定义转场动画:
719
720  1. 构建一个自定义转场动画工具类CustomNavigationUtils,通过一个Map管理各页面的自定义动画对象CustomTransition。页面在创建时注册其自定义转场动画对象,在销毁时取消注册。
721  2. 实现一个转场协议对象[NavigationAnimatedTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navigationanimatedtransition11)。其中,timeout属性表示转场结束的超时时间,默认为1000ms,transition属性为自定义的转场动画方法。开发者需在此实现自己的转场动画逻辑,系统在转场开始时会调用此方法,onTransitionEnd为转场结束时的回调。
722  3. 调用customNavContentTransition方法并返回实现的转场协议对象,若返回undefined,则使用系统默认转场。
723
724  具体示例代码可参考[Navigation自定义转场示例](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#示例3设置可交互转场动画)。
725
726- NavDestination自定义转场
727
728  NavDestination支持自定义转场动画,通过设置[customTransition](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#customtransition15)属性即可实现单个页面的自定义转场效果。要实现这一功能,需完成以下步骤:
729
730  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为可选参数,分别对应动画结束后的回调、动画持续时间、动画曲线类型与开始前的延时。若在转场代理中返回多个转场协议对象,这些动画效果将逐层叠加。
731  2. 通过调用NavDestination组件的customTransition属性,并传入上述实现的转场代理,完成自定义转场的设置。
732
733  具体示例代码可以参考[NavDestination自定义转场示例](../reference/apis-arkui/arkui-ts/ts-basic-components-navdestination.md#示例2设置navdestination自定义转场)。
734
735### 共享元素转场
736
737NavDestination之间切换时可以通过[geometryTransition](../reference/apis-arkui/arkui-ts/ts-transition-animation-geometrytransition.md#geometrytransition)实现共享元素转场。配置了共享元素转场的页面同时需要关闭系统默认的转场动画。
7381. 为需要实现共享元素转场的组件添加geometryTransition属性,id参数必须在两个NavDestination之间保持一致。
739
740    ```ts
741    // 起始页配置共享元素id
742    NavDestination() {
743      Column() {
744        // ...
745        // $r('app.media.startIcon')需要替换为开发者所需的资源文件
746        Image($r('app.media.startIcon'))
747        .geometryTransition('sharedId')
748        .width(100)
749        .height(100)
750      }
751    }
752    .title('FromPage')
753
754    // 目的页配置共享元素id
755    NavDestination() {
756      Column() {
757        // ...
758        // $r('app.media.startIcon')需要替换为开发者所需的资源文件
759        Image($r('app.media.startIcon'))
760        .geometryTransition('sharedId')
761        .width(200)
762        .height(200)
763      }
764    }
765    .title('ToPage')
766    ```
767
7682. 将页面路由的操作,放到animateTo动画闭包中,配置对应的动画参数以及关闭系统默认的转场。
769
770    ```ts
771    NavDestination() {
772      Column() {
773        Button('跳转目的页')
774        .width('80%')
775        .height(40)
776        .margin(20)
777        .onClick(() => {
778            this.getUIContext()?.animateTo({ duration: 1000 }, () => {
779              this.pageStack.pushPath({ name: 'ToPage' }, false)
780            });
781        })
782      }
783    }
784    .title('FromPage')
785    ```
786
787## 跨包动态路由
788
789通过静态import页面再进行路由跳转的方式会造成不同模块之间的依赖耦合,以及首页加载时间长等问题。
790
791动态路由设计的初衷旨在解决多个模块(HAR/HSP)能够复用相同的业务逻辑,实现各业务模块间的解耦,同时支持路由功能的扩展与整合。
792
793**动态路由的优势:**
794
795- 路由定义除了跳转的URL以外,可以丰富的配置扩展信息,如横竖屏默认模式,是否需要鉴权等等,做路由跳转时统一处理。
796- 给每个路由页面设置一个名字,按照名称进行跳转而不是文件路径。
797- 页面的加载可以使用动态import(按需加载),防止首个页面加载大量代码导致卡顿。
798
799动态路由提供[系统路由表](#系统路由表)和[自定义路由表](#自定义路由表)两种实现方式。
800
801- 系统路由表相对自定义路由表,使用更简单,只需要添加对应页面跳转配置项,即可实现页面跳转。
802
803- 自定义路由表使用起来更复杂,但是可以根据应用业务进行定制处理。
804
805支持自定义路由表和系统路由表混用。
806
807### 系统路由表
808
809系统路由表是动态路由的一种实现方式。从API version 12开始,Navigation支持使用系统路由表的方式进行动态路由。各业务模块([HSP](../quick-start/in-app-hsp.md)/[HAR](../quick-start/har-package.md))中需要独立配置route_map.json文件,在触发路由跳转时,应用只需要通过NavPathStack提供的路由方法,传入需要路由的页面配置名称,此时系统会自动完成路由模块的动态加载、页面组件构建,并完成路由跳转,从而实现了开发层面的模块解耦。系统路由表支持模拟器但不支持预览器。其主要步骤如下:
810
8111. 在跳转目标模块的配置文件[module.json5](../quick-start/module-configuration-file.md)添加路由表配置:
812
813   ```json
814     {
815       "module" : {
816         "routerMap": "$profile:route_map"
817       }
818     }
819   ```
8202. 添加完路由配置文件地址后,需要在工程resources/base/profile中创建route_map.json文件。添加如下配置信息:
821
822   ```json
823     {
824       "routerMap": [
825         {
826           "name": "PageOne",
827           "pageSourceFile": "src/main/ets/pages/PageOne.ets",
828           "buildFunction": "PageOneBuilder",
829           "data": {
830             "description" : "this is PageOne"
831           }
832         }
833       ]
834     }
835   ```
836
837    配置说明如下:
838
839   | 配置项 | 说明 |
840   |---|---|
841   | name | 可自定义的跳转页面名称。|
842   | pageSourceFile | 跳转目标页在包内的路径,相对src目录的相对路径。|
843   | buildFunction | 跳转目标页的入口函数名称,必须以@Builder修饰。 |
844   | data | 应用自定义字段。可以通过配置项读取接口getConfigInRouteMap获取。|
845
8463. 在跳转目标页面中,需要配置入口Builder函数,函数名称需要和route_map.json配置文件中的buildFunction保持一致,否则在编译时会报错。
847
848   ```ts
849     // 跳转页面入口函数
850     @Builder
851     export function PageOneBuilder() {
852       PageOne();
853     }
854
855     @Component
856     struct PageOne {
857       pathStack: NavPathStack = new NavPathStack();
858
859       build() {
860         NavDestination() {
861         }
862         .title('PageOne')
863         .onReady((context: NavDestinationContext) => {
864            this.pathStack = context.pathStack;
865         })
866       }
867     }
868   ```
8694. 通过pushPathByName等路由接口进行页面跳转。(注意:此时Navigation中可以不用配置navDestination属性。)
870
871   ```ts
872     @Entry
873     @Component
874     struct Index {
875       pageStack : NavPathStack = new NavPathStack();
876
877       build() {
878         Navigation(this.pageStack){
879         }.onAppear(() => {
880           this.pageStack.pushPathByName("PageOne", null, false);
881         })
882         .hideNavBar(true)
883       }
884     }
885   ```
886
887### 自定义路由表
888
889自定义路由表是动态路由的一种实现方式。开发者可以通过自定义路由表的方式来实现跨包动态路由,具体实现方法请参考<!--RP1-->[Navigation自定义动态路由](https://gitcode.com/openharmony/applications_app_samples/tree/master/code/BasicFeature/ApplicationModels/DynamicRouter)<!--RP1End--> 示例。
890
891**实现方案:**
892
8931. 定义页面跳转配置项。
894   - 使用资源文件进行定义,通过资源管理[@ohos.resourceManager](../reference/apis-localization-kit/js-apis-resource-manager.md)在运行时对资源文件解析。
895   - 在ets文件中配置路由加载配置项,一般包括路由页面名称(即pushPath等接口中页面的别名),文件所在模块名称(hsp/har的模块名),加载页面在模块内的路径(相对src目录的路径)。
8962. 加载目标跳转页面,通过[动态import](../arkts-utils/arkts-dynamic-import.md)将跳转目标页面所在的模块在运行时加载,在模块加载完成后,调用模块中的方法,通过import在模块的方法中加载模块中显示的目标页面,并返回页面加载完成后定义的Builder函数。
8973. 触发页面跳转,在Navigation的[navDestination](../reference/apis-arkui/arkui-ts/ts-basic-components-navigation.md#navdestination10)属性执行步骤2中加载的Builder函数,即可跳转到目标页面。
898
899## 导航示例
900
901### 创建导航首页
902实现步骤为:
903
9041.使用Navigation创建导航主页,并创建导航控制器NavPathStack以此来实现不同页面之间的跳转。
905
9062.在Navigation中增加List组件,来定义导航主页中不同的一级界面。
907
9083.在List内的组件添加onClick方法,并在其中使用导航控制器NavPathStack的pushPathByName方法,使组件可以在点击之后从当前页面跳转到输入参数name在路由表内对应的页面。
909```ts
910@Entry
911@Component
912struct NavigationDemo {
913  @Provide('pathInfos') pathInfos: NavPathStack = new NavPathStack();
914  private listArray: Array<string> = ['WLAN', 'Bluetooth', 'Personal Hotspot', 'Connect & Share'];
915
916  build() {
917    Column() {
918      Navigation(this.pathInfos) {
919        TextInput({ placeholder: '输入关键字搜索' })
920          .width('90%')
921          .height(40)
922          .margin({ bottom: 10 })
923
924        // 通过List定义导航的一级界面
925        List({ space: 12, initialIndex: 0 }) {
926          ForEach(this.listArray, (item: string) => {
927            ListItem() {
928              Row() {
929                Row() {
930                  Text(`${item.slice(0, 1)}`)
931                    .fontColor(Color.White)
932                    .fontSize(14)
933                    .fontWeight(FontWeight.Bold)
934                }
935                .width(30)
936                .height(30)
937                .backgroundColor('#a8a8a8')
938                .margin({ right: 20 })
939                .borderRadius(20)
940                .justifyContent(FlexAlign.Center)
941
942                Column() {
943                  Text(item)
944                    .fontSize(16)
945                    .margin({ bottom: 5 })
946                }
947                .alignItems(HorizontalAlign.Start)
948
949                Blank()
950
951                Row()
952                  .width(12)
953                  .height(12)
954                  .margin({ right: 15 })
955                  .border({
956                    width: { top: 2, right: 2 },
957                    color: 0xcccccc
958                  })
959                  .rotate({ angle: 45 })
960              }
961              .borderRadius(15)
962              .shadow({ radius: 100, color: '#ededed' })
963              .width('90%')
964              .alignItems(VerticalAlign.Center)
965              .padding({ left: 15, top: 15, bottom: 15 })
966              .backgroundColor(Color.White)
967            }
968            .width('100%')
969            .onClick(() => {
970              this.pathInfos.pushPathByName(`${item}`, '详情页面参数'); // 将name指定的NaviDestination页面信息入栈,传递的参数为param
971            })
972          }, (item: string): string => item)
973        }
974        .listDirection(Axis.Vertical)
975        .edgeEffect(EdgeEffect.Spring)
976        .sticky(StickyStyle.Header)
977        .chainAnimation(false)
978        .width('100%')
979      }
980      .width('100%')
981      .mode(NavigationMode.Auto)
982      .title('设置') // 设置标题文字
983    }
984    .size({ width: '100%', height: '100%' })
985    .backgroundColor(0xf4f4f5)
986  }
987}
988```
989
990### 创建导航子页
991导航子页1实现步骤为:
992
9931.使用NavDestination,来创建导航子页PageOne。
994
9952.创建导航控制器NavPathStack并在onReady时进行初始化,获取当前所在的导航控制器,以此来实现不同页面之间的跳转。
996
9973.在子页面内的组件添加onClick,并在其中使用导航控制器NavPathStack的pop方法,使组件可以在点击之后弹出路由栈栈顶元素实现页面的返回。
998
999```ts
1000//PageOne.ets
1001@Builder
1002export function PageOneBuilder(name: string, param: string) {
1003  PageOne({ name: name, value: param });
1004}
1005
1006@Component
1007export struct PageOne {
1008  pathInfos: NavPathStack = new NavPathStack();
1009  name: string = '';
1010  @State value: string = '';
1011
1012  build() {
1013    NavDestination() {
1014      Column() {
1015        Text(`${this.name}设置页面`)
1016          .width('100%')
1017          .fontSize(20)
1018          .fontColor(0x333333)
1019          .textAlign(TextAlign.Center)
1020          .textShadow({
1021            radius: 2,
1022            offsetX: 4,
1023            offsetY: 4,
1024            color: 0x909399
1025          })
1026          .padding({ top: 30 })
1027        Text(`${JSON.stringify(this.value)}`)
1028          .width('100%')
1029          .fontSize(18)
1030          .fontColor(0x666666)
1031          .textAlign(TextAlign.Center)
1032          .padding({ top: 45 })
1033        Button('返回')
1034          .width('50%')
1035          .height(40)
1036          .margin({ top: 50 })
1037          .onClick(() => {
1038            //弹出路由栈栈顶元素,返回上个页面
1039            this.pathInfos.pop();
1040          })
1041      }
1042      .size({ width: '100%', height: '100%' })
1043    }.title(`${this.name}`)
1044    .onReady((ctx: NavDestinationContext) => {
1045      // NavDestinationContext获取当前所在的导航控制器
1046      this.pathInfos = ctx.pathStack;
1047    })
1048  }
1049}
1050```
1051导航子页2实现步骤为:
1052
10531.使用NavDestination,来创建导航子页PageTwo。
1054
10552.创建导航控制器NavPathStack并在onReady时进行初始化,获取当前所在的导航控制器,以此来实现不同页面之间的跳转。
1056
10573.在子页面内的组件添加onClick,并在其中使用导航控制器NavPathStack的pushPathByName方法,使组件可以在点击之后从当前页面跳转到输入参数name在路由表内对应的页面。
1058```ts
1059//PageTwo.ets
1060@Builder
1061export function PageTwoBuilder(name: string) {
1062  PageTwo({ name: name });
1063}
1064
1065@Component
1066export struct PageTwo {
1067  pathInfos: NavPathStack = new NavPathStack();
1068  name: string = '';
1069  private listArray: Array<string> = ['Projection', 'Print', 'VPN', 'Private DNS', 'NFC'];
1070
1071  build() {
1072    NavDestination() {
1073      Column() {
1074        List({ space: 12, initialIndex: 0 }) {
1075          ForEach(this.listArray, (item: string) => {
1076            ListItem() {
1077              Row() {
1078                Row() {
1079                  Text(`${item.slice(0, 1)}`)
1080                    .fontColor(Color.White)
1081                    .fontSize(14)
1082                    .fontWeight(FontWeight.Bold)
1083                }
1084                .width(30)
1085                .height(30)
1086                .backgroundColor('#a8a8a8')
1087                .margin({ right: 20 })
1088                .borderRadius(20)
1089                .justifyContent(FlexAlign.Center)
1090
1091                Column() {
1092                  Text(item)
1093                    .fontSize(16)
1094                    .margin({ bottom: 5 })
1095                }
1096                .alignItems(HorizontalAlign.Start)
1097
1098                Blank()
1099
1100                Row()
1101                  .width(12)
1102                  .height(12)
1103                  .margin({ right: 15 })
1104                  .border({
1105                    width: { top: 2, right: 2 },
1106                    color: 0xcccccc
1107                  })
1108                  .rotate({ angle: 45 })
1109              }
1110              .borderRadius(15)
1111              .shadow({ radius: 100, color: '#ededed' })
1112              .width('90%')
1113              .alignItems(VerticalAlign.Center)
1114              .padding({ left: 15, top: 15, bottom: 15 })
1115              .backgroundColor(Color.White)
1116            }
1117            .width('100%')
1118            .onClick(() => {
1119              this.pathInfos.pushPathByName(`${item}`, '页面设置参数');
1120            })
1121          }, (item: string): string => item)
1122        }
1123        .listDirection(Axis.Vertical)
1124        .edgeEffect(EdgeEffect.Spring)
1125        .sticky(StickyStyle.Header)
1126        .width('100%')
1127      }
1128      .size({ width: '100%', height: '100%' })
1129    }.title(`${this.name}`)
1130    .onReady((ctx: NavDestinationContext) => {
1131      // NavDestinationContext获取当前所在的导航控制器
1132      this.pathInfos = ctx.pathStack;
1133    })
1134  }
1135}
1136```
1137
1138### 创建路由跳转
1139实现步骤为:
1140
11411.工程配置文件[module.json5](../quick-start/module-configuration-file.md)中配置 {"routerMap": "$profile:router_map"}。
1142
11432.router_map.json中配置全局路由表,导航控制器NavPathStack可根据路由表中的name将对应页面信息入栈。
1144```ts
1145{
1146  "routerMap" : [
1147    {
1148      "name" : "WLAN",
1149      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1150      "buildFunction" : "PageOneBuilder"
1151    },
1152    {
1153      "name" : "Bluetooth",
1154      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1155      "buildFunction" : "PageOneBuilder"
1156    },
1157    {
1158      "name" : "Personal Hotspot",
1159      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1160      "buildFunction" : "PageOneBuilder"
1161    },
1162    {
1163      "name" : "Connect & Share",
1164      "pageSourceFile"  : "src/main/ets/pages/PageTwo.ets",
1165      "buildFunction" : "PageTwoBuilder"
1166    },
1167    {
1168      "name" : "Projection",
1169      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1170      "buildFunction" : "PageOneBuilder"
1171    },
1172    {
1173      "name" : "Print",
1174      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1175      "buildFunction" : "PageOneBuilder"
1176    },
1177    {
1178      "name" : "VPN",
1179      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1180      "buildFunction" : "PageOneBuilder"
1181    },
1182    {
1183      "name" : "Private DNS",
1184      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1185      "buildFunction" : "PageOneBuilder"
1186    },
1187    {
1188      "name" : "NFC",
1189      "pageSourceFile"  : "src/main/ets/pages/PageOne.ets",
1190      "buildFunction" : "PageOneBuilder"
1191    }
1192  ]
1193}
1194```
1195
1196![zh-cn_image_0000001588458252](figures/arkts-navigation-transition_1.gif)
1197<!--RP2--><!--RP2End-->