HarmonyOS--路由管理--组件导航 (Navigation)

打印 上一主题 下一主题

主题 878|帖子 878|积分 2634

文档中央

什么是组件导航 (Navigation) ?

1、Navigation是路由容器组件,一般作为首页的根容器,包括单栏(Stack)、分栏(Split)和自顺应(Auto)三种显示模式
2、Navigation组件适用于模块内和跨模块的路由切换,一次开发,多端部署场景。
3、通过组件级路由能力实现更加天然流畅的转场体验,并提供多种标题栏样式来呈现更好的标题和内容联动效果
4、在不同尺寸的装备上,Navigation组件可以或许自顺应显示巨细,主动切换分栏展示效果。
5、Navigation组件告急包含​导航页(NavBar)和子页(NavDestination)
6、导航页由标题栏(Titlebar,包含菜单栏menu)、内容区(Navigation子组件)和工具栏(Toolbar)构成
7、其中导航页可以通过hideNavBar属性进行隐蔽,导航页不存在页面栈中,导航页和子页,以及子页之间可以通过路由操纵进行切换。
设置页面显示模式

自顺应模式

  1. Navigation() {
  2.   ...
  3. }
  4. .mode(NavigationMode.Auto)
复制代码
1、Navigation组件默认为自顺应模式,此时mode属性为NavigationMode.Auto
2、自顺应模式下,当页面宽度大于便是一定阈值( API version 9及从前:520vp,API version 10及以后:600vp )时,Navigation组件采用分栏模式,反之采用单栏模式。
单页面模式

  1. Navigation() {
  2.   ...
  3. }
  4. .mode(NavigationMode.Stack)
复制代码
分栏模式

  1. Navigation() {
  2.   ...
  3. }
  4. .mode(NavigationMode.Split)
复制代码
设置标题栏模式

Mini模式

   平凡型标题栏,用于一级页面不必要突出标题的场景。
  1. Navigation() {
  2.   ...
  3. }
  4. .titleMode(NavigationTitleMode.Mini)
复制代码
Full模式

   强调型标题栏,用于一级页面必要突出标题的场景。
  1. Navigation() {
  2.   ...
  3. }
  4. .titleMode(NavigationTitleMode.Full)
复制代码
设置菜单栏

   1、菜单栏位于Navigation组件的右上角,开发者可以通过menus属性进行设置。
  2、menus支持Array<NavigationMenuItem>和CustomBuilder两种参数类型
  3、使用Array<NavigationMenuItem>类型时,竖屏最多支持显示3个图标,横屏最多支持显示5个图标,多余的图标会被放入主动生成的更多图标。
  1. let TooTmp: NavigationMenuItem = {'value': "", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
  2. // resource中引入
  3. let TooTmp: NavigationMenuItem = {'value': "", 'icon': "resources/base/media/ic_public_highlights.svg", 'action': ()=> {}}
  4. Navigation() {
  5.   ...
  6. }
  7. .menus([TooTmp,
  8.   TooTmp,
  9.   TooTmp])
复制代码
设置工具栏

   工具栏位于Navigation组件的底部,开发者可以通过toolbarConfiguration属性进行设置。
  1. let TooTmp: ToolbarItem = {'value': "func", 'icon': "./image/ic_public_highlights.svg", 'action': ()=> {}}
  2. let TooBar: ToolbarItem[] = [TooTmp,TooTmp,TooTmp]
  3. Navigation() {
  4.   ...
  5. }
  6. .toolbarConfiguration(TooBar)
复制代码
子页面

1、NavDestination是Navigation子页面的根容器,用于承载子页面的一些特殊属性以及生命周期等
2、NavDestination可以设置独立的标题栏和菜单栏等属性,使用方法与Navigation相同。3、NavDestination也可以通过mode属性设置不同的显示类型,用于满足不同页面的诉求。
页面显示类型

标准类型

NavDestination组件默认为标准类型,此时mode属性为NavDestinationMode.STANDARD。标准类型的NavDestination的生命周期跟随其在NavPathStack页面栈中的位置变化而改变。
弹窗类型

NavDestination设置mode为NavDestinationMode.DIALOG弹窗类型,此时整个NavDestination默认透显着示。弹窗类型的NavDestination显示和消散时不会影响下层标准类型的NavDestination的显示和生命周期,两者可以同时显示。
  1. // Dialog NavDestination
  2. @Entry
  3. @Component
  4. struct Index {
  5.    @Provide('NavPathStack') pageStack: NavPathStack = new NavPathStack() // 声明一个页面栈
  6.    @Builder
  7.    PagesMap(name: string) {
  8.      if (name == 'DialogPage') {
  9.        DialogPage()
  10.      }
  11.    }
  12.    build() {
  13.      Navigation(this.pageStack) { // 传入页面栈
  14.        Button('Push DialogPage')
  15.          .margin(20)
  16.          .width('80%')
  17.          .onClick(() => {
  18.            this.pageStack.pushPathByName('DialogPage', ''); // 拉起子页面
  19.          })
  20.      }
  21.      .mode(NavigationMode.Stack)
  22.      .title('Main')
  23.      .navDestination(this.PagesMap) // 挂载子页面
  24.    }
  25. }
  26. @Component
  27. export struct DialogPage {
  28.    @Consume('NavPathStack') pageStack: NavPathStack;
  29.    build() {
  30.      NavDestination() { // 声明子页面
  31.        ...
  32.      }
  33.      .backgroundColor('rgba(0,0,0,0.5)')
  34.      .hideTitleBar(true)
  35.      .mode(NavDestinationMode.DIALOG) // 弹窗类型
  36.    }
  37. }
复制代码
页面生命周期

1、Navigation作为路由容器,其生命周期承载在NavDestination组件上,以组件变乱的情势开放。
2、其生命周期大抵可分为三类,自定义组件生命周期、通用组件生命周期和自有生命周期
3、aboutToAppear和aboutToDisappear是自定义组件的生命周期
4、OnAppear和OnDisappear是组件的通用生命周期
5、剩下的六个生命周期为NavDestination独有(红色标注部门)

NavDestination声明周期
1、onWillAppear:NavDestination创建后,挂载到组件树之前执行,在该方法中更改状态变量会在当前帧显示生效。
2、onWillShow:NavDestination组件结构显示之前执行,此时页面不可见(应用切换到前台不会触发)。
3、onShown:NavDestination组件结构显示之后执行,此时页面已完成结构。
4、onWillHide:NavDestination组件触发隐蔽之前执行(应用切换到背景不会触发)。
5、onHidden:NavDestination组件触发隐蔽后执行(非栈顶页面push进栈,栈顶页面pop出栈或应用切换到背景)。
6、onWillDisappear:NavDestination组件即将销毁之前执行,如果有转场动画,会在动画前触发(栈顶页面pop出栈)
自定义组件生命周期
1、aboutToAppear:在创建自定义组件后,执行其build()函数之前执行(NavDestination创建之前),答应在该方法中改变状态变量,更改将在后续执行build()函数中生效。
2、aboutToDisappear:自定义组件析构销毁之前执行,不答应在该方法中改变状态变量。
通用生命周期
1、onAppear:通用生命周期变乱,NavDestination组件挂载到组件树时执行
2、onDisappear:通用生命周期变乱,NavDestination组件从组件树上卸载销毁时执行。
页面监听和查询

页面状态监听

1、通过@ohos.arkui.observer提供的注册接口可以注册NavDestination生命周期变化的监听,使用方式如下:
  1. observer.on('navDestinationUpdate', (info) => {
  2.      console.info('NavDestination state update', JSON.stringify(info));
  3. });
复制代码
2、也可以注册页面切换的状态回调,能在页面发生路由切换的时候拿到对应的页面信息NavDestinationSwitchInfo,而且提供了UIAbilityContext和UIContext不同范围的监听:
  1. // 在UIAbility中使用
  2. import observer from '@ohos.arkui.observer';
  3. import { UIContext } from '@ohos.arkui.UIContext';
  4. // callBackFunc 是开发者定义的监听回调函数
  5. function callBackFunc(info: observer.NavDestinationSwitchInfo) {}
  6. observer.on('navDestinationSwitch', this.context, callBackFunc);
  7. // 可以通过窗口的getUIContext()方法获取对应的UIContent
  8. uiContext: UIContext | null = null;
  9. observer.on('navDestinationSwitch', this.uiContext, callBackFunc);
复制代码
页面信息查询

自定义组件提供queryNavDestinationInfo方法,可以在NavDestination内部查询到当前所属页面的信息,返回值为NavDestinationInfo,若查询不到则返回undefined。
  1. import observer from '@ohos.arkui.observer';
  2. // NavDestination内的自定义组件
  3. @Component
  4. struct MyComponent {
  5.    navDesInfo: observer.NavDestinationInfo | undefined
  6.    aboutToAppear(): void {
  7.      this.navDesInfo = this.queryNavDestinationInfo();
  8.    }
  9.    build() {
  10.        Column() {
  11.          Text("所属页面Name: " + this.navDesInfo?.name)
  12.        }.width('100%').height('100%')
  13.    }
  14. }
复制代码
路由操纵

1、Navigation路由干系的操纵都是基于页面栈NavPathStack提供的方法进行,每个Navigation都必要创建并传入一个NavPathStack对象,
2、用于管理页面。告急涉及页面跳转、页面返回、页面替换、页面删除、参数获取、路由拦截等功能。
  1. @Entry
  2. @Component
  3. struct Index {
  4.   // 创建一个页面栈对象并传入Navigation
  5.   pageStack: NavPathStack = new NavPathStack()
  6.   build() {
  7.     Navigation(this.pageStack) {
  8.     }
  9.     .title('Main')
  10.   }
  11. }
复制代码
页面跳转

1、平凡跳转,通过页面的name去跳转,并可以携带param。
  1. this.pageStack.pushPath({ name: "PageOne", param: "PageOne Param" })
  2. this.pageStack.pushPathByName("PageOne", "PageOne Param")
复制代码
2、带返回回调的跳转,跳转时添加onPop回调,能在页面出栈时获取返回信息,并进行处理。
  1. this.pageStack.pushPathByName('PageOne', "PageOne Param", (popInfo) => {
  2. console.log('Pop page name is: ' + popInfo.info.name + ', result: ' + JSON.stringify(popInfo.result))
  3. });
复制代码
3、带错误码的跳转,跳转结束会触发异步回调,返回错误码信息。
  1. this.pageStack.pushDestinationByName('PageOne', "PageOne Param")
  2. .catch((error: BusinessError) => {
  3.     console.error(`Push destination failed, error code = ${error.code}, error.message = ${error.message}.`);
  4. }).then(() => {
  5. console.error('Push destination succeed.');
  6. });
复制代码
页面返回

  1. // 返回到上一页
  2. this.pageStack.pop()
  3. // 返回到上一个PageOne页面
  4. this.pageStack.popToName("PageOne")
  5. // 返回到索引为1的页面
  6. this.pageStack.popToIndex(1)
  7. // 返回到根首页(清除栈中所有页面)
  8. this.pageStack.clear()
复制代码
页面替换

  1. // 将栈顶页面替换为PageOne
  2. this.pageStack.replacePath({ name: "PageOne", param: "PageOne Param" })
  3. this.pageStack.replacePathByName("PageOne", "PageOne Param")
复制代码
页面删除

  1. // 删除栈中name为PageOne的所有页面
  2. this.pageStack.removeByName("PageOne")
  3. // 删除指定索引的页面
  4. this.pageStack.removeByIndexes([1,3,5])
复制代码
参数获取

  1. // 获取栈中所有页面name集合
  2. this.pageStack.getAllPathName()
  3. // 获取索引为1的页面参数
  4. this.pageStack.getParamByIndex(1)
  5. // 获取PageOne页面的参数
  6. this.pageStack.getParamByName("PageOne")
  7. // 获取PageOne页面的索引集合
  8. this.pageStack.getIndexByName("PageOne")
复制代码
路由拦截

   NavPathStack提供了setInterception方法,用于设置Navigation页面跳转拦截回调。该方法必要传入一个NavigationInterception对象,该对象包含三个回调函数:
   名称
  形貌
  willShow
  页面跳转前回调,答应操纵栈,在当前跳转生效。
  didShow
  页面跳转后回调,在该回调中操纵栈会在下一次跳转生效。
  modeChange
  Navigation单双栏显示状态发生变动时触发该回调。
  例子
  1. this.pageStack.setInterception({
  2.   willShow: (from: NavDestinationContext | "navBar", to: NavDestinationContext | "navBar",
  3.     operation: NavigationOperation, animated: boolean) => {
  4.     if (typeof to === "string") {
  5.       console.log("target page is navigation home page.");
  6.       return;
  7.     }
  8.     // 将跳转到PageTwo的路由重定向到PageOne
  9.     let target: NavDestinationContext = to as NavDestinationContext;
  10.     if (target.pathInfo.name === 'PageTwo') {
  11.       target.pathStack.pop();
  12.       target.pathStack.pushPathByName('PageOne', null);
  13.     }
  14.   }
  15. })
复制代码





免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

种地

金牌会员
这个人很懒什么都没写!

标签云

快速回复 返回顶部 返回列表