chore: remove mockjs

This commit is contained in:
chansee97 2024-02-29 15:05:08 +08:00
parent d700af4bc4
commit b13b156bb9
34 changed files with 332 additions and 891 deletions

View File

@ -2,7 +2,7 @@ import type { ProxyOptions } from 'vite'
/** /**
* @description: vite代理字段 * @description: vite代理字段
* @param {*} env - * @param {*} envConfig -
*/ */
export function createViteProxy(envConfig: ServiceEnvConfig) { export function createViteProxy(envConfig: ServiceEnvConfig) {
const proxy: Record<string, string | ProxyOptions> = { const proxy: Record<string, string | ProxyOptions> = {
@ -11,11 +11,6 @@ export function createViteProxy(envConfig: ServiceEnvConfig) {
changeOrigin: true, changeOrigin: true,
rewrite: path => path.replace(new RegExp(`^${envConfig.urlPattern}`), ''), rewrite: path => path.replace(new RegExp(`^${envConfig.urlPattern}`), ''),
}, },
[envConfig.secondUrlPattern]: {
target: envConfig.secondUrl,
changeOrigin: true,
rewrite: path => path.replace(new RegExp(`^${envConfig.secondUrlPattern}`), ''),
},
} }
return proxy return proxy

View File

@ -1,4 +1,6 @@
import viteCompression from 'vite-plugin-compression' // https://github.com/vbenjs/vite-plugin-compression/blob/main/README.zh_CN.md import viteCompression from 'vite-plugin-compression'
// https://github.com/vbenjs/vite-plugin-compression/blob/main/README.zh_CN.md
export default (env: ImportMetaEnv) => { export default (env: ImportMetaEnv) => {
// 默认使用gzip压缩 // 默认使用gzip压缩

View File

@ -3,9 +3,6 @@ import UnoCSS from '@unocss/vite'
import vue from './vue' import vue from './vue'
import compress from './compress' import compress from './compress'
import unplugin from './unplugin' import unplugin from './unplugin'
import mock from './mock'
// import { viteMockServe } from 'vite-plugin-mock' // https://github.com/vbenjs/vite-plugin-mock/blob/main/README.zh_CN.md
/** /**
* @description: vite插件配置 * @description: vite插件配置
@ -13,7 +10,7 @@ import mock from './mock'
* @return {*} * @return {*}
*/ */
export function setVitePlugins(env: ImportMetaEnv) { export function setVitePlugins(env: ImportMetaEnv) {
const plugins: PluginOption[] = [...vue, UnoCSS(), ...unplugin, mock] const plugins: PluginOption[] = [...vue, UnoCSS(), ...unplugin]
// 是否压缩 // 是否压缩
if (env.VITE_COMPRESS_OPEN === 'Y') if (env.VITE_COMPRESS_OPEN === 'Y')
plugins.push(compress(env)) plugins.push(compress(env))

View File

@ -1,10 +0,0 @@
import { viteMockServe } from 'vite-plugin-mock' // https://github.com/vbenjs/vite-plugin-mock/blob/main/README.zh_CN.md
export default viteMockServe({
mockPath: 'mock',
prodEnabled: true,
injectCode: `
import { setupMockServer } from '../mock';
setupMockServer();
`,
})

View File

@ -3,10 +3,14 @@ import Components from 'unplugin-vue-components/vite'
import AutoImport from 'unplugin-auto-import/vite' import AutoImport from 'unplugin-auto-import/vite'
import { NaiveUiResolver } from 'unplugin-vue-components/resolvers' import { NaiveUiResolver } from 'unplugin-vue-components/resolvers'
import Icons from 'unplugin-icons/vite' // https://github.com/antfu/unplugin-icons import Icons from 'unplugin-icons/vite'
// https://github.com/antfu/unplugin-icons
import IconsResolver from 'unplugin-icons/resolver' import IconsResolver from 'unplugin-icons/resolver'
import { createSvgIconsPlugin } from 'vite-plugin-svg-icons' // https://github.com/vbenjs/vite-plugin-svg-icons/blob/main/README.zh_CN.md import { createSvgIconsPlugin } from 'vite-plugin-svg-icons'
// https://github.com/vbenjs/vite-plugin-svg-icons/blob/main/README.zh_CN.md
export default [ export default [
AutoImport({ AutoImport({
@ -15,7 +19,8 @@ export default [
/\.[tj]sx?$/, /\.[tj]sx?$/,
/\.vue$/, /\.vue$/,
/\.vue\?vue/, /\.vue\?vue/,
/\.md$/], /\.md$/,
],
dts: 'src/typings/auto-imports.d.ts', dts: 'src/typings/auto-imports.d.ts',
}), }),
Components({ Components({
@ -29,7 +34,7 @@ export default [
}), }),
createSvgIconsPlugin({ createSvgIconsPlugin({
// 指定需要缓存的图标文件夹 // 指定需要缓存的图标文件夹
iconDirs: [path.resolve(process.cwd(), 'src/assets/icons')], iconDirs: [path.resolve(__dirname, 'src/assets/icons')],
// 指定symbolId格式 // 指定symbolId格式
symbolId: 'icon-[dir]-[name]', symbolId: 'icon-[dir]-[name]',
// inject: 'body-last', // inject: 'body-last',

View File

@ -1,6 +0,0 @@
import { createProdMockServer } from 'vite-plugin-mock/es/createProdMockServer'
import api from './module'
export function setupMockServer() {
createProdMockServer(api)
}

View File

@ -1,3 +0,0 @@
import user from './user'
export default [...user]

View File

@ -1,28 +0,0 @@
import { mock } from 'mockjs'
import { resultSuccess } from '../utils'
const userList = mock({
'list|20': [
{
'id': '@id',
'name': '@cname',
'age|20-36': 36,
'gender|1': ['0', '1', null],
'email': '@email("qq.com")',
'address': '@county(true)',
'role|1': ['super', 'admin', 'user'],
'disabled|1': true,
},
],
})
export default [
{
url: '/mock/userList',
timeout: 1000,
method: 'get',
response: () => {
return resultSuccess(userList.list)
},
},
]

View File

@ -1,493 +0,0 @@
import Mock from 'mockjs'
import { resultFailed, resultSuccess } from '../utils'
const Random = Mock.Random
const token = () => Random.string('upper', 32, 32)
const userData = [
{
userId: '1',
username: 'super',
password: '123456',
nickName: '超级管理员大人',
avatar: 'https://s2.loli.net/2023/08/26/6Xh3wuCV2eMZnEp.png',
role: 'super',
},
{
userId: '2',
username: 'admin',
password: '123456',
nickName: '管理员大人',
avatar: 'https://s2.loli.net/2023/08/26/6Xh3wuCV2eMZnEp.png',
role: 'admin',
},
{
userId: '3',
username: 'user',
password: '123456',
nickName: '用户大人',
avatar: 'https://s2.loli.net/2023/08/26/6Xh3wuCV2eMZnEp.png',
role: 'user',
},
]
const userRoutes = [
{
name: 'dashboard',
path: '/dashboard',
meta: {
title: '仪表盘',
requiresAuth: true,
icon: 'icon-park-outline:analysis',
},
children: [
{
name: 'dashboard_workbench',
path: '/dashboard/workbench',
meta: {
title: '工作台',
requiresAuth: true,
icon: 'icon-park-outline:alarm',
},
},
{
name: 'dashboard_monitor',
path: '/dashboard/monitor',
meta: {
title: '监控页',
requiresAuth: true,
icon: 'icon-park-outline:anchor',
},
},
],
},
{
name: 'test',
path: '/test',
meta: {
title: '多级菜单演示',
requiresAuth: true,
icon: 'icon-park-outline:list',
},
children: [
{
name: 'test1',
path: '/test/test1',
meta: {
title: '接口功能测试',
requiresAuth: true,
icon: 'icon-park-outline:list',
},
},
{
name: 'test2',
path: '/test/test2',
meta: {
title: '多级菜单子页',
requiresAuth: true,
icon: 'icon-park-outline:list',
},
children: [
{
name: 'test2_detail',
path: '/test/test2/detail',
meta: {
title: '多级菜单的详情页',
requiresAuth: true,
icon: 'icon-park-outline:list',
hide: true,
activeMenu: '/test/test2',
},
},
],
},
{
name: 'test3',
path: '/test/test3',
meta: {
title: '多级菜单',
requiresAuth: true,
icon: 'icon-park-outline:list',
},
children: [
{
name: 'test4',
path: '/test/test3/test4',
meta: {
title: '多级菜单3-1',
requiresAuth: true,
icon: 'icon-park-outline:list',
},
},
],
},
],
},
{
name: 'list',
path: '/list',
meta: {
title: '列表页',
requiresAuth: true,
icon: 'icon-park-outline:list-two',
},
children: [
{
name: 'list_commonList',
path: '/list/commonList',
meta: {
title: '常用列表',
requiresAuth: true,
icon: 'icon-park-outline:list-view',
},
},
{
name: 'list_cardList',
path: '/list/cardList',
meta: {
title: '卡片列表',
requiresAuth: true,
icon: 'icon-park-outline:view-grid-list',
},
},
],
},
{
name: 'plugin',
path: '/plugin',
meta: {
title: '组件示例',
requiresAuth: true,
icon: 'icon-park-outline:application-one',
},
children: [
{
name: 'plugin_charts',
path: '/plugin/charts',
meta: {
title: '图表',
requiresAuth: true,
icon: 'icon-park-outline:chart-line',
},
children: [
{
name: 'plugin_echarts',
path: '/plugin/charts/echarts',
meta: {
title: 'ECharts',
requiresAuth: true,
icon: 'icon-park-outline:chart-proportion',
},
},
{
name: 'plugin_antV',
path: '/plugin/charts/antV',
meta: {
title: 'antV',
requiresAuth: true,
icon: 'ant-design:ant-design-outlined',
},
},
],
},
{
name: 'PluginMap',
path: '/plugin/map',
meta: {
title: '地图',
requiresAuth: true,
icon: 'carbon:map',
keepAlive: true,
},
},
{
name: 'plugin_editor',
path: '/plugin/editor',
meta: {
title: '编辑器',
requiresAuth: true,
icon: 'icon-park-outline:editor',
},
children: [
{
name: 'plugin_md',
path: '/plugin/editor/md',
meta: {
title: 'MarkDown',
requiresAuth: true,
icon: 'ri:markdown-line',
},
},
{
name: 'plugin_rich',
path: '/plugin/editor/rich',
meta: {
title: '富文本',
requiresAuth: true,
icon: 'icon-park-outline:edit-one',
},
},
],
},
{
name: 'plugin_clipboard',
path: '/plugin/clipboard',
meta: {
title: '剪贴板',
requiresAuth: true,
icon: 'icon-park-outline:clipboard',
},
},
{
name: 'plugin_icons',
path: '/plugin/icons',
meta: {
title: '图标',
requiresAuth: true,
icon: 'icon-park-outline:winking-face-with-open-eyes',
},
},
{
name: 'plugin_QRCode',
path: '/plugin/QRCode',
meta: {
title: '二维码',
requiresAuth: true,
icon: 'icon-park-outline:two-dimensional-code',
},
},
],
},
{
name: 'docments',
path: '/docments',
meta: {
title: '外链文档',
requiresAuth: true,
icon: 'icon-park-outline:file-doc',
},
children: [
{
name: 'docments_vue',
path: '/docments/vue',
meta: {
title: 'vue',
requiresAuth: true,
icon: 'logos:vue',
},
},
{
name: 'docments_vite',
path: '/docments/vite',
meta: {
title: 'vite',
requiresAuth: true,
icon: 'logos:vitejs',
},
},
{
name: 'docments_vueuse',
path: '/docments/vueuse',
meta: {
title: 'VueUse外链',
requiresAuth: true,
icon: 'logos:vueuse',
herf: 'https://vueuse.org/guide/',
},
},
],
},
{
name: 'permission',
path: '/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',
meta: {
title: '异常页',
requiresAuth: true,
icon: 'icon-park-outline:error-computer',
},
children: [
{
name: '403',
path: '/error/403',
meta: {
title: '403页',
requiresAuth: true,
icon: 'carbon:error',
order: 3,
},
},
{
name: '404',
path: '/error/404',
meta: {
title: '404页',
requiresAuth: true,
icon: 'icon-park-outline:error',
order: 2,
},
},
{
name: '500',
path: '/error/500',
meta: {
title: '500页',
requiresAuth: true,
icon: 'carbon:data-error',
order: 1,
},
},
],
},
{
name: 'setting',
path: '/setting',
meta: {
title: '系统设置',
requiresAuth: true,
icon: 'icon-park-outline:setting',
},
children: [
{
name: 'setting_account',
path: '/setting/account',
meta: {
title: '用户设置',
requiresAuth: true,
icon: 'icon-park-outline:every-user',
},
},
{
name: 'setting_dictionary',
path: '/setting/dictionary',
meta: {
title: '字典设置',
requiresAuth: true,
icon: 'icon-park-outline:book-one',
},
},
{
name: 'setting_menu',
path: '/setting/menu',
meta: {
title: '菜单设置',
requiresAuth: true,
icon: 'icon-park-outline:application-menu',
},
},
{
name: 'setting_system',
path: '/setting/system',
meta: {
title: '系统配置',
requiresAuth: true,
icon: 'icon-park-outline:coordinate-system',
},
},
],
},
{
name: 'userCenter',
path: '/userCenter',
meta: {
title: '个人中心',
requiresAuth: true,
icon: 'carbon:user-avatar-filled-alt',
},
},
{
name: 'about',
path: '/about',
meta: {
title: '关于',
requiresAuth: true,
icon: 'icon-park-outline:info',
},
},
]
export default [
{
url: '/mock/auth/login',
method: 'post',
response: (options: Service.MockOption) => {
const { username = undefined, password = undefined } = options.body
if (!username || !password)
return resultFailed(null, '账号密码不全')
const userInfo = userData.find(item => item.username === username && item.password === password)
if (userInfo) {
return {
code: 200,
message: 'ok',
data: {
id: userInfo.userId,
accessToken: token(),
refreshToken: token(),
},
}
}
return resultFailed(null, '账号密码错误')
},
},
{
url: '/mock/updateToken',
method: 'post',
response: () => {
return resultSuccess({ token: token(), refreshToken: token() })
},
},
{
url: '/mock/getUserInfo',
method: 'get',
response: (options: any) => {
const { userId } = options.query
if (!userId)
return resultFailed(null, '未传入用户id')
const userInfo = userData.find(item => item.userId === userId)
if (userInfo)
return resultSuccess(userInfo)
return resultFailed(null, '未找到用户信息,请检查提交参数')
},
},
{
url: '/mock/getUserRoutes',
method: 'post',
response: () => {
return resultSuccess(userRoutes)
},
},
]

View File

@ -1,16 +0,0 @@
import Mock from 'mockjs'
export function resultSuccess(data: any, msg?: string) {
return Mock.mock({
code: 200,
data,
msg: msg || 'success',
})
}
export function resultFailed(data: any, msg?: string) {
return Mock.mock({
code: 500,
data,
msg: msg || 'failed',
})
}

View File

@ -1,5 +1,6 @@
{ {
"name": "nova-admin", "name": "nova-admin",
"type": "module",
"version": "0.1.0", "version": "0.1.0",
"private": true, "private": true,
"description": "", "description": "",
@ -70,14 +71,12 @@
"@iconify-json/icon-park-outline": "^1.1.15", "@iconify-json/icon-park-outline": "^1.1.15",
"@iconify/vue": "^4.1.1", "@iconify/vue": "^4.1.1",
"@types/crypto-js": "^4.2.2", "@types/crypto-js": "^4.2.2",
"@types/mockjs": "^1.0.10",
"@types/node": "^20.11.22", "@types/node": "^20.11.22",
"@types/qs": "^6.9.12", "@types/qs": "^6.9.12",
"@vitejs/plugin-vue": "^5.0.4", "@vitejs/plugin-vue": "^5.0.4",
"@vitejs/plugin-vue-jsx": "^3.1.0", "@vitejs/plugin-vue-jsx": "^3.1.0",
"eslint": "^8.57.0", "eslint": "^8.57.0",
"lint-staged": "^15.2.2", "lint-staged": "^15.2.2",
"mockjs": "^1.1.0",
"naive-ui": "^2.38.1", "naive-ui": "^2.38.1",
"sass": "^1.71.1", "sass": "^1.71.1",
"simple-git-hooks": "^2.9.0", "simple-git-hooks": "^2.9.0",
@ -89,7 +88,6 @@
"vite": "^5.1.4", "vite": "^5.1.4",
"vite-bundle-visualizer": "^1.0.1", "vite-bundle-visualizer": "^1.0.1",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-mock": "^3.0.1",
"vite-plugin-svg-icons": "^2.0.1", "vite-plugin-svg-icons": "^2.0.1",
"vue-tsc": "^1.8.27" "vue-tsc": "^1.8.27"
}, },

View File

@ -1,7 +1,9 @@
<script setup lang="ts"> <script setup lang="ts">
import type { ToolbarNames } from 'md-editor-v3' import type { ToolbarNames } from 'md-editor-v3'
import { MdEditor } from 'md-editor-v3' // https://imzbf.github.io/md-editor-v3/zh-CN/docs import { MdEditor } from 'md-editor-v3'
// https://imzbf.github.io/md-editor-v3/zh-CN/docs
import 'md-editor-v3/lib/style.css' import 'md-editor-v3/lib/style.css'
import { useAppStore } from '@/store' import { useAppStore } from '@/store'

View File

@ -22,7 +22,8 @@ export function setupCopy(app: App) {
} }
function copyHandler(this: any) { function copyHandler(this: any) {
if (!clipboardEnable()) return if (!clipboardEnable())
return
copy(this._copyText) copy(this._copyText)
window.$message?.success('复制成功') window.$message?.success('复制成功')
} }

View File

@ -22,16 +22,20 @@ export function usePermission() {
function hasPermission( function hasPermission(
permission: Auth.RoleType | Auth.RoleType[] | undefined, permission: Auth.RoleType | Auth.RoleType[] | undefined,
) { ) {
if (!permission) return true if (!permission)
return true
if (!authStore.userInfo) return false if (!authStore.userInfo)
return false
const { role } = authStore.userInfo const { role } = authStore.userInfo
let has = role === 'super' let has = role === 'super'
if (!has) { if (!has) {
if (isArray(permission)) has = permission.includes(role) if (isArray(permission))
has = permission.includes(role)
if (isString(permission)) has = permission === role if (isString(permission))
has = permission === role
} }
return has return has
} }

View File

@ -6,7 +6,7 @@ interface Ilogin {
} }
export function fetchLogin(params: Ilogin) { export function fetchLogin(params: Ilogin) {
return mockRequest.post<any>('/auth/login', params) return mockRequest.post<any>('/login', params)
} }
export function fetchUpdateToken(params: any) { export function fetchUpdateToken(params: any) {
return mockRequest.post<ApiAuth.loginToken>('/updateToken', params) return mockRequest.post<ApiAuth.loginToken>('/updateToken', params)
@ -14,6 +14,6 @@ export function fetchUpdateToken(params: any) {
export function fetchUserInfo(params: any) { export function fetchUserInfo(params: any) {
return mockRequest.get<Auth.UserInfo>('/getUserInfo', { params }) return mockRequest.get<Auth.UserInfo>('/getUserInfo', { params })
} }
export function fetchUserRoutes(params: { userId: number }) { export function fetchUserRoutes(params: { id: number }) {
return mockRequest.post<AppRoute.Route[]>('/getUserRoutes', params) return mockRequest.get<AppRoute.Route[]>('/getUserRoutes', { params })
} }

View File

@ -79,7 +79,7 @@ export function handleResponseError(response: AxiosResponse) {
/** /**
* @description: * @description:
* @param {Record} apiData * @param {Record} data
* @param {Service} config axios字段配置 * @param {Service} config axios字段配置
* @return {*} * @return {*}
*/ */
@ -127,12 +127,12 @@ export async function handleRefreshToken(config: AxiosRequestConfig) {
const refreshToken = local.get('refreshToken') const refreshToken = local.get('refreshToken')
const { data } = await fetchUpdateToken(refreshToken) const { data } = await fetchUpdateToken(refreshToken)
if (data) { if (data) {
local.set('refreshToken', data.token) local.set('refreshToken', data.accessToken)
local.set('token', data.refreshToken) local.set('token', data.refreshToken)
// 设置token // 设置token
if (config.headers) if (config.headers)
typeof config.headers.set === 'function' && config.headers.set('Authorization', `Bearer ${data.token || ''}`) typeof config.headers.set === 'function' && config.headers.set('Authorization', `Bearer ${data.accessToken || ''}`)
return config return config
} }

View File

@ -12,4 +12,6 @@ export const request = createRequest({
}) })
// export const secondRequest = createRequest({ baseURL: isHttpProxy ? secondUrlPattern : secondUrl }); // export const secondRequest = createRequest({ baseURL: isHttpProxy ? secondUrlPattern : secondUrl });
export const mockRequest = createRequest({ baseURL: '/mock' }) export const mockRequest = createRequest({ baseURL: 'https://mock.apifox.com/m1/4071143-0-default' }, {
msgKey: 'message',
})

View File

@ -100,7 +100,6 @@ export function createRequest(
*/ */
function handleDelete<T>( function handleDelete<T>(
url: string, url: string,
params?: any,
config?: AxiosRequestConfig, config?: AxiosRequestConfig,
) { ) {
return asyncRequest<T>({ url, method: 'delete', config }) return asyncRequest<T>({ url, method: 'delete', config })

View File

@ -5,7 +5,7 @@ import { useAppRouter } from '@/hooks'
import { local } from '@/utils' import { local } from '@/utils'
const emptyInfo: Auth.UserInfo = { const emptyInfo: Auth.UserInfo = {
userId: 0, id: 0,
userName: '', userName: '',
nickName: '', nickName: '',
avatar: '', avatar: '',
@ -96,7 +96,7 @@ export const useAuthStore = defineStore('auth-store', {
local.set('refreshToken', refreshToken) local.set('refreshToken', refreshToken)
this.token = accessToken this.token = accessToken
this.refreshToken = refreshToken this.refreshToken = refreshToken
const { error, data } = await fetchUserInfo({ userId: id }) const { error, data } = await fetchUserInfo({ id })
if (error) if (error)
return catchSuccess return catchSuccess
// 请求/存储用户信息 // 请求/存储用户信息

View File

@ -150,11 +150,11 @@ export const useRouteStore = defineStore('route-store', {
// 根据用户id来获取用户的路由 // 根据用户id来获取用户的路由
const userInfo = local.get('userInfo') const userInfo = local.get('userInfo')
if (!userInfo || !userInfo.userId) if (!userInfo || !userInfo.id)
return return
const { data: routes } = await fetchUserRoutes({ const { data: routes } = await fetchUserRoutes({
userId: userInfo.userId, id: userInfo.id,
}) })
if (!routes) if (!routes)

View File

@ -6,14 +6,14 @@
*, *,
::before, ::before,
::after { ::after {
box-sizing: border-box; box-sizing: border-box;
/* 1 */ /* 1 */
border-width: 0; border-width: 0;
/* 2 */ /* 2 */
border-style: solid; border-style: solid;
/* 2 */ /* 2 */
border-color: currentColor; border-color: currentColor;
/* 2 */ /* 2 */
} }
/* /*
@ -24,17 +24,17 @@
*/ */
html { html {
line-height: 1.5; line-height: 1.5;
/* 1 */ /* 1 */
-webkit-text-size-adjust: 100%; -webkit-text-size-adjust: 100%;
/* 2 */ /* 2 */
-moz-tab-size: 4; -moz-tab-size: 4;
/* 3 */ /* 3 */
tab-size: 4; tab-size: 4;
/* 3 */ /* 3 */
font-family: ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, font-family: ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial,
'Noto Sans', sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol', 'Noto Color Emoji'; 'Noto Sans', sans-serif, 'Apple Color Emoji', 'Segoe UI Emoji', 'Segoe UI Symbol', 'Noto Color Emoji';
/* 4 */ /* 4 */
} }
/* /*
@ -43,10 +43,10 @@ html {
*/ */
body { body {
margin: 0; margin: 0;
/* 1 */ /* 1 */
line-height: inherit; line-height: inherit;
/* 2 */ /* 2 */
} }
/* /*
@ -56,12 +56,12 @@ body {
*/ */
hr { hr {
height: 0; height: 0;
/* 1 */ /* 1 */
color: inherit; color: inherit;
/* 2 */ /* 2 */
border-top-width: 1px; border-top-width: 1px;
/* 3 */ /* 3 */
} }
/* /*
@ -69,7 +69,7 @@ Add the correct text decoration in Chrome, Edge, and Safari.
*/ */
abbr:where([title]) { abbr:where([title]) {
text-decoration: underline dotted; text-decoration: underline dotted;
} }
/* /*
@ -82,8 +82,8 @@ h3,
h4, h4,
h5, h5,
h6 { h6 {
font-size: inherit; font-size: inherit;
font-weight: inherit; font-weight: inherit;
} }
/* /*
@ -91,8 +91,8 @@ Reset links to optimize for opt-in styling instead of opt-out.
*/ */
a { a {
color: inherit; color: inherit;
text-decoration: inherit; text-decoration: inherit;
} }
/* /*
@ -101,7 +101,7 @@ Add the correct font weight in Edge and Safari.
b, b,
strong { strong {
font-weight: bolder; font-weight: bolder;
} }
/* /*
@ -113,10 +113,10 @@ code,
kbd, kbd,
samp, samp,
pre { pre {
font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', 'Courier New', monospace; font-family: ui-monospace, SFMono-Regular, Menlo, Monaco, Consolas, 'Liberation Mono', 'Courier New', monospace;
/* 1 */ /* 1 */
font-size: 1em; font-size: 1em;
/* 2 */ /* 2 */
} }
/* /*
@ -124,7 +124,7 @@ Add the correct font size in all browsers.
*/ */
small { small {
font-size: 80%; font-size: 80%;
} }
/* /*
@ -133,18 +133,18 @@ Prevent `sub` and `sup` elements from affecting the line height in all browsers.
sub, sub,
sup { sup {
font-size: 75%; font-size: 75%;
line-height: 0; line-height: 0;
position: relative; position: relative;
vertical-align: baseline; vertical-align: baseline;
} }
sub { sub {
bottom: -0.25em; bottom: -0.25em;
} }
sup { sup {
top: -0.5em; top: -0.5em;
} }
/* /*
@ -154,12 +154,12 @@ sup {
*/ */
table { table {
text-indent: 0; text-indent: 0;
/* 1 */ /* 1 */
border-color: inherit; border-color: inherit;
/* 2 */ /* 2 */
border-collapse: collapse; border-collapse: collapse;
/* 3 */ /* 3 */
} }
/* /*
@ -173,18 +173,18 @@ input,
optgroup, optgroup,
select, select,
textarea { textarea {
font-family: inherit; font-family: inherit;
/* 1 */ /* 1 */
font-size: 100%; font-size: 100%;
/* 1 */ /* 1 */
line-height: inherit; line-height: inherit;
/* 1 */ /* 1 */
color: inherit; color: inherit;
/* 1 */ /* 1 */
margin: 0; margin: 0;
/* 2 */ /* 2 */
padding: 0; padding: 0;
/* 3 */ /* 3 */
} }
/* /*
@ -193,7 +193,7 @@ Remove the inheritance of text transform in Edge and Firefox.
button, button,
select { select {
text-transform: none; text-transform: none;
} }
/* /*
@ -205,11 +205,11 @@ button,
[type='button'], [type='button'],
[type='reset'], [type='reset'],
[type='submit'] { [type='submit'] {
-webkit-appearance: button; -webkit-appearance: button;
/* 1 */ /* 1 */
/* background-color: transparent; 2 */ /* background-color: transparent; 2 */
background-image: none; background-image: none;
/* 2 */ /* 2 */
} }
/* /*
@ -217,7 +217,7 @@ Use the modern Firefox focus style for all focusable elements.
*/ */
:-moz-focusring { :-moz-focusring {
outline: auto; outline: auto;
} }
/* /*
@ -225,7 +225,7 @@ Remove the additional `:invalid` styles in Firefox. (https://github.com/mozilla/
*/ */
:-moz-ui-invalid { :-moz-ui-invalid {
box-shadow: none; box-shadow: none;
} }
/* /*
@ -233,7 +233,7 @@ Add the correct vertical alignment in Chrome and Firefox.
*/ */
progress { progress {
vertical-align: baseline; vertical-align: baseline;
} }
/* /*
@ -242,7 +242,7 @@ Correct the cursor style of increment and decrement buttons in Safari.
::-webkit-inner-spin-button, ::-webkit-inner-spin-button,
::-webkit-outer-spin-button { ::-webkit-outer-spin-button {
height: auto; height: auto;
} }
/* /*
@ -251,10 +251,10 @@ Correct the cursor style of increment and decrement buttons in Safari.
*/ */
[type='search'] { [type='search'] {
-webkit-appearance: textfield; -webkit-appearance: textfield;
/* 1 */ /* 1 */
outline-offset: -2px; outline-offset: -2px;
/* 2 */ /* 2 */
} }
/* /*
@ -262,7 +262,7 @@ Remove the inner padding in Chrome and Safari on macOS.
*/ */
::-webkit-search-decoration { ::-webkit-search-decoration {
-webkit-appearance: none; -webkit-appearance: none;
} }
/* /*
@ -271,10 +271,10 @@ Remove the inner padding in Chrome and Safari on macOS.
*/ */
::-webkit-file-upload-button { ::-webkit-file-upload-button {
-webkit-appearance: button; -webkit-appearance: button;
/* 1 */ /* 1 */
font: inherit; font: inherit;
/* 2 */ /* 2 */
} }
/* /*
@ -282,7 +282,7 @@ Add the correct display in Chrome and Safari.
*/ */
summary { summary {
display: list-item; display: list-item;
} }
/* /*
@ -302,24 +302,24 @@ hr,
figure, figure,
p, p,
pre { pre {
margin: 0; margin: 0;
} }
fieldset { fieldset {
margin: 0; margin: 0;
padding: 0; padding: 0;
} }
legend { legend {
padding: 0; padding: 0;
} }
ol, ol,
ul, ul,
menu { menu {
list-style: none; list-style: none;
margin: 0; margin: 0;
padding: 0; padding: 0;
} }
/* /*
@ -327,7 +327,7 @@ Prevent resizing textareas horizontally by default.
*/ */
textarea { textarea {
resize: vertical; resize: vertical;
} }
/* /*
@ -337,10 +337,10 @@ textarea {
input::placeholder, input::placeholder,
textarea::placeholder { textarea::placeholder {
opacity: 1; opacity: 1;
/* 1 */ /* 1 */
color: #9ca3af; color: #9ca3af;
/* 2 */ /* 2 */
} }
/* /*
@ -349,14 +349,14 @@ Set the default cursor for buttons.
button, button,
[role='button'] { [role='button'] {
cursor: pointer; cursor: pointer;
} }
/* /*
Make sure disabled buttons don't get the pointer cursor. Make sure disabled buttons don't get the pointer cursor.
*/ */
:disabled { :disabled {
cursor: default; cursor: default;
} }
/* /*
@ -373,11 +373,11 @@ audio,
iframe, iframe,
embed, embed,
object { object {
display: block; display: block;
/* 1 */ /* 1 */
vertical-align: middle; vertical-align: middle;
/* 2 */ /* 2 */
object-fit: cover; object-fit: cover;
} }
/* /*
@ -386,8 +386,8 @@ Constrain images and videos to the parent width and preserve their intrinsic asp
img, img,
video { video {
max-width: 100%; max-width: 100%;
height: auto; height: auto;
} }
/* /*
@ -395,9 +395,9 @@ Ensure the default browser behavior of the `hidden` attribute.
*/ */
[hidden] { [hidden] {
display: none; display: none;
} }
.dark { .dark {
color-scheme: dark; color-scheme: dark;
} }

40
src/typings/api.d.ts vendored
View File

@ -3,31 +3,31 @@
/** 后端返回的用户相关类型 */ /** 后端返回的用户相关类型 */
declare namespace ApiAuth { declare namespace ApiAuth {
/** 返回的用户信息 */ /** 返回的用户信息 */
type UserInfo = Auth.UserInfo; type UserInfo = Auth.UserInfo
/* 登录token字段 */ /* 登录token字段 */
interface loginToken { interface loginToken {
accessToken: string; accessToken: string
avatar?: string; avatar?: string
email?: string; email?: string
id: number; id: number
nickname?: string; nickname?: string
notes?: string; notes?: string
refreshToken: string; refreshToken: string
tel?: string; tel?: string
username: string; username: string
} }
} }
declare namespace CommonList { declare namespace CommonList {
/* 返回的性别类型 */ /* 返回的性别类型 */
type GenderType = '0' | '1' | null; type GenderType = '0' | '1' | null
interface UserList { interface UserList {
id: number; id: number
name: string; name: string
age: number; age: number
gender: GenderType; gender: GenderType
email: string; email: string
address: string; address: string
role: Auth.RoleType; role: Auth.RoleType
disabled: boolean; disabled: boolean
} }
} }

View File

@ -1,38 +1,37 @@
/** 用户相关模块 */ /** 用户相关模块 */
declare namespace Auth { declare namespace Auth {
/** 用户角色类型 */ /** 用户角色类型 */
type RoleType = 'super' | 'admin' | 'user'; type RoleType = 'super' | 'admin' | 'user'
interface UserInfo { interface UserInfo {
/** 用户id */ /** 用户id */
userId: number; id: number
/** 用户名 */ /** 用户名 */
userName: string; userName: string
/* 用户称呼 */ /* 用户称呼 */
nickName: string; nickName: string
/* 用户头像 */ /* 用户头像 */
avatar: string; avatar: string
/** 用户角色类型 */ /** 用户角色类型 */
role: RoleType; role: RoleType
} }
} }
/* 系统消息 */ /* 系统消息 */
declare namespace Message { declare namespace Message {
interface Tab { interface Tab {
key: number; key: number
name: string; name: string
badgeProps?: import('naive-ui').BadgeProps; badgeProps?: import('naive-ui').BadgeProps
list: List[]; list: List[]
} }
interface List { interface List {
id: number; id: number
title: string; title: string
icon: string; icon: string
tagTitle?: string; tagTitle?: string
tagType?: 'error' | 'info' | 'success' | 'warning'; tagType?: 'error' | 'info' | 'success' | 'warning'
description?: string; description?: string
isRead?: boolean; isRead?: boolean
date: string; date: string
} }
} }

View File

@ -1,35 +1,35 @@
interface Window { interface Window {
$loadingBar?: import('naive-ui').LoadingBarApi; $loadingBar?: import('naive-ui').LoadingBarApi
$dialog?: import('naive-ui').DialogApi; $dialog?: import('naive-ui').DialogApi
$message?: import('naive-ui').MessageApi; $message?: import('naive-ui').MessageApi
$notification?: import('naive-ui').NotificationApi; $notification?: import('naive-ui').NotificationApi
} }
declare const AMap: any; declare const AMap: any
declare const BMap: any; declare const BMap: any
interface GolbalConfig { interface GolbalConfig {
app: { app: {
proxyUrl: Record<ServiceEnvType, ServiceEnvConfig>; proxyUrl: Record<ServiceEnvType, ServiceEnvConfig>
}; }
} }
declare namespace NaiveUI { declare namespace NaiveUI {
type ThemeColor = 'default' | 'error' | 'primary' | 'info' | 'success' | 'warning'; type ThemeColor = 'default' | 'error' | 'primary' | 'info' | 'success' | 'warning'
} }
declare namespace UnionKey { declare namespace UnionKey {
/* http请求头content-type类型 */ /* http请求头content-type类型 */
type ContentType = 'application/json' | 'application/x-www-form-urlencoded' | 'multipart/form-data'; type ContentType = 'application/json' | 'application/x-www-form-urlencoded' | 'multipart/form-data'
} }
declare namespace Storage { declare namespace Storage {
interface Session { interface Session {
demoKey: string; demoKey: string
} }
interface Local { interface Local {
userInfo: Auth.UserInfo; userInfo: Auth.UserInfo
token: string; token: string
refreshToken: string; refreshToken: string
tabsRoutes: string; tabsRoutes: string
login_account: any; login_account: any
} }
} }

View File

@ -1,6 +1,6 @@
declare module '~icons/*' { declare module '~icons/*' {
import type { FunctionalComponent, SVGAttributes } from 'vue'; import type { FunctionalComponent, SVGAttributes } from 'vue'
const component: FunctionalComponent<SVGAttributes>; const component: FunctionalComponent<SVGAttributes>
export default component; export default component
} }

View File

@ -2,37 +2,37 @@ declare namespace AppRoute {
/** 单个路由的类型结构(动态路由模式:后端返回此类型结构的路由) */ /** 单个路由的类型结构(动态路由模式:后端返回此类型结构的路由) */
interface Route { interface Route {
/** 路由名称(路由唯一标识) */ /** 路由名称(路由唯一标识) */
name: string; name: string
/** 路由路径 */ /** 路由路径 */
path: string; path: string
/** 路由重定向 */ /** 路由重定向 */
redirect?: string; redirect?: string
/** 子路由 */ /** 子路由 */
children?: Route[]; children?: Route[]
/** 路由描述 */ /** 路由描述 */
meta: RouteMeta; meta: RouteMeta
/** 路由属性 */ /** 路由属性 */
// props?: boolean | Record<string, any> | ((to: any) => Record<string, any>); // props?: boolean | Record<string, any> | ((to: any) => Record<string, any>);
} }
/** 路由描述 */ /** 路由描述 */
interface RouteMeta { interface RouteMeta {
/* 页面标题,通常必选。 */ /* 页面标题,通常必选。 */
title: string; title: string
/* 图标,一般配合菜单使用 */ /* 图标,一般配合菜单使用 */
icon?: string; icon?: string
/* 是否需要登录权限。*/ /* 是否需要登录权限。 */
requiresAuth?: boolean; requiresAuth?: boolean
/* 可以访问的角色 */ /* 可以访问的角色 */
roles?: Auth.RoleType[]; roles?: Auth.RoleType[]
/* 是否开启页面缓存 */ /* 是否开启页面缓存 */
keepAlive?: boolean; keepAlive?: boolean
/* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */ /* 有些路由我们并不想在菜单中显示,比如某些编辑页面。 */
hide?: boolean; hide?: boolean
/* 菜单排序。 */ /* 菜单排序。 */
order?: number; order?: number
/* 嵌套外链 */ /* 嵌套外链 */
herf?: string; herf?: string
/** 当前路由需要选中的菜单项(用于跳转至不在左侧菜单显示的路由且需要高亮某个菜单的情况) */ /** 当前路由需要选中的菜单项(用于跳转至不在左侧菜单显示的路由且需要高亮某个菜单的情况) */
activeMenu?: string; activeMenu?: string
} }
} }

View File

@ -1,4 +1,5 @@
import 'vue-router'; import 'vue-router'
declare module 'vue-router' { declare module 'vue-router' {
interface RouteMeta extends AppRoute.RouteMeta {} interface RouteMeta extends AppRoute.RouteMeta {}
} }

View File

@ -1,52 +1,45 @@
/** 请求的相关类型 */ /** 请求的相关类型 */
declare namespace Service { declare namespace Service {
/** 后端接口返回的数据结构配置 */ /** 后端接口返回的数据结构配置 */
interface BackendResultConfig { interface BackendResultConfig {
/** 表示后端请求状态码的属性字段 */ /** 表示后端请求状态码的属性字段 */
codeKey: string; codeKey: string
/** 表示后端请求数据的属性字段 */ /** 表示后端请求数据的属性字段 */
dataKey: string; dataKey: string
/** 表示后端消息的属性字段 */ /** 表示后端消息的属性字段 */
msgKey: string; msgKey: string
/** 后端业务上定义的成功请求的状态 */ /** 后端业务上定义的成功请求的状态 */
successCode: number | string; successCode: number | string
}
type RequestErrorType = 'Axios' | 'Response' | 'Business';
type RequestCode = string | number;
interface RequestError {
/** 请求服务的错误类型 */
type: RequestErrorType;
/** 错误码 */
code: RequestCode;
/** 错误信息 */
msg: string;
}
/** 自定义的请求成功结果 */
interface SuccessResult<T = any> {
/** 请求错误 */
error: null;
/** 请求数据 */
data: T;
}
/** 自定义的请求失败结果 */
interface FailedResult {
/** 请求错误 */
error: RequestError;
/** 请求数据 */
data: null;
}
/** 自定义的请求结果 */
type RequestResult<T = any> = SuccessResult<T> | FailedResult;
interface MockOption {
url: Record<string, any>;
body: Record<string, any>;
query: Record<string, any>;
headers: Record<string, any>;
} }
}
type RequestErrorType = 'Axios' | 'Response' | 'Business'
type RequestCode = string | number
interface RequestError {
/** 请求服务的错误类型 */
type: RequestErrorType
/** 错误码 */
code: RequestCode
/** 错误信息 */
msg: string
}
/** 自定义的请求成功结果 */
interface SuccessResult<T = any> {
/** 请求错误 */
error: null
/** 请求数据 */
data: T
}
/** 自定义的请求失败结果 */
interface FailedResult {
/** 请求错误 */
error: RequestError
/** 请求数据 */
data: null
}
/** 自定义的请求结果 */
type RequestResult<T = any> = SuccessResult<T> | FailedResult
}

View File

@ -1,5 +1,6 @@
declare module '*.vue' { declare module '*.vue' {
import { DefineComponent } from 'vue'; import type { DefineComponent } from 'vue'
const component: DefineComponent;
export default component; const component: DefineComponent
export default component
} }

View File

@ -44,9 +44,9 @@ const columns: DataTableColumns = [
} as const } as const
if (rowData.gender) { if (rowData.gender) {
return ( return (
<NTag type={tagType[rowData.gender]}> <NTag type={tagType[rowData.gender]}>
{genderLabels[rowData.gender]} {genderLabels[rowData.gender]}
</NTag> </NTag>
) )
} }
}, },
@ -83,13 +83,13 @@ const columns: DataTableColumns = [
const rowData = row as unknown as CommonList.UserList const rowData = row as unknown as CommonList.UserList
return ( return (
<NSwitch <NSwitch
value={rowData.disabled} value={rowData.disabled}
onUpdateValue={disabled => onUpdateValue={disabled =>
handleUpdateDisabled(disabled, rowData.id) handleUpdateDisabled(disabled, rowData.id)}
}> >
{{ checked: () => '启用', unchecked: () => '禁用' }} {{ checked: () => '启用', unchecked: () => '禁用' }}
</NSwitch> </NSwitch>
) )
}, },
}, },
@ -100,19 +100,20 @@ const columns: DataTableColumns = [
render: (row) => { render: (row) => {
const rowData = row as unknown as CommonList.UserList const rowData = row as unknown as CommonList.UserList
return ( return (
<NSpace justify={'center'}> <NSpace justify="center">
<NButton <NButton
size={'small'} size="small"
onClick={() => handleEditTable(rowData)}> onClick={() => handleEditTable(rowData)}
编辑 >
</NButton> 编辑
<NPopconfirm onPositiveClick={() => sendMail(rowData.id)}> </NButton>
{{ <NPopconfirm onPositiveClick={() => sendMail(rowData.id)}>
default: () => '确认删除', {{
trigger: () => <NButton size={'small'}>删除</NButton>, default: () => '确认删除',
}} trigger: () => <NButton size="small">删除</NButton>,
</NPopconfirm> }}
</NSpace> </NPopconfirm>
</NSpace>
) )
}, },
}, },
@ -131,7 +132,7 @@ onMounted(() => {
async function getUserList() { async function getUserList() {
startLoading() startLoading()
await fetchUserList().then((res: any) => { await fetchUserList().then((res: any) => {
listData.value = res.data listData.value = res.data.list
endLoading() endLoading()
}) })
} }

View File

@ -50,7 +50,7 @@ function handleValidateClick() {
<n-descriptions label-placement="left" :column="2" :title="`傍晚好,${userInfo?.userName},这里是简单的个人中心模板`"> <n-descriptions label-placement="left" :column="2" :title="`傍晚好,${userInfo?.userName},这里是简单的个人中心模板`">
<n-descriptions-item label="id"> <n-descriptions-item label="id">
{{ userInfo?.userId }} {{ userInfo?.id }}
</n-descriptions-item> </n-descriptions-item>
<n-descriptions-item label="用户名"> <n-descriptions-item label="用户名">
{{ userInfo?.userName }} {{ userInfo?.userName }}

View File

@ -1,25 +1,25 @@
{ {
"compilerOptions": { "compilerOptions": {
"baseUrl": ".",
"target": "ESNext", "target": "ESNext",
"module": "ESNext",
"useDefineForClassFields": true,
"moduleResolution": "Node",
"strict": true,
"jsx": "preserve", "jsx": "preserve",
// "sourceMap": true,
"removeComments": false, //
"resolveJsonModule": true,
"isolatedModules": true,
"esModuleInterop": true,
"allowSyntheticDefaultImports": true,
"lib": ["ESNext", "DOM"], "lib": ["ESNext", "DOM"],
"skipLibCheck": true, "useDefineForClassFields": true,
"baseUrl": ".",
"module": "ESNext",
"moduleResolution": "Node",
"paths": { "paths": {
"~/*": ["./*"], "~/*": ["./*"],
"@/*": ["./src/*"] "@/*": ["./src/*"]
}, }, //
"types": ["node", "vite/client", "naive-ui/volar"] "resolveJsonModule": true,
"types": ["node", "vite/client", "naive-ui/volar"],
"strict": true,
// "sourceMap": true,
"removeComments": false,
"allowSyntheticDefaultImports": true,
"esModuleInterop": true,
"isolatedModules": true,
"skipLibCheck": true
}, },
// "include": ["src/**/*.d.ts", "src/**/*.vue", "./*.d.ts"] // "include": ["src/**/*.d.ts", "src/**/*.vue", "./*.d.ts"]
"exclude": ["node_modules", "dist"] "exclude": ["node_modules", "dist"]

View File

@ -1,4 +1,6 @@
import { defineConfig, presetAttributify, presetUno } from 'unocss' // https://github.com/unocss/unocss import { defineConfig, presetAttributify, presetUno } from 'unocss'
// https://github.com/unocss/unocss
export default defineConfig({ export default defineConfig({
presets: [presetUno({ dark: 'class' }), presetAttributify()], presets: [presetUno({ dark: 'class' }), presetAttributify()],

View File

@ -4,17 +4,13 @@ import { defineConfig, loadEnv } from 'vite'
import { createViteProxy, setVitePlugins } from './build' import { createViteProxy, setVitePlugins } from './build'
import { proxyConfig } from './src/config' import { proxyConfig } from './src/config'
// 当前执行node命令时文件夹的地址工作目录
const rootPath: string = resolve(process.cwd())
const srcPath = `${rootPath}/src`
// https://vitejs.dev/config/ // https://vitejs.dev/config/
export default defineConfig(({ mode }: ConfigEnv) => { export default defineConfig(({ mode }: ConfigEnv) => {
// 在开发环境下 command 的值为 serve 生产环境下为 build // 在开发环境下 command 的值为 serve 生产环境下为 build
// 根据当前工作目录中的 `mode` 加载 .env 文件 // 根据当前工作目录中的 `mode` 加载 .env 文件
// 设置第三个参数为 '' 来加载所有环境变量,而不管是否有 `VITE_` 前缀。 // 设置第三个参数为 '' 来加载所有环境变量,而不管是否有 `VITE_` 前缀。
const env = loadEnv(mode, process.cwd(), '') as unknown as ImportMetaEnv const env = loadEnv(mode, __dirname, '') as unknown as ImportMetaEnv
const envConfig = proxyConfig[mode as ServiceEnvType] const envConfig = proxyConfig[mode as ServiceEnvType]
return { return {
@ -22,8 +18,7 @@ export default defineConfig(({ mode }: ConfigEnv) => {
plugins: setVitePlugins(env), plugins: setVitePlugins(env),
resolve: { resolve: {
alias: { alias: {
'~': rootPath, '@': resolve(__dirname, 'src'),
'@': srcPath,
}, },
}, },
server: { server: {