uni-app 自定义路由封装模块详解(附源码逐行解读)

🚀uni-app 自定义路由封装模块详解(附源码逐行解读)

📌 请收藏 + 点赞 + 关注,获取更多 uni-app 项目实用技巧!

在实际 uni-app 项目中,我们常常需要对 uni.navigateTouni.switchTab 等 API 做一层封装,以便统一处理页面跳转、参数传递、登录拦截等逻辑。本篇将完整展示一份功能强大的路由封装方案,并逐行解释其实现逻辑,帮助你构建更可控、易扩展的项目架构。


📦源码展示(含说明性注释)

👇以下是完整源码,已集成:

  • 页面路径分析
  • query 和 params 分离
  • 登录拦截(导航守卫)
  • 页面跳转封装(支持多种跳转模式)
  • 首页识别与返回逻辑
  • 页签识别
  • 页面栈方法调用
  • 登录回调钩子

📄 完整代码如下(点击右侧小箭头可展开逐行解读):

点击展开完整源码逐行解读
// 引入 lodash 的 last 方法,返回数组最后一个元素
import { last } from "lodash-es";// 引入 ctx 插件中编译期注入的页面配置、tabBar、subPackages 等
import { ctx } from "virtual:ctx";// 项目中封装的 localStorage 工具
import { storage } from "../utils";// 引入全局 config 配置
import { config } from "../../config";type PushOptions = string | {path: string; // 路径mode?: "navigateTo" | "redirectTo" | "reLaunch" | "switchTab" | "preloadPage"; // 跳转方式events?: { [key: string]: (data: any) => void }; // 页面间事件通信query?: { [key: string]: any }; // URL 参数params?: { [key: string]: any }; // 缓存参数isGuard?: boolean; // 是否启用导航守卫[key: string]: any;
};type Tabs = {text?: string;pagePath: string;iconPath?: string;selectedIconPath?: string;[key: string]: any;
}[];// 获取所有页面配置
const routes = [...ctx.pages];// 处理子包中的页面路径
if (ctx.subPackages) {ctx.subPackages.forEach((a) => {a.pages.forEach((b) => {routes.push({...b,path: a.root + "/" + b.path,});});});
}// 注册钩子函数
const fn: { [key: string]: (...args: any[]) => any } = {};// 路由核心对象
const router = {// 读取 tabBar 配置get tabs(): Tabs {if (ctx.tabBar) {return ctx.tabBar.list || [];} else {return [];}},// 全局样式globalStyle: ctx.globalStyle,// 所有路由routes,// 当前页面 URL query 参数get query() {const info = this.info();return { ...info?.query };},// 非 URL 参数,通过缓存传递get params() {return storage.get("router-params") || {};},// 页面路径配置get pages() {return {home: "/" + (ctx.tabBar ? this.tabs[0].pagePath : ctx.pages[0].path),...config.app.pages,};},// 当前页面信息对象currentPage() {return last(getCurrentPages())!;},// 当前路径get path() {return router.info()?.path;},// 当前页面完整信息info() {const page = last(getCurrentPages());if (page) {const { route, $page, $vm, $getAppWebview }: any = page;const q: any = {};// 解析 query 参数try {$page?.fullPath.split("?")[1].split("&").forEach((e: string) => {const [k, v] = e.split("=");q[k] = decodeURIComponent(v);});} catch (e) {}const style = this.routes.find((e) => e.path == route)?.style;return {$vm,$getAppWebview,path: `/${route}`,fullPath: $page?.fullPath,query: q || {},isTab: this.isTab(route),style,isCustomNavbar: style?.navigationStyle == "custom",};}return null;},// 页面跳转主函数push(options: PushOptions) {if (typeof options === "string") {options = { path: options, mode: "navigateTo" };}let {path,mode = "navigateTo",animationType,animationDuration,events,success,fail,complete,query,params,isGuard = true,} = options;// 拼接 query 到 URLif (query) {let arr = [];for (let i in query) {if (query[i] !== undefined) arr.push(`${i}=${query[i]}`);}path += "?" + arr.join("&");}// 缓存传参if (params) {storage.set("router-params", params);}const data = {url: path,animationType,animationDuration,events,success,fail,complete,};// 如果目标是 tab 页,强制使用 switchTabif (this.isTab(path)) {mode = "switchTab";}const next = () => {switch (mode) {case "navigateTo":uni.navigateTo(data); break;case "redirectTo":uni.redirectTo(data); break;case "reLaunch":uni.reLaunch(data); break;case "switchTab":uni.switchTab(data); break;case "preloadPage":uni.preloadPage(data); break;}};// 启用导航守卫if (fn.beforeEach && isGuard) {fn.beforeEach({ path: options.path, query }, next, (opt) => this.push(opt));} else {next();}},// 返回上一页或首页back(options?: UniApp.NavigateBackOptions) {if (this.isFirstPage()) {this.home();} else {uni.navigateBack(options || {});}},// 执行当前页面某个方法callMethod(name: string, data?: any) {const { $vm } = this.info()!;if ($vm && $vm.$.exposed?.[name]) {return $vm.$.exposed[name](data);}},// 是否第一页(判断是否需要返回首页)isFirstPage() {return getCurrentPages().length == 1;},// 是否是当前路径isCurrentPage(path: string) {return this.info()?.path === path;},// 返回首页home() {this.push(this.pages.home);},// 跳转 Tab 页switchTab(name: string) {const item = this.tabs.find((e) => e.pagePath.includes(name));if (item) {this.push({path: `/${item.pagePath}`,mode: "switchTab",});} else {console.error("Not found tab", name);}},// 是否是 Tab 页isTab(path: string) {return !!this.tabs.find((e) => path === `/${e.pagePath}`);},// 跳转登录页(支持 reLaunch)login(options?: { reLaunch: boolean }) {const { reLaunch = false } = options || {};this.push({path: this.pages.login,mode: reLaunch ? "reLaunch" : "navigateTo",isGuard: false,});},// 登录成功后的回调处理nextLogin(type?: string) {const pages = getCurrentPages();const index = pages.findIndex((e) => this.pages.login.includes(e.route!));if (index <= 0) {this.home();} else {this.back({ delta: pages.length - index });}storage.set("loginType", type);if (fn.afterLogin) fn.afterLogin();uni.$emit("afterLogin", { type });},// 注册路由钩子函数(beforeEach)beforeEach(callback: (to: any, next: () => void, reject: (opt: PushOptions) => void) => void) {fn.beforeEach = callback;},// 登录后执行回调afterLogin(callback: () => void) {fn.afterLogin = callback;},
};export { router };

✍️ 核心功能说明(重点功能归纳)

功能模块描述说明
router.push()支持全模式页面跳转,封装 query/params,支持守卫
router.info()获取当前页面详细信息
router.callMethod()跨组件执行 exposed 方法
router.isTab()判断路径是否为 Tab 页
router.beforeEach()注册跳转拦截器
router.nextLogin()登录回调重定向功能
router.pages自动生成首页路径与配置路径

✅总结

该路由封装模块适用于 uni-app 项目中需要进行页面跳转逻辑统一管理的场景,具备:

  • 💡 统一跳转 API:支持 navigateTo、switchTab、reLaunch 等
  • 🔒 导航守卫机制:登录拦截与后置回调
  • 🔄 query/params 分离处理
  • 🧩 模块化配置,支持挂载 ctx

你可以在此基础上继续拓展如:权限校验、页面缓存、历史记录管理、动画过渡管理等功能。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.pswp.cn/bicheng/84541.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

QML显示图片问题解决办法

以前用qtwediget的时候&#xff0c;好像是放在qlabel或者什么组件上面&#xff0c;把图片的路径放上去就可以直接加载&#xff0c;但我用QML创建界面的时候就遇到了问题&#xff0c;哦对&#xff0c;qtwedget用qpixmap组件显示图片&#xff0c;也有image。话说回来&#xff0c;…

Vue中使用jsx

1. jsx的babel配置 1.1 在项目中使用jsx&#xff0c;需要添加对jsx的支持&#xff1a; jsx通常会通过Babel来进行转换(React编写的jsx就是通过babel转换的)Vue中&#xff0c;只需要在Babel中配置对应的插件即可以下列举需要支持转换的案例&#xff1a; template -> vue-l…

Spring Cache+Redis缓存方案 vs 传统redis缓存直接使用RedisTemplate 方案对比

结合 Spring Cache 和 Redis 的缓存方案&#xff08;即 Spring Cache Redis&#xff09;相较于普通的 Redis 缓存使用&#xff08;如直接通过 RedisTemplate 操作&#xff09;&#xff0c;具有以下显著优势&#xff1a; 具体实现方案请参考&#xff1a;Spring CacheRedis缓存…

Web应用安全漏洞扫描:原理、常用方法及潜在风险解析?

Web应用安全的关键环节在于进行漏洞扫描&#xff0c;这种扫描通过自动化或半自动化的方式&#xff0c;对应用进行安全测试。它能揭示出配置错误、代码缺陷等众多安全风险。接下来&#xff0c;我将详细阐述这些情况。 扫描原理 它主要模拟攻击者的行为&#xff0c;以探测和攻击…

Spring中@Value注解:原理、加载顺序与实战指南

文章目录 前言一、Value注解的核心原理1.1 容器启动阶段&#xff1a;环境准备1.2 Bean实例化阶段&#xff1a;后置处理器介入1.3 值解析阶段&#xff1a;双引擎处理1. 占位符解析&#xff08;${...}&#xff09;2. SpEL表达式解析&#xff08;#{...}&#xff09; 1.4 类型转换与…

MySQL 8配置文件详解

MySQL 8 配置文件详解 MySQL 8 的配置文件(my.cnf或my.ini)是MySQL服务器启动时读取的主要配置文件&#xff0c;它包含了服务器运行所需的各种参数设置。以下是MySQL 8配置文件的详细解析&#xff1a; 配置文件位置 MySQL 8 会按照以下顺序查找配置文件&#xff1a; /etc/m…

台湾住宅IP哪家好,怎么找到靠谱的海外住宅IP代理商

探索台湾住宅IP&#xff1a;如何找到靠谱的海外住宅IP代理商&#xff1f; 在当今数字化时代&#xff0c;海外住宅IP的需求日益增长&#xff0c;尤其在跨境电商、网络营销、数据抓取等领域。对于需要台湾住宅IP的用户来说&#xff0c;找到一家靠谱的海外住宅IP代理商至关重要。本…

读研一些毕业感想

回首过往三年&#xff0c;从踌躇迷茫到明晰坚定&#xff0c;从稚嫩懵懂到明理成熟&#xff0c;一切只觉轻舟已过万重山。 依稀记得我拉着行李箱跋山涉水来到学校的那天&#xff0c;早上从广东中山乘坐10小时高铁到北京西&#xff0c;然后坐1一个多小时地铁到学校&#x…

《飞算JavaAI:稳定、高效、跨平台的AI编程工具优势解析》

随着人工智能技术的不断发展&#xff0c;AI编程工具越来越成为开发者们在研究和应用AI模型时不可或缺的利器。国内外的AI编程工具多种多样&#xff0c;涵盖了从基础编程语言、框架到图形化界面的多种选择。然而&#xff0c;在这些工具中&#xff0c;飞算JavaAI作为一种基于Java…

day27/60重写(补充)

DAY 27 函数专题2&#xff1a;装饰器 ps&#xff1a;第一期day27对应5月16日 知识点回顾&#xff1a; 装饰器的思想&#xff1a;进一步复用函数的装饰器写法注意内部函数的返回值 作业&#xff1a; 编写一个装饰器 logger&#xff0c;在函数执行前后打印日志信息&#xff08;如…

网传西门子12亿美元收购云原生工业软件,云化PLM系统转机在协同

近日&#xff0c;网传西门子将以12亿美元全现金交易收购云原生MES公司FlexFact&#xff0c;并整合其技术至Xcelerator工业软件平台。如果此次收购动作完成&#xff0c;将会成为西门子加速工业云转型的标志性动作&#xff0c;背后的意义也极为深远&#xff0c;不仅会直接响应竞争…

大模型笔记_检索增强生成(RAG)

1. RAG的概念 RAG&#xff08;Retrieval-Augmented Generation&#xff09; 是一种结合 信息检索&#xff08;Retrieval&#xff09;与文本生成&#xff08;Generation&#xff09;的模型架构&#xff0c;旨在通过动态引入外部知识库或实时数据&#xff0c;提升大语言模型&…

Spring Security是如何完成身份认证的?

1. 用户名和密码被过滤器获取到&#xff0c;封装成 Authentication ,通常情况下是 UsernamePasswordAuthenticationToken 这个实现类。 2. AuthenticationManager 身份管理器负责验证这个 Authentication 3. 认证成功后&#xff0c; AuthenticationManager 身份管理器返回一…

Python爬虫实战:研究xmltodict库相关技术

1. 引言 1.1 研究背景与意义 气象数据是环境研究、农业生产、城市规划等领域的重要基础。随着互联网技术的发展,越来越多的气象数据以 XML 格式在网络上公开。XML(可扩展标记语言)因其结构化和自描述性的特点,成为数据交换的标准格式之一。然而,这些数据通常分散在不同的…

中小企业无线局域网络搭建与优化指南

1. 引言&#xff1a;无线网络——驱动中国中小企业数字化转型的引擎 无线网络已成为现代企业运营的基础设施&#xff0c;直接影响员工工作效率和客户体验。随着Wi-Fi7技术的成熟和普及&#xff0c;中小企业网络建设正迎来全新机遇。在数字经济浪潮席卷全球的今天&#xff0c;无…

【已解决】python的kafka-python包连接kafka报认证失败

先说原因&#xff1a;安装python包的时候&#xff0c;多装了一个kafka的包&#xff1a;kafka 1.3.5 我把py文件打包成二进制文件&#xff0c;在linux上执行就一直报认证失败&#xff0c;后来确认登录信息、认证方式没有问题&#xff0c;把这个kafka包卸载…

传输层协议TCP(下)

上一篇https://blog.csdn.net/Small_entreprene/article/details/148193741?sharetypeblogdetail&sharerId148193741&sharereferPC&sharesourceSmall_entreprene&sharefrommp_from_link 接下来&#xff0c;我们来谈论TCP具体的机制&#xff01; 具体TCP机制 …

洛谷B3612 【深进1.例1】求区间和

题目描述 给定 n 个正整数组成的数列 a1​,a2​,⋯,an​ 和 m 个区间 [li​,ri​]&#xff0c;分别求这 m 个区间的区间和。 输入格式 第一行&#xff0c;为一个正整数 n 。 第二行&#xff0c;为 n 个正整数 a1​,a2​,⋯,an​ 第三行&#xff0c;为一个正整数 m 。 接下…

debian12 修改MariaDB数据库存储位置报错

debian12 修改MariaDB数据库存储位置到home报错 MariaDB 修改存储路径后启动失败问题解决 更改数据存储位置 如果需要将数据存储到其他位置&#xff08;如更大的分区&#xff09;&#xff1a; 停止 MariaDB 服务&#xff1a; bash sudo systemctl stop mariadb 创建新目录并设…

【评测】flux-dev文生图模型初体验

回到目录 【评测】flux-dev文生图模型初体验 1. 安装基础环境 参考 modelscope的Flux.1-dev页面 2. 使用tongyi写提示词 帮我用英文写3个&#xff0c;文生图片1024*1024的提示词&#xff0c;准备用flux.dev生成用 [pic03] 3. 运行代码 4090D满载运行&#xff0c; 1min左…