From 299d805aa6311bfad4e5ed6c06ef450c47f93a8e Mon Sep 17 00:00:00 2001
From: "chen.home" <1147347984@qq.com>
Date: Sun, 19 Mar 2023 23:49:39 +0800
Subject: [PATCH] =?UTF-8?q?feat(project):=20=E5=A2=9E=E5=8A=A0=E6=9D=83?=
=?UTF-8?q?=E9=99=90=E6=8E=A7=E5=88=B6=E7=AE=A1=E7=90=86?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
README.md | 16 +++++--
index.html | 2 +-
mock/module/user.ts | 37 +++++++++++++--
public/{logo.svg => favicon.svg} | 0
src/hooks/useSystem.ts | 32 +++++++++++++
src/router/guard/dynamic.ts | 18 ++++++--
src/router/guard/permission.ts | 10 +++-
src/router/routes/index.ts | 4 +-
src/store/modules/route.ts | 16 +++++--
src/typings/business.d.ts | 4 +-
src/typings/route.d.ts | 2 +-
src/utils/is.ts | 56 +++++++++++------------
src/views/permission/justSuper/index.vue | 13 ++++++
src/views/permission/permission/index.vue | 14 ++++++
vite.config.ts | 2 +-
15 files changed, 174 insertions(+), 52 deletions(-)
rename public/{logo.svg => favicon.svg} (100%)
create mode 100644 src/views/permission/justSuper/index.vue
create mode 100644 src/views/permission/permission/index.vue
diff --git a/README.md b/README.md
index 178a8ff..7f0f71f 100644
--- a/README.md
+++ b/README.md
@@ -11,15 +11,21 @@
## 🌈 介绍
-一个基于Vue3、Vite3、Typescript、pinia、Naive UI、Vue-Router的后台管理免费开源模板,助力提高开发效率,让大家早点下班做自己的事情
+[Ench-admin](https://github.com/chen-see/ench-admin)一个基于Vue3、Vite3、Typescript、pinia、Naive UI、Vue-Router的后台管理免费开源模板,助力提高开发效率,让大家早点下班做自己的事情
## 😎 线上预览地址
- [Ench-Admin 预览地址](https://ench-admin.vercel.app/)
-## 💾 代码仓库
+## ⚡特性
-- [github](https://github.com/chen-see/ench-admin)
+- **最新流行技术栈** - 基于Vue3、Vite、TypeScript、NaiveUI、Pinia等最新技术栈开发
+- **网络请求功能封装** - 完善的axios封装和配置,统一的响应处理和多场景能力
+- **权限控制** - 完善的前后端权限管理方案
+- **路由系统** - 支持本地静态路由和后台返回路由两种获取模式
+- **组件封装** - 对日常使用频率较高的组件二次封装,满足基础工作需求
+- **主题配置** - 黑暗主题适配
+- **代码规范** - 完整支持的代码风格规范和代码提交规范
## 🚧 安装使用
@@ -51,7 +57,7 @@ pnpm commit
| last 2 versions | last 2 versions | last 2 versions | last 2 versions |
## 🙌 学习交流
-
+Ench-Admin 是完全开源免费的项目,旨在帮助开发者更方便地进行中大型管理系统开发,有使用问题欢迎在QQ交流群内提问。
## 🧩贡献
如果您发现了任何问题或有改进建议,请创建一个issue或提交一个PR。我们欢迎您的贡献!
@@ -60,6 +66,6 @@ pnpm commit
如果感觉本项目对你工作或学习有帮助,请帮我点一个✨Star,这将是对我极大的鼓励与支持。
-## 🧾许可证
+## 🧾License
该项目采用MIT许可证,详见[LICENSE](LICENSE)文件。
diff --git a/index.html b/index.html
index dcaef9e..1a5f1c0 100644
--- a/index.html
+++ b/index.html
@@ -2,7 +2,7 @@
-
+
<%= title %>
diff --git a/mock/module/user.ts b/mock/module/user.ts
index 6b20150..8de0344 100644
--- a/mock/module/user.ts
+++ b/mock/module/user.ts
@@ -7,11 +7,10 @@ const token = () => Random.string('upper', 32, 32);
const userInfo = {
userId: '1',
- userName: 'admin',
+ userName: 'iamsee',
realName: '管理员大人',
avatar: 'https://z3.ax1x.com/2021/10/29/5jnWgf.jpg',
- role: 'admin',
- password: '123456',
+ role: "user",
};
const userRoutes = [
{
@@ -284,6 +283,37 @@ const userRoutes = [
},
],
},
+ {
+ name: 'permission',
+ path: '/permission',
+ redirect: '/permission/permission',
+ meta: {
+ title: '权限示例',
+ requiresAuth: true,
+ icon: 'icon-park-outline:people-safe',
+ },
+ children: [
+ {
+ name: 'permission_permission',
+ path: '/permission/permission',
+ meta: {
+ title: '权限示例',
+ requiresAuth: true,
+ icon: 'icon-park-outline:right-user',
+ },
+ },
+ {
+ name: 'permission_justSuper',
+ path: '/permission/justSuper',
+ meta: {
+ title: '超管super可见',
+ requiresAuth: true,
+ roles:['super'],
+ icon: 'icon-park-outline:wrong-user',
+ },
+ },
+ ]
+ },
{
name: 'error',
path: '/error',
@@ -292,6 +322,7 @@ const userRoutes = [
title: '异常页',
requiresAuth: true,
icon: 'icon-park-outline:error-computer',
+
},
children: [
{
diff --git a/public/logo.svg b/public/favicon.svg
similarity index 100%
rename from public/logo.svg
rename to public/favicon.svg
diff --git a/src/hooks/useSystem.ts b/src/hooks/useSystem.ts
index af23208..0b99a91 100644
--- a/src/hooks/useSystem.ts
+++ b/src/hooks/useSystem.ts
@@ -1,3 +1,7 @@
+import { useAuthStore } from '@/store';
+import { isArray, isString } from '@/utils';
+
+
interface AppInfo {
/** 项目名称 */
name: string;
@@ -21,3 +25,31 @@ export function useAppInfo(): AppInfo {
desc,
};
}
+
+/** 权限判断 */
+export function usePermission() {
+ const authStore = useAuthStore()
+
+ function hasPermission(permission: Auth.RoleType | Auth.RoleType[] | undefined) {
+
+ if (!permission) return true
+
+ const { role } = authStore.userInfo
+
+ let has = role === 'super';
+ if (!has) {
+ if (isArray(permission)) {
+ has = (permission as Auth.RoleType[]).includes(role);
+ }
+ if (isString(permission)) {
+ has = (permission as Auth.RoleType) === role;
+ }
+ }
+ return has;
+ }
+
+ return {
+ hasPermission
+ };
+}
+
diff --git a/src/router/guard/dynamic.ts b/src/router/guard/dynamic.ts
index 53bf495..bf0b6bf 100644
--- a/src/router/guard/dynamic.ts
+++ b/src/router/guard/dynamic.ts
@@ -1,6 +1,7 @@
import { RouteRecordRaw } from 'vue-router';
import { BasicLayout } from '@/layouts/index';
-import { useRouteStore } from '@/store';
+import { useRouteStore, useAuthStore } from '@/store';
+import { usePermission } from '@/hooks'
// 引入所有页面
const modules = import.meta.glob('../../views/**/*.vue');
@@ -22,6 +23,13 @@ function FlatAuthRoutes(routes: AppRoute.Route[]) {
return result;
}
+function filterPermissionRoutes(routes: AppRoute.Route[]) {
+ const { hasPermission } = usePermission();
+ return routes.filter((route) => {
+ return hasPermission(route.meta.roles)
+ })
+}
+
function createCatheRoutes(routes: AppRoute.Route[]) {
return routes
.filter((item) => {
@@ -32,11 +40,13 @@ function createCatheRoutes(routes: AppRoute.Route[]) {
export async function createDynamicRoutes(routes: AppRoute.Route[]) {
// 数组降维成一维数组,然后删除所有的childen
const flatRoutes = FlatAuthRoutes(routes);
- // 对降维后的数组过滤需要缓存的路由name数组
+ /* 路由权限过滤 */
+ const permissionRoutes = filterPermissionRoutes(flatRoutes)
+ // 过滤需要缓存的路由name数组
const routeStore = useRouteStore();
- routeStore.cacheRoutes = createCatheRoutes(flatRoutes);
+ routeStore.cacheRoutes = createCatheRoutes(permissionRoutes);
// 生成路由,有redirect的不需要引入文件
- const mapRoutes = flatRoutes.map((item: any) => {
+ const mapRoutes = permissionRoutes.map((item: any) => {
if (!item.redirect) {
// 动态加载对应页面
item['component'] = modules[`../../views${item.path}/index.vue`];
diff --git a/src/router/guard/permission.ts b/src/router/guard/permission.ts
index 646276e..e9f9595 100644
--- a/src/router/guard/permission.ts
+++ b/src/router/guard/permission.ts
@@ -13,6 +13,7 @@ export async function createPermissionGuard(
// 判断有无TOKEN,登录鉴权
const isLogin = Boolean(getToken());
+
if (!isLogin) {
if (to.name === 'login') {
next();
@@ -36,9 +37,16 @@ export async function createPermissionGuard(
}
}
+ // 权限路由已经加载,仍然未找到,重定向到404
+ if (to.name === 'not-found') {
+ next({ name: 'not-found', replace: true });
+ return false;
+ }
+
// 判断当前页是否在login,则定位去首页
if (to.name === 'login') {
next({ path: '/appRoot' })
+ return false;
}
// 设置菜单高亮
@@ -52,5 +60,5 @@ export async function createPermissionGuard(
tabStore.addTab(to);
// 设置高亮标签;
tabStore.setCurrentTab(to.name as string);
- next();
+ next()
}
diff --git a/src/router/routes/index.ts b/src/router/routes/index.ts
index 2930db1..99f1a50 100644
--- a/src/router/routes/index.ts
+++ b/src/router/routes/index.ts
@@ -10,7 +10,7 @@ export const routes: RouteRecordRaw[] = [
component: BasicLayout,
children: [
{
- path: '/no-found',
+ path: '/not-found',
name: 'not-found',
component: () => import('@/views/error/not-found/index.vue'),
meta: {
@@ -38,7 +38,7 @@ export const routes: RouteRecordRaw[] = [
},
{
path: '/:pathMatch(.*)*',
- redirect: '/no-found',
+ redirect: '/not-found',
},
],
},
diff --git a/src/store/modules/route.ts b/src/store/modules/route.ts
index 43e5d0a..d3b951b 100644
--- a/src/store/modules/route.ts
+++ b/src/store/modules/route.ts
@@ -1,10 +1,12 @@
import { defineStore } from 'pinia';
-import { renderIcon, getUserInfo } from '@/utils';
+import { renderIcon, getUserInfo ,isEmpty} from '@/utils';
import { MenuOption } from 'naive-ui';
import { createDynamicRoutes } from '@/router/guard/dynamic';
import { router } from '@/router';
import { fetchUserRoutes } from '@/service';
import { staticRoutes } from '@/router/modules';
+import { useAuthStore } from '@/store';
+import { usePermission } from '@/hooks'
interface RoutesStatus {
isInitAuthRoute: boolean;
@@ -78,20 +80,28 @@ export const useRouteStore = defineStore('route-store', {
},
//* 将返回的路由表渲染成侧边栏 */
transformAuthRoutesToMenus(userRoutes: AppRoute.Route[]): MenuOption[] {
+ const authStore = useAuthStore()
+ const { role } = authStore.userInfo
return userRoutes
+ /** 隐藏不需要显示的菜单 */
.filter((item) => {
return !item.meta.hide;
})
+ .filter((item: AppRoute.Route) => {
+ const { hasPermission } = usePermission();
+ return hasPermission(item.meta.roles)
+ })
+ /** 转换为侧边菜单数据结构 */
.map((item) => {
const target: MenuOption = {
label: item.meta.title,
key: item.path,
};
- // 判断有无图标
+ /** 判断有无图标 */
if (item.meta.icon) {
target.icon = renderIcon(item.meta.icon);
}
- // 判断子元素
+ /** 判断子元素 */
if (item.children) {
const children = this.transformAuthRoutesToMenus(item.children);
// 只有子元素有且不为空时才添加
diff --git a/src/typings/business.d.ts b/src/typings/business.d.ts
index d67298f..a284142 100644
--- a/src/typings/business.d.ts
+++ b/src/typings/business.d.ts
@@ -13,7 +13,7 @@ declare namespace Auth {
token: string;
refreshToken: string;
}
-
+ type RoleType = 'super' | 'admin' | 'manage' | 'user';
interface UserInfo {
/** 用户id */
userId: string;
@@ -57,7 +57,7 @@ declare namespace CommonList {
gender: '0' | '1' | null;
email: string;
address: string;
- role: 'super' | 'admin' | 'user';
+ role: RoleType;
disabled: boolean;
}
}
diff --git a/src/typings/route.d.ts b/src/typings/route.d.ts
index 516d9b9..cfd43e5 100644
--- a/src/typings/route.d.ts
+++ b/src/typings/route.d.ts
@@ -23,7 +23,7 @@ declare namespace AppRoute {
/* 是否需要登录权限。*/
requiresAuth?: boolean;
/* 可以访问的角色 */
- roles?: string[];
+ roles?: Auth.RoleType[];
/* 是否开启页面缓存 */
keepAlive?: boolean;
/* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */
diff --git a/src/utils/is.ts b/src/utils/is.ts
index 7b8aa38..0d0a342 100644
--- a/src/utils/is.ts
+++ b/src/utils/is.ts
@@ -5,18 +5,42 @@ export function is(val: unknown, type: string) {
return toString.call(val) === `[object ${type}]`;
}
-export function isDef(val?: T): val is T {
- return typeof val !== 'undefined';
+export function isString(val: unknown): val is string {
+ return is(val, 'String');
+}
+
+export function isNumber(val: unknown): val is number {
+ return is(val, 'Number');
+}
+
+export function isBoolean(val: unknown): val is boolean {
+ return is(val, 'Boolean');
+}
+
+export function isNull(val: unknown): val is null {
+ return val === null;
}
export function isUnDef(val?: T): val is T {
return !isDef(val);
}
+export function isDef(val?: T): val is T {
+ return typeof val !== 'undefined';
+}
+
+export function isNullOrUnDef(val: unknown): val is null | undefined {
+ return isUnDef(val) || isNull(val);
+}
+
export function isObject(val: any): val is Record {
return val !== null && is(val, 'Object');
}
+export function isArray(val: any): val is Array {
+ return val && Array.isArray(val);
+}
+
export function isEmpty(val: T): val is T {
if (isArray(val) || isString(val)) {
return val.length === 0;
@@ -37,29 +61,11 @@ export function isDate(val: unknown): val is Date {
return is(val, 'Date');
}
-export function isNull(val: unknown): val is null {
- return val === null;
-}
-
-export function isNullAndUnDef(val: unknown): val is null | undefined {
- return isUnDef(val) && isNull(val);
-}
-
-export function isNullOrUnDef(val: unknown): val is null | undefined {
- return isUnDef(val) || isNull(val);
-}
-
-export function isNumber(val: unknown): val is number {
- return is(val, 'Number');
-}
export function isPromise(val: unknown): val is Promise {
return is(val, 'Promise') && isObject(val) && isFunction(val.then) && isFunction(val.catch);
}
-export function isString(val: unknown): val is string {
- return is(val, 'String');
-}
export function isFunction(val: unknown): val is Function {
return typeof val === 'function';
@@ -69,18 +75,11 @@ export function isFile(val: T | unknown): val is T {
return is(val, 'File');
}
-export function isBoolean(val: unknown): val is boolean {
- return is(val, 'Boolean');
-}
export function isRegExp(val: unknown): val is RegExp {
return is(val, 'RegExp');
}
-export function isArray(val: any): val is Array {
- return val && Array.isArray(val);
-}
-
export function isWindow(val: any): val is Window {
return typeof window !== 'undefined' && is(val, 'Window');
}
@@ -93,9 +92,8 @@ export const isServer = typeof window === 'undefined';
export const isClient = !isServer;
-export function isUrl(path: T): boolean {
+export function isUrl(path: string): boolean {
const reg =
/(((^https?:(?:\/\/)?)(?:[-;:&=\+\$,\w]+@)?[A-Za-z0-9.-]+(?::\d+)?|(?:www.|[-;:&=\+\$,\w]+@)[A-Za-z0-9.-]+)((?:\/[\+~%\/.\w-_]*)?\??(?:[-\+=&;%@.\w_]*)#?(?:[\w]*))?)$/;
- // @ts-expect-error
return reg.test(path);
}
diff --git a/src/views/permission/justSuper/index.vue b/src/views/permission/justSuper/index.vue
new file mode 100644
index 0000000..54bc675
--- /dev/null
+++ b/src/views/permission/justSuper/index.vue
@@ -0,0 +1,13 @@
+
+
+ 只有超级管理员可见
+
+
+
+
+
+
\ No newline at end of file
diff --git a/src/views/permission/permission/index.vue b/src/views/permission/permission/index.vue
new file mode 100644
index 0000000..02e0b1d
--- /dev/null
+++ b/src/views/permission/permission/index.vue
@@ -0,0 +1,14 @@
+
+
+ 权限示例:
+ 当前权限:{{ role }}
+
+
+
+
+
+
diff --git a/vite.config.ts b/vite.config.ts
index a7e7e13..ae83b36 100644
--- a/vite.config.ts
+++ b/vite.config.ts
@@ -30,7 +30,7 @@ export default defineConfig(({ command, mode }: ConfigEnv) => {
server: {
host: '0.0.0.0',
port: 3000,
- open: true,
+ open: false,
proxy: isOpenProxy ? createViteProxy(envConfig) : undefined,
},
preview: {