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  41 42 将mode属性设置为NavigationMode.Stack,Navigation组件即可设置为单页面显示模式。 43 44 45 ```ts 46 Navigation() { 47 // ... 48 } 49 .mode(NavigationMode.Stack) 50 ``` 51 52  53 54- 分栏模式 55 56 分栏模式适用于宽屏设备,分为左右两部分,发生路由跳转时,只有右边子页会被替换。 57 58 **图2** 分栏布局示意图 59 60  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  208 209 210## 设置标题栏模式 211 212标题栏在界面顶部,用于呈现界面名称和操作入口,Navigation组件通过titleMode属性设置标题栏模式。 213 214> **说明:** 215> Navigation或NavDestination未设置主副标题并且没有返回键时,不显示标题栏。 216 217- Mini模式 218 219 普通型标题栏,用于一级页面不需要突出标题的场景。 220 221 **图3** Mini模式标题栏 222 223  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  241 242 243 ```ts 244 Navigation() { 245 // ... 246 } 247 .titleMode(NavigationTitleMode.Full) 248 ``` 249 250 251## 设置菜单栏 252 253菜单栏位于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个图标,多余的图标会被放入自动生成的更多图标。 254 255**图5** 设置了3个图标的菜单栏 256 257 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 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 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_SINGLETON或LaunchMode.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  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 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 1197<!--RP2--><!--RP2End-->