Vue后台管理系统开发,相关代码的笔记。
- Vue学习
- 2022-07-22
- 1142热度
- 0评论
开发记录
从零开始开发后台管理系统,还是有很多值得记录的地方。构建工具Vite、使用Vue3。
1.批量导入指定目录的组件
/*
* @author 友人a丶
* @date 2022-07-11
* @app Vue应用对象
* */
export default function (app) {
/*
* 指定要导入的文件目录
* 直接加载用{eager:true},懒加载用glob
* */
const modules = import.meta.glob(['@/layouts/*/index.js', '@/components/*/index.js']);
for (let i in modules) {
let name = /(.*)?\/(.*)\/index.js/.exec(i);
/*直接引入组件*/
app.component(name[2], modules[i].default);
/*异步组件*/
app.component(name[2], defineAsyncComponent(modules[i]));
}
}
直接加载的时候modules就是元素为组件对象的数组,懒加载的时候是元素为import方法的函数数组 。
2.应用初始化
/*
* @author 友人a丶
* @date 2022-07-11
*
* 引导系统初始化
* 初始化全局响应拦截器
* 初始化路由守卫
* 初始化用户登录
* */
import loadGuard from "./loadGuard"
import loadInterceptor from "./loadInterceptor"
import watchRem from "./watchRem";
export default async function (){
/*
* 自动调整rem
* */
watchRem();
/*
* 加载拦截器
* */
loadInterceptor();
/*
* 加载路由守卫
* */
loadGuard();
}
3.自动调整rem
/*
* @author 友人a丶
* @date 2022-07-11
* 自动调整rem的大小
* */
export default function () {
let l = () => {
let r = document.documentElement, o = r.offsetWidth / 100;
o < 16 && (o = 16), r.style.fontSize = o + "px", window.rem = o
};
l();
window.addEventListener("resize",()=>l());
}
4.导航守卫
/*
* @author 友人a丶
* @date 2022-07-11
*
* 加载全局路由守卫
* */
import {router} from "@/router";
import userStore from "@/stores/user";
import load from "@/common/load";
import systems from "@/stores/system";
import NProgress from 'nprogress'
import loadUser from "@/service/loadUser";
/*进度条*/
NProgress.configure({showSpinner: false})
// 不需要拦截的路由配置
const ignoreRoute = {
names: ['404', '403'], //根据路由名称匹配
paths: ['/login'], //根据路由fullPath匹配
/**
* 判断路由是否包含在该配置中
* @param route vue-router 的 route 对象
* @returns {boolean}
*/
includes(route) {
return ignoreRoute.names.includes(route.name) || ignoreRoute.paths.includes(route.path)
}
}
/*
* 加载路由守卫
* */
export default function () {
console.log("加载路由守卫...");
let user = userStore();//全局状态
/*
* 加载进度条
* */
router.beforeEach((to, from, next) => {
// start progress bar
if (!NProgress.isStarted()) {
NProgress.start()
}
next()
});
/*
* 判断系统是否初始化
* */
router.beforeEach(async (to, from, next) => {
if(!systems().loaded){
await loadUser();//加载用户信息初始化系统
}
next(); //下一个
});
/*
* 已登录时,访问登录页面,让它走
* */
router.beforeEach( (to, from, next) => {
if (user.role != 0 && (to.path == "/login")) {
next('/');
}else{
next(); //下一个
}
});
/*
* 判断是否登录
* */
router.beforeEach((to, from, next) => {
/*
* 判断是否需要拦截
* */
console.log("登录判断守卫激活....")
if (ignoreRoute.includes(to)) {
next();
} else {
/*
* 角色为0,代表未登录
* */
if (user.role == 0) {
next({path: '/login'});
} else {
next();
}
}
})
/*
* 判断用户权限
* */
router.beforeEach((to, from, next) => {
console.log("权限判断守卫激活....")
/*
* 判断是否需要拦截
* */
if (ignoreRoute.includes(to)) {
next();
} else {
/*
* 判断用户权限
* */
if (user.role < to.meta.role) {
load.error("您无权限访问该页面....");
next({path: '/403'});
} else {
next();
}
}
})
/*
* 切换页面标题
* */
router.beforeEach((to, from, next) => {
document.title = to.name
next();
})
/*
* 结束进度条
* */
router.afterEach(() => {
// finish progress bar
NProgress.done()
});
}
5.axios拦截器
/*
* @author 友人a丶
* @date 2022-07-11
*
* 加载axio拦截器
* */
import axios from "axios";
import load from "@/common/load";
import {router} from "@/router";
import apis from '@/service/api';
// 不需要拦截的接口
const ignoreApi = {
api: [
apis.login
],
includes(api) {
/*
* 判断当前请求的接口是否在忽略的列表
* */
for(let item of ignoreApi.api){
let reg=new RegExp(`.*${item}.*`);
if(reg.test(api)){
return true;
}
}
return false;
}
}
/*
* 注册响应拦截器
* */
export default function (){
console.log("加载拦截器...");
axios.interceptors.response.use(function (res){
console.log("请求接口:"+res.config.url)
console.log(res);
/*
* 判断是否需要拦截
* */
if(ignoreApi.includes(res.config.url)){
return res;
}
/*
* 判断用户登录是否失效
* */
if(res.code == -1){
load.confirm("当前登录状态已失效,请您重新登录!",()=>{
router.replace('/login')
})
}
return res;
});
}
6.获取某个路由的子路由(用于生成菜单)
/*
* 操作路由的相关方法
* */
import {router} from "@/router/index";
/*
* 获取某个路由项的子项
* */
export function getChildren(path) {
let routes=router.getRoutes();
for (let i of routes) {
if (i.path == path) {
return i.children;
}
}
}
7.简单的弹出封装(antd design vue)
import {
message,
Modal
} from "ant-design-vue";
let hide = [];
export default {
loading(text = '加载中...') {
hide.push(message.loading(text, 0))
},
loaded() {
if (hide.length > 0) {
let timer=setTimeout(()=>{
hide[hide.length - 1]()
hide.splice(hide.length - 1, 1)
},500);
}
},
error(text = '加载异常') {
message.error(text);
},
success(text = 'ok!') {
message.success(text);
},
confirm(text, callback = null) {
Modal.confirm({
title: '提示',
centered: true,
content: text,
maskClosable: false,
onOk: (close) => {
close(); //关闭
if (callback) {
callback()
}
}
})
}
}
8.退出登录
/*
* @author 友人a丶
* @date 2022-07-11
* 用户退出登录
* */
import user from "@/stores/user";
import Cookies from "js-cookie";
import {router} from "@/router";
import load from "@/common/load";
export default function () {
load.confirm("确认退出登录吗?",()=>{
user().$reset(); //重置用户数据的状态管理器
/*
* 清空cookie
* */
Object.keys(Cookies.get()).forEach((item)=>{
Cookies.remove(item);
})
/*
* 跳转登录界面
* */
router.replace('/login');
})
}
9.获取需要缓存的组件列表
/*
* @author 友人a丶
* @date 2022-07-11
* name代表组件名
* 获取需要缓存的组件
* */
import routes from '@/router'
export function getChached(path='') {
let routes=router.getRoutes();
let cahced=[]; //是否开启缓存
/*
* 为空代表获取所有一级组件
* */
if(path == ""){
routes.forEach((item)=>{
if(item.meta.cache){
cahced.push(item.meta.cache);
}
});
}else{
/*遍历目标子组件*/
for (let i of routes) {
if (i.path == path) {
i.children.forEach((item)=>{
if(item.meta.cache){
cahced.push(item.meta.cache);
}
});
}
}
}
return cahced;
}
需要考虑
1.如何让显示的菜单响应路由的变化(跳转到某个页面,自动选中某个菜单)?
本身菜单被点击了,自己会变化被选中的状态,需要考虑的是从其他页面跳转过来的时候,如何正常匹配显示被选的菜单;
路由包括静态的路由和有变化的参数路由,某些情况下还会具有参数。
- router.matched,与给定路由地址匹配的标准化的路由记录数组。
- 正则匹配,搭配计算属性;假设业务场景:【顶部是一级菜单,用于打开一个新页面,每个页面都有自身的菜单(二级菜单),菜单下面加包括子菜单】,首先就需要根据上方一级菜单的变化匹配二级菜单,还需要根据当前路由判断哪个子菜单被选中;所有需要条件有:代表当前路由的响应式变量、代表当前一级路由的子路由的响应式变量、代表被选中的菜单的响应式变量,最终如下:
let selectedKeys = computed({ get() { let current = route.fullPath; for (let i = 0; i < items.length; i++) { //断言右边是空或者?或者/ // 完整匹配或者带参数匹配 let regexp=new RegExp(`(?:.*${items[i].path}$)|(?:.*${items[i].path}[\?\/].*)`); if(regexp.test(current)){ return [i]; } } }, set(value) { return; } } );
考虑到参数路由和带有?相关参数的路由,所以正则匹配的是
当前路由与对应的路由完全相等
以及部分相等的同时右边为/或者?
提示由此还需考虑父路由存在相似的路由片段时,匹配的优先级的问题
2.如何组织目录?
- 代表页面的组件一般以文件夹的形式通过index.js导出组件,方便观察层次结构,并且页面组件一般都会拆分JS模块,通过文件夹也更加方便文件的分类,保持目录的简洁。
- 其他的组件,如果设计到大量的逻辑,需要拆分JS模块,可以用文件夹,如何很简单的直接用.vue文件即可。
- 如何让父子组件的层级更加清晰?首先名字可以按层级写;parent-children.vue。
- 名字较长的组件用“-”分割,更加友好。
3.结构型的组件划分?
- 将布局看组架子(布局组件)、视图看做需要的内容(视图组件),布局承载内容;
- 通过全局状态的设置来动态调整布局组件的显示和隐藏。
4.如何组织无限层级的子路由作为菜单?
模板方式实现起来非常的麻烦,JSX的方式更加适合这种需求;
- 首先需要根据当前路由获取一个可以作为祖先的父级路由对象
5.运行中的router
getRoutes();
获取一个包括所有路由项的数组;不同层次的路由path属性都是从根节点开始的;路由的children属性则是内的子路由是相对路径
不管是push、redirect、route-link,都可以进行相对路径(dynamic)或者绝对路径(/dynamic)跳转;
[
{
"path": "/spread/tencent",
"name": "腾讯广告",
"meta": {
"role": 0,
"icon": "icon-guangdiantong"
},
"props": {
"default": false
},
"children": [],
"instances": {},
"leaveGuards": {},
"updateGuards": {},
"enterCallbacks": {},
"components": {}
}
]