【鸿蒙实战开发】如何在Navigation中完成路由拦截

介绍

本示例介绍在Navigation中如何完成路由拦截:首次登录时记录登录状态,再次登录时可以直接访问主页无需重复登录,当退出登录时,下次需重新登录。

使用说明

1.点击Navigation路由拦截案例。
2.在弹出的半模态页面中勾选"阅读并同意协议",点击按钮"手机号码一键登录"。
3.进入主页,点击返回上级页面按钮,重新点击Navigation路由拦截案例,即可直接进入主页,不需要重复登录。
4.点击主页的"退出登录"按钮,退出案例,此时点击Navigation路由拦截案例需重新登录。

实现思路

1、在路由模块增加路由拦截器interceptor.ets,定义拦截容器、注册方法和公共拦截逻辑,interceptor.ets

/**
 * 定义拦截实现接口
 *
 * @param routerInfo 需要拦截的路由名
 * @param param 路由参数
 */
export interface InterceptorExecute {
executeFunction(router: RouterInfo, param?: string): boolean;
}

/**
 * 定义拦截器方法
 */
export class Interceptor {
  // 定义拦截器容器
  private static list: Array<InterceptorExecute> = [];

  /**
   * 注册拦截页面
   *
   * @param interceptorFnc 子模块传过来的自定义拦截函数
   */
  public static registerInterceptorPage(interceptorFnc: InterceptorExecute): void {
    Interceptor.list.push(interceptorFnc);
  }

  /**
   * 公共拦截器逻辑
   *
   * @param routerInfo 接收传过来的路由名
   * @param param 路由参数
   */
  public static interceptor(routerInfo: RouterInfo, param?: string): boolean {
    // 循环拦截器容器中所有的子模块自定义的拦截函数
    for (let i = 0; i < Interceptor.list.length; i++) {
      if (Interceptor.list[i].executeFunction(routerInfo, param))
        return true; // 如果子模块拦截函数返回true,即需要拦截
    }
    // 否则就放行
    return false;
  }
}

2、当点击本案例时,触发在路由模块的动态路由.push()中的interceptor的公共拦截方法(此处需动态路由完成加载后执行否则首次路由拦截失败),DynamcicsRouter.ets

// 通过获取页面栈跳转到指定页面
public static async push(routerInfo: RouterInfo, param?: string): Promise<void> {
  if (isImportSucceed) {
    // TODO:知识点:在路由模块的动态路由.push()中调用拦截方法,此处必须等待动态路由加载完成后再进行拦截,否则页面加载不成功,导致无法注册拦截的函数,出现首次点进案例时拦截失效。
    if (Interceptor.interceptor(routerInfo, param)) {
      return;
    }
}

3、子模块中定义业务具体拦截逻辑,做具体的拦截实现:通过routerInfo判断目的地为"我的页面"时判断登录状态是"未登录",此时执行跳转到登录页并返回true给拦截容器list(告知需拦截),已登录返回false,放行。并且注册到拦截器容器list中,interceptorPage.ets。

// 子模块实现拦截接口,做具体的拦截实现
export class MyPageInterceptorExecute implements InterceptorExecute {
  executeFunction(routerInfo: RouterInfo, param?: string): boolean {
    // 通过routerInfo判断目的地为"我的页面"时判断登录状态是"未登录",此时执行跳转到登录页并返回true给拦截容器list(告知需拦截),已登录返回false,放行。
    if (routerInfo !== undefined && routerInfo.pageName === RouterInfo.NAVIGATION_INTERCEPTOR.pageName) {
      // 如果未登录
      if (!AppStorage.get("login")) {
        // 跳转登录页
        DynamicsRouter.push(RouterInfo.MULTI_MODAL_TRANSITION, param);
        return true; // true:路由拦截
      } else {
        return false; // false:否则放行
      }
    }
    // 通过routerInfo判断目的地为"登录页面"时放行。
    if (routerInfo !== undefined && routerInfo.pageName === RouterInfo.MULTI_MODAL_TRANSITION.pageName) {
      return false;
    }
    return false; // false,路由放行
  }
}

// 拦截器注册拦截函数
Interceptor.registerInterceptorPage(new MyPageInterceptorExecute());

4、拦截器获取拦截容器list中所有注册过的子模块的拦截函数,如果子模块拦截函数返回true,即需要拦截,否则放行。

/**
 * 公共拦截器逻辑
 *
 * @param routerInfo 接收传过来的路由名
 * @param param 路由参数
 */
public static interceptor(routerInfo: RouterInfo, param?: string): boolean {
  // 循环拦截器容器中所有的子模块自定义的拦截函数
  for (let i = 0; i < Interceptor.list.length; i++) {
    if (Interceptor.list[i].executeFunction(routerInfo, param))
      return true; // 如果子模块拦截函数返回true,即需要拦截
  }
  // 否则就放行
  return false;
}

5、通过循环拦截容器list得到返回true时通知动态路由不再继续跳转, 否则返回false,通知动态路由继续执行跳转,跳转到我的页面,DynamcicsRouter.ets。

// 通过获取页面栈跳转到指定页面
public static async push(routerInfo: RouterInfo, param?: string): Promise<void> {
  if (isImportSucceed) {
    // 当返回true时执行拦截,通知动态路由不再继续跳转
    if (Interceptor.interceptor(routerInfo, param)) {
      return;
    }
    ... // 当返回false,通知动态路由继续执行跳转,跳转到我的页面
  }

6、在登录页点击:本机号码一键登录后,登陆成功,登陆状态置为true,且跳转到主页,HalfModalWindow

Button($r('app.string.multimodaltransion_phone_start_login'))
  .onClick(() => {
    if (this.isConfirmed) {
      AppStorage.set("login", true); // 登录状态置为已登录
      DynamicsRouter.pop();
      DynamicsRouter.push(RouterInfo.NAVIGATION_INTERCEPTOR); // 路由跳转
    })

7、详情页中点击:注销登录,登录状态置为false,退出登录,interceptorPage.ets

@StorageLink('login') hasLogin: boolean = true;
        
Button($r('app.string.naviagtion_interceptor_loginout'))
  .onClick(() => {
    this.hasLogin = false; // 注销登录
    DynamicsRouter.pop(); // 退出登录
  })
  .width("100%")

高性能知识点

不涉及

工程结构&模块类型

navigationinterceptor                           // har类型
|---src/main/ets/view
|   |---interceptorPage.ets                     // 视图层-主页
|---src/main/ets/model
|   |---DataModel.ets                           // 模型层-数据模块
routermodule                                    // har类型
|---src/main/ets/interceptpr
|   |---interceptor.ets                         // 模型层-拦截器方法类
|---src/main/ets/router
|   |---DynamticRouter.ets                      // 模型层-动态路由方法类

模块依赖

har包-common库中UX标准

@ohos/routermodule(动态路由)

写在最后

●如果你觉得这篇内容对你还蛮有帮助,我想邀请你帮我两个小忙:
●点赞,转发,有你们的 『点赞和评论』,才是我创造的动力。
●关注小编,同时可以期待后续文章ing ,不定期分享原创知识。

在这里插入图片描述

### 鸿蒙系统中 Navigation 路由拦截机制及其实现方法 鸿蒙系统的 `Navigation` 组件提供了强大的路由管理能力,其中包括路由拦截功能。这种功能允许开发者在用户导航至目标页面之前执行特定逻辑,例如验证用户的登录状态、权限检查或者显示提示框等。 #### 1. **核心概念** 路由拦截的核心在于通过 `setInterception` 或者自定义拦截器来实现对路由请求的控制。当用户尝试访问某个页面时,拦截器会先被触发,从而决定是否继续跳转或转向其他页面[^2]。 #### 2. **实现流程** ##### (1)创建拦截拦截器可以通过继承 `InterceptorExecute` 接口并重写其 `executeFunction` 方法来定制化拦截逻辑。以下是拦截器的一个典型实现: ```typescript // 定义拦截器类 export class MyPageInterceptorExecute implements InterceptorExecute { executeFunction(routerInfo: RouterInfo, param?: string): boolean { // 判断当前要跳转的目标页面名称 if (routerInfo !== undefined && routerInfo.pageName === "MyPage") { // 检查用户是否已登录 const isLoggedIn = AppStorage.get("isLoggedIn"); if (!isLoggedIn) { // 用户未登录,则跳转到登录页面 DynamticRouter.push("LoginPage", param); return true; // 返回true表示拦截成功,阻止原定跳转 } } // 对于其他页面或者已经登录的情况,放行 return false; } } ``` 此代码片段展示了如何根据目标页面名以及用户的登录状态来决定是否拦截跳转操作[^4]。 ##### (2)注册拦截完成拦截器编写后,需要将其注册到全局拦截器列表中以便生效: ```typescript // 注册拦截器 Interceptor.registerInterceptorPage(new MyPageInterceptorExecute()); ``` 这一步骤确保每次路由切换都会经过该拦截器的检测[^4]。 ##### (3)配置动态路由 如果项目中有多个可能需要拦截的路径,建议统一维护一份路由映射表,并结合动态路由技术灵活调整行为模式。例如,在某些特殊情况下临时启用或禁用某项拦截规则。 ```typescript const RouterInfo = { NAVIGATION_INTERCEPTOR: { pageName: 'MyPage' }, MULTI_MODAL_TRANSITION: { pageName: 'LoginPage' } }; ``` 以上代码用于定义不同页面对应的标识符,便于后续逻辑中的匹配过程更加清晰明了[^4]。 #### 3. **实际应用场景举例** - **登录校验**:如前所述,可以在用户试图打开受保护资源(如个人中心)前确认身份合法性。 - **权限审核**:针对部分高级功能模块实施细粒度授权策略,防止越权访问。 - **引导页展示**:首次启动应用时自动呈现新手指引界面而非默认首页。 #### 4. **注意事项** 为了提升用户体验,应尽量减少不必要的中断动作;同时也要注意性能开销问题——复杂的异步任务可能会延迟响应速度甚至引发卡顿现象。因此设计合理的缓存方案显得尤为重要[^3]。 --- ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值