Vite+vue3的约定式路由实践

  • Vite+vue3的约定式路由实践;
  • 实现了类似umi的约定式路由,并实现了全局layouts的和404的路由;
  • 本文大量借鉴了下面链接的文章,侵权删

路由规则

生成的方式,我们尽量与 umi.js 保持一致, 并实现(umijs)的约定式 layouts404路由。但避免一个问题:避免将不需要的组件映射成路由

  • pages/index.tsx/
  • pages/blog/index.tsx/blog
  • pages/about.tsx/about

umi中,我们可以在config.ts中这样配置来避免将不需要的组件映射成路由

javascript 复制代码
conventionRoutes: {
    // 规定只有index文件会被识别成路由
    exclude: [
      /(?<!(index|\[index\]|404)(\.(js|jsx|ts|tsx)))$/,
      /model\.(j|t)sx?$/,
      /\.test\.(j|t)sx?$/,
      /service\.(j|t)sx?$/,
      /models\//,
      /components\//,
      /services\//,
    ],
  },

在这里,我们暂时使用这个正则,也可以根据项目中是否使用了jsx来增加.vue文件后缀

ini 复制代码
const modules = import.meta.glob("/src/pages/**/{index,404}.{ts,tsx}");

工程目录示例

复制代码
├─layouts
│  │  index.tsx
├─pages
│  │
│  └─demo
│      │  index.tsx
│      │
│      └─demo-child
│              hello-world.tsx
│              index.tsx

话不多说,直接上代码

route/plugin.ts 复制代码
import { set } from "lodash-es";

import type { RouteRecordRaw } from "vue-router";
/**
 * 根据 pages 目录生成路径配置
 */
function generatePathConfig(): Record<string, any> {
  // 扫描 src/pages 下的所有具有路由文件
  const modules = import.meta.glob("/src/pages/**/{index,404}.{ts,tsx}");

  const pathConfig = {};
  Object.keys(modules).forEach((filePath) => {
    const routePath = filePath
      // 去除 src/pages 不相关的字符
      .replace("/src/pages/", "")
      // 去除文件名后缀
      .replace(/.tsx?/, "")
      // 转换动态路由 $[foo].tsx => :foo
      .replace(/\$\[([\w-]+)]/, ":$1")
      // 转换以 $ 开头的文件
      .replace(/\$([\w-]+)/, "$1")
      // 以目录分隔
      .split("/");
    // 使用 lodash.set 合并为一个对象
    set(pathConfig, routePath, modules[filePath]);
  });
  return pathConfig;
}

/**
 * 将文件路径配置映射为 vue-router 路由
 */
function mapPathConfigToRoute(cfg: Record<string, any>): RouteRecordRaw[] {
  // route 的子节点为数组
  return Object.entries(cfg).map(([routePath, child]): RouteRecordRaw => {
    let currentRoute: RouteRecordRaw = {
      path: routePath,
      name: routePath,
    };
    // () => import() 语法判断 转换为组件
    if (typeof child.index === "function" || typeof child === "function") {
      currentRoute.component = child.index || child;
    }
    const { index, ...rest } = child;
    // 还有子级为目录,查找下一层级
    if (Object.keys(rest).length > 0) {
      currentRoute.children = mapPathConfigToRoute(rest);
    }
    return currentRoute;
  });
}

// 提取公共layouts
const getLayouts = (): (() => Promise<any>) => {
  const layouts = import.meta.glob("/src/layouts/index.{ts,tsx}");
  if (!layouts) return () => Promise.resolve(undefined);
  return Object.entries(layouts)[0][1];
};

function generateRouteConfig(): RouteRecordRaw[] {
  const { ...pathConfig } = generatePathConfig();
  return [
    {
      path: "",
      name: "",
      component: getLayouts(),
      children: mapPathConfigToRoute(pathConfig),
    },
    { path: "/:pathMatch(.*)*", redirect: "/404" },
  ];
}

const routeConfig = generateRouteConfig();
export { routeConfig };

在router中使用

typescript 复制代码
import { createRouter, createWebHistory } from "vue-router";
import { routeConfig } from "./plugin";

const router = createRouter({
  history: createWebHistory(""),
  routes: routeConfig,
});

router.beforeEach(async (to: any) => {
  
});

router.afterEach(() => {
  
});

export default router;

结语

暂无

相关推荐
还是大剑师兰特6 分钟前
Vue3 按钮切换示例(启动 / 关闭互斥显示)
开发语言·javascript·vue.js
SuperEugene9 分钟前
前端代码注释规范:Vue 实战避坑,让 3 年后的自己还能看懂代码|项目规范篇
前端·javascript·vue.js
凉辰1 小时前
uniapp实现生成海报功能 (开箱即用)
javascript·vue.js·小程序·uni-app
OpenTiny社区1 小时前
TinyRobot Skills技巧大公开:让 AI 成为你的 “UI 搭建”副驾驶
前端·vue.js·ai编程
乌拉那拉丹2 小时前
vue3 配置跨域 (vite.config.ts中配置)
前端·vue.js
angerdream2 小时前
最新版vue3+TypeScript开发入门到实战教程之DOM操作
javascript·vue.js
Lee川2 小时前
Vue Router 4 核心精讲:从原理到面试实战
前端·vue.js
A923A2 小时前
【Vue3大事件 | 项目笔记】第六天
vue.js·笔记·前端框架·前端项目
学以智用2 小时前
Vue3 + Vue Router 4 完整示例(可直接运行)
前端·vue.js