feat(service.utils): 完善axiso处理流程,loacal存储

This commit is contained in:
Coffee-crocodile 2023-01-12 16:45:11 +08:00
parent ef6392615b
commit b33cb9e353
17 changed files with 1285 additions and 781 deletions

View File

@ -3,408 +3,416 @@ import { resultSuccess } from '../utils';
const Random = Mock.Random; const Random = Mock.Random;
const token = Random.string('upper', 32, 32); const token = () => Random.string('upper', 32, 32);
const userInfo = { const userInfo = {
userId: '1', userId: '1',
userName: 'admin', userName: 'admin',
realName: '管理员大人', realName: '管理员大人',
avatar: 'https://z3.ax1x.com/2021/10/29/5jnWgf.jpg', avatar: 'https://z3.ax1x.com/2021/10/29/5jnWgf.jpg',
role: 'admin', role: 'admin',
password: '123456', password: '123456',
}; };
const userRoutes = [ const userRoutes = [
{ {
name: 'dashboard', name: 'dashboard',
path: '/dashboard', path: '/dashboard',
redirect: '/dashboard/workbench', redirect: '/dashboard/workbench',
meta: { meta: {
title: '仪表盘', title: '仪表盘',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:analysis', icon: 'icon-park-outline:analysis',
}, },
children: [ children: [
{ {
name: 'dashboard_workbench', name: 'dashboard_workbench',
path: '/dashboard/workbench', path: '/dashboard/workbench',
meta: { meta: {
title: '工作台', title: '工作台',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:alarm', icon: 'icon-park-outline:alarm',
}, },
}, },
{ {
name: 'dashboard_monitor', name: 'dashboard_monitor',
path: '/dashboard/monitor', path: '/dashboard/monitor',
meta: { meta: {
title: '监控页', title: '监控页',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:anchor', icon: 'icon-park-outline:anchor',
}, },
}, },
], ],
}, },
{ {
name: 'test', name: 'test',
path: '/test', path: '/test',
redirect: '/test/test1', redirect: '/test/test1',
meta: { meta: {
title: '多级菜单演示', title: '多级菜单演示',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list', icon: 'icon-park-outline:list',
}, },
children: [ children: [
{ {
name: 'test1', name: 'test1',
path: '/test/test1', path: '/test/test1',
meta: { meta: {
title: '多级菜单1', title: '多级菜单1',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list', icon: 'icon-park-outline:list',
}, },
}, },
{ {
name: 'test2', name: 'test2',
path: '/test/test2', path: '/test/test2',
meta: { meta: {
title: '多级菜单2', title: '多级菜单2',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list', icon: 'icon-park-outline:list',
}, },
children: [ children: [
{ {
name: 'test2_detail', name: 'test2_detail',
path: '/test/test2/detail', path: '/test/test2/detail',
meta: { meta: {
title: '多级菜单2的详情页', title: '多级菜单2的详情页',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list', icon: 'icon-park-outline:list',
hide: true, hide: true,
activeMenu: '/test/test2', activeMenu: '/test/test2',
}, },
}, },
], ],
}, },
{ {
name: 'test3', name: 'test3',
path: '/test/test3', path: '/test/test3',
meta: { meta: {
title: '多级菜单3', title: '多级菜单3',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list', icon: 'icon-park-outline:list',
}, },
children: [ children: [
{ {
name: 'test4', name: 'test4',
path: '/test/test3/test4', path: '/test/test3/test4',
meta: { meta: {
title: '多级菜单3-1', title: '多级菜单3-1',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list', icon: 'icon-park-outline:list',
}, },
}, },
], ],
}, },
], ],
}, },
{ {
name: 'list', name: 'list',
path: '/list', path: '/list',
redirect: '/list/commonList', redirect: '/list/commonList',
meta: { meta: {
title: '列表页', title: '列表页',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list-two', icon: 'icon-park-outline:list-two',
}, },
children: [ children: [
{ {
name: 'list_commonList', name: 'list_commonList',
path: '/list/commonList', path: '/list/commonList',
meta: { meta: {
title: '常用列表', title: '常用列表',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:list-view', icon: 'icon-park-outline:list-view',
}, },
}, },
{ {
name: 'list_cardList', name: 'list_cardList',
path: '/list/cardList', path: '/list/cardList',
meta: { meta: {
title: '卡片列表', title: '卡片列表',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:view-grid-list', icon: 'icon-park-outline:view-grid-list',
}, },
}, },
], ],
}, },
{ {
name: 'plugin', name: 'plugin',
path: '/plugin', path: '/plugin',
redirect: '/plugin/charts', redirect: '/plugin/charts',
meta: { meta: {
title: '组件示例', title: '组件示例',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:application-one', icon: 'icon-park-outline:application-one',
}, },
children: [ children: [
{ {
name: 'plugin_charts', name: 'plugin_charts',
path: '/plugin/charts', path: '/plugin/charts',
meta: { meta: {
title: '图表', title: '图表',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:chart-line', icon: 'icon-park-outline:chart-line',
}, },
children: [ children: [
{ {
name: 'plugin_echarts', name: 'plugin_echarts',
path: '/plugin/charts/echarts', path: '/plugin/charts/echarts',
meta: { meta: {
title: 'ECharts', title: 'ECharts',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:chart-proportion', icon: 'icon-park-outline:chart-proportion',
}, },
}, },
{ {
name: 'plugin_antV', name: 'plugin_antV',
path: '/plugin/charts/antV', path: '/plugin/charts/antV',
meta: { meta: {
title: 'antV', title: 'antV',
requiresAuth: true, requiresAuth: true,
icon: 'ant-design:ant-design-outlined', icon: 'ant-design:ant-design-outlined',
}, },
}, },
], ],
}, },
{ {
name: 'plugin_map', name: 'plugin_map',
path: '/plugin/map', path: '/plugin/map',
meta: { meta: {
title: '地图', title: '地图',
requiresAuth: true, requiresAuth: true,
icon: 'carbon:map', icon: 'carbon:map',
keepAlive: true, keepAlive: true,
}, },
}, },
{ {
name: 'plugin_editor', name: 'plugin_editor',
path: '/plugin/editor', path: '/plugin/editor',
meta: { meta: {
title: '编辑器', title: '编辑器',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:editor', icon: 'icon-park-outline:editor',
}, },
children: [ children: [
{ {
name: 'plugin_md', name: 'plugin_md',
path: '/plugin/editor/md', path: '/plugin/editor/md',
meta: { meta: {
title: 'MarkDown', title: 'MarkDown',
requiresAuth: true, requiresAuth: true,
icon: 'ri:markdown-line', icon: 'ri:markdown-line',
}, },
}, },
{ {
name: 'plugin_rich', name: 'plugin_rich',
path: '/plugin/editor/rich', path: '/plugin/editor/rich',
meta: { meta: {
title: '富文本', title: '富文本',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:edit-one', icon: 'icon-park-outline:edit-one',
}, },
}, },
], ],
}, },
{ {
name: 'plugin_clipboard', name: 'plugin_clipboard',
path: '/plugin/clipboard', path: '/plugin/clipboard',
meta: { meta: {
title: '剪贴板', title: '剪贴板',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:clipboard', icon: 'icon-park-outline:clipboard',
}, },
}, },
{ {
name: 'plugin_icons', name: 'plugin_icons',
path: '/plugin/icons', path: '/plugin/icons',
meta: { meta: {
title: '图标', title: '图标',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:winking-face-with-open-eyes', icon: 'icon-park-outline:winking-face-with-open-eyes',
}, },
}, },
{ {
name: 'plugin_QRCode', name: 'plugin_QRCode',
path: '/plugin/QRCode', path: '/plugin/QRCode',
meta: { meta: {
title: '二维码', title: '二维码',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:two-dimensional-code', icon: 'icon-park-outline:two-dimensional-code',
}, },
}, },
], ],
}, },
{ {
name: 'docments', name: 'docments',
path: '/docments', path: '/docments',
redirect: '/docments/not-found', redirect: '/docments/not-found',
meta: { meta: {
title: '外链文档', title: '外链文档',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:file-doc', icon: 'icon-park-outline:file-doc',
}, },
children: [ children: [
{ {
name: 'docments_vue', name: 'docments_vue',
path: '/docments/vue', path: '/docments/vue',
meta: { meta: {
title: 'vue', title: 'vue',
requiresAuth: true, requiresAuth: true,
icon: 'logos:vue', icon: 'logos:vue',
}, },
}, },
{ {
name: 'docments_vite', name: 'docments_vite',
path: '/docments/vite', path: '/docments/vite',
meta: { meta: {
title: 'vite', title: 'vite',
requiresAuth: true, requiresAuth: true,
icon: 'logos:vitejs', icon: 'logos:vitejs',
}, },
}, },
{ {
name: 'docments_vueuse', name: 'docments_vueuse',
path: '/docments/vueuse', path: '/docments/vueuse',
meta: { meta: {
title: 'VueUse外链', title: 'VueUse外链',
requiresAuth: true, requiresAuth: true,
icon: 'logos:vueuse', icon: 'logos:vueuse',
herf: 'https://vueuse.org/guide/', herf: 'https://vueuse.org/guide/',
}, },
}, },
], ],
}, },
{ {
name: 'error', name: 'error',
path: '/error', path: '/error',
redirect: '/error/not-found', redirect: '/error/not-found',
meta: { meta: {
title: '异常页', title: '异常页',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:error-computer', icon: 'icon-park-outline:error-computer',
}, },
children: [ children: [
{ {
name: 'not-found', name: 'not-found',
path: '/error/not-found', path: '/error/not-found',
meta: { meta: {
title: '404页', title: '404页',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:error', icon: 'icon-park-outline:error',
}, },
}, },
{ {
name: 'not-permission', name: 'not-permission',
path: '/error/not-permission', path: '/error/not-permission',
meta: { meta: {
title: '403页', title: '403页',
requiresAuth: true, requiresAuth: true,
icon: 'carbon:error', icon: 'carbon:error',
}, },
}, },
{ {
name: 'service-error', name: 'service-error',
path: '/error/service-error', path: '/error/service-error',
meta: { meta: {
title: '500页', title: '500页',
requiresAuth: true, requiresAuth: true,
icon: 'carbon:data-error', icon: 'carbon:data-error',
}, },
}, },
], ],
}, },
{ {
name: 'setting', name: 'setting',
path: '/setting', path: '/setting',
redirect: '/setting/account', redirect: '/setting/account',
meta: { meta: {
title: '系统设置', title: '系统设置',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:setting', icon: 'icon-park-outline:setting',
}, },
children: [ children: [
{ {
name: 'setting_account', name: 'setting_account',
path: '/setting/account', path: '/setting/account',
meta: { meta: {
title: '用户设置', title: '用户设置',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:user', icon: 'icon-park-outline:user',
}, },
}, },
{ {
name: 'setting_dictionary', name: 'setting_dictionary',
path: '/setting/dictionary', path: '/setting/dictionary',
meta: { meta: {
title: '字典设置', title: '字典设置',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:book-one', icon: 'icon-park-outline:book-one',
}, },
}, },
{ {
name: 'setting_menu', name: 'setting_menu',
path: '/setting/menu', path: '/setting/menu',
meta: { meta: {
title: '菜单设置', title: '菜单设置',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:application-menu', icon: 'icon-park-outline:application-menu',
}, },
}, },
{ {
name: 'setting_system', name: 'setting_system',
path: '/setting/system', path: '/setting/system',
meta: { meta: {
title: '系统配置', title: '系统配置',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:coordinate-system', icon: 'icon-park-outline:coordinate-system',
}, },
}, },
], ],
}, },
{ {
name: 'about', name: 'about',
path: '/about', path: '/about',
meta: { meta: {
title: '关于', title: '关于',
requiresAuth: true, requiresAuth: true,
icon: 'icon-park-outline:info', icon: 'icon-park-outline:info',
}, },
}, },
]; ];
export default [ export default [
{ {
url: '/mock/login', url: '/mock/login',
timeout: 1000, timeout: 1000,
method: 'post', method: 'post',
response: () => { response: () => {
return resultSuccess({ token }); return resultSuccess({ token: token(), refreshToken: token() });
}, },
}, },
{ {
url: '/mock/getUserInfo', url: '/mock/updateToken',
timeout: 1000, timeout: 1000,
method: 'get', method: 'post',
response: () => { response: () => {
return resultSuccess(userInfo); return resultSuccess({ token: token(), refreshToken: token() });
}, },
}, },
{ {
url: '/mock/getUserRoutes', url: '/mock/getUserInfo',
timeout: 1000, timeout: 1000,
method: 'post', method: 'get',
response: () => { response: () => {
return resultSuccess(userRoutes); return resultSuccess(userInfo);
}, },
}, },
{
url: '/mock/getUserRoutes',
timeout: 1000,
method: 'post',
response: () => {
return resultSuccess(userRoutes);
},
},
]; ];

View File

@ -12,13 +12,7 @@
<script setup lang="ts"> <script setup lang="ts">
import { useAppStore } from './store'; import { useAppStore } from './store';
import { import { zhCN, dateZhCN, GlobalThemeOverrides, useOsTheme } from 'naive-ui';
zhCN,
dateZhCN,
GlobalThemeOverrides,
useOsTheme,
darkTheme,
} from 'naive-ui';
import themeConfig from './theme.json'; import themeConfig from './theme.json';
const locale = zhCN; const locale = zhCN;

View File

@ -26,9 +26,9 @@
<div /> <div />
</div> </div>
</div> </div>
<h2 class="text-28px font-500 text-#646464"> <n-h1 class="z-1">
{{ title }} {{ title }}
</h2> </n-h1>
</div> </div>
</template> </template>
@ -45,7 +45,7 @@ const { title } = useAppInfo();
align-items: center; align-items: center;
justify-content: center; justify-content: center;
flex-direction: column; flex-direction: column;
gap: 10vh; gap: 15vh;
position: fixed; position: fixed;
background-color: aliceblue; background-color: aliceblue;
z-index: 1; z-index: 1;
@ -112,8 +112,7 @@ const { title } = useAppInfo();
right: var(--right); right: var(--right);
bottom: var(--bottom); bottom: var(--bottom);
left: var(--left); left: var(--left);
transform: rotateY(var(--rotateY)) rotateX(var(--rotateX)) transform: rotateY(var(--rotateY)) rotateX(var(--rotateX)) translateZ(var(--translateZ));
translateZ(var(--translateZ));
} }
.boxes .box > div:nth-child(1) { .boxes .box > div:nth-child(1) {

View File

@ -1,3 +1,20 @@
/** 默认实例的Aixos配置 */
export const DEFAULT_AXIOS_OPTIONS = {
// 请求超时时间,默认15秒
timeout: 15 * 1000,
};
/** 默认实例的后端字段配置 */
export const DEFAULT_BACKEND_OPTIONS = {
codeKey: 'code',
dataKey: 'data',
msgKey: 'msg',
successCode: 200,
};
/** 错误信息的显示时间 */
export const ERROR_MSG_DURATION = 3 * 1000;
/** 默认的请求错误code */ /** 默认的请求错误code */
export const DEFAULT_REQUEST_ERROR_CODE = 'DEFAULT'; export const DEFAULT_REQUEST_ERROR_CODE = 'DEFAULT';
@ -32,3 +49,9 @@ export const ERROR_STATUS = {
505: '505: http版本不支持该请求~', 505: '505: http版本不支持该请求~',
[DEFAULT_REQUEST_ERROR_CODE]: DEFAULT_REQUEST_ERROR_MSG, [DEFAULT_REQUEST_ERROR_CODE]: DEFAULT_REQUEST_ERROR_MSG,
}; };
/** token刷新的code */
export const REFRESH_TOKEN_CODE = [888, 999];
/** 没有错误提示的code */
export const ERROR_NO_TIP_STATUS = [10000];

View File

@ -1,9 +1,11 @@
/* 缓存的Key值 */ /* 缓存的Key值 */
export enum EnumStorageKey { export enum EnumStorageKey {
/* 用户信息 */ /* 用户信息 */
userInfo = '__USER_INFO__', userInfo = '__USER_INFO__',
/* token */ /* token */
token = '__TOKEN__', token = '__TOKEN__',
/* 标签栏信息 */ /* refreshToken */
tabsRoutes = '__TABS_ROUTES__', refreshToken = '__REFRESH_TOKEN__',
/* 标签栏信息 */
tabsRoutes = '__TABS_ROUTES__',
} }

View File

@ -1,8 +1,16 @@
<template> <template>
<n-dropdown trigger="click" :options="options" @select="handleSelect"> <n-dropdown
trigger="click"
:options="options"
@select="handleSelect"
>
<HeaderButton> <HeaderButton>
<n-avatar round size="large" :src="authStore.userInfo?.avatar" /> <n-avatar
{{ authStore.userInfo?.realName }} round
size="large"
:src="userInfo.avatar"
/>
{{ userInfo.realName }}
</HeaderButton> </HeaderButton>
</n-dropdown> </n-dropdown>
</template> </template>
@ -12,28 +20,28 @@ import HeaderButton from '../common/HeaderButton.vue';
import { renderIcon } from '@/utils/icon'; import { renderIcon } from '@/utils/icon';
import { useAuthStore } from '@/store'; import { useAuthStore } from '@/store';
const authStore = useAuthStore(); const { userInfo, resetAuthStore } = useAuthStore();
const options = [ const options = [
{ {
label: '个人中心', label: '个人中心',
key: '/presonalCenter', key: '/presonalCenter',
icon: renderIcon('icon-park-outline:grinning-face'), icon: renderIcon('icon-park-outline:grinning-face'),
}, },
{ {
type: 'divider', type: 'divider',
key: 'd1', key: 'd1',
}, },
{ {
label: '退出登录', label: '退出登录',
key: 'loginOut', key: 'loginOut',
icon: renderIcon('icon-park-outline:logout'), icon: renderIcon('icon-park-outline:logout'),
}, },
]; ];
const handleSelect = (key: string | number) => { const handleSelect = (key: string | number) => {
if (key === 'loginOut') { if (key === 'loginOut') {
authStore.resetAuthStore(); resetAuthStore();
} }
}; };
</script> </script>

View File

@ -1,15 +1,18 @@
import { mockRequest } from '../http'; import { mockRequest } from '../http';
interface Ilogin { interface Ilogin {
userName: string; userName: string;
password: string; password: string;
} }
export function fetchLogin(params: Ilogin) { export function fetchLogin(params: Ilogin) {
return mockRequest.post('/login', params); return mockRequest.post('/login', params);
}
export function fetchUpdateToken(params: string) {
return mockRequest.post('/updateToken', params);
} }
export function fetchUserInfo() { export function fetchUserInfo() {
return mockRequest.get('/getUserInfo'); return mockRequest.get('/getUserInfo');
} }
export function fetchUserRoutes(params: string) { export function fetchUserRoutes(params: string) {
return mockRequest.post('/getUserRoutes', params); return mockRequest.post('/getUserRoutes', params);
} }

View File

@ -2,29 +2,48 @@ import { request } from '../http';
import { mockRequest } from '../http'; import { mockRequest } from '../http';
interface Itest { interface Itest {
data: string; data: string;
} }
/* get方法测试 */ /* get方法测试 */
export function fetachGet() { export function fetachGet() {
return request.get('/getAPI'); return request.get('/getAPI');
} }
/* post方法测试 */ /* post方法测试 */
export function fetachPost(params: Itest) { export function fetachPost(params: Itest) {
return request.post('/postAPI', params); return request.post('/postAPI', params);
} }
/* delete方法测试 */ /* delete方法测试 */
export function fetachDelete() { export function fetachDelete() {
return request.Delete('/deleteAPI'); return request.Delete('/deleteAPI');
} }
/* put方法测试 */ /* put方法测试 */
export function fetachPut(params: Itest) { export function fetachPut(params: Itest) {
return request.put('/putAPI', params); return request.put('/putAPI', params);
} }
/* patch方法测试 */ /* patch方法测试 */
export function fetachPatch(params: Itest) { export function fetachPatch(params: Itest) {
return request.patch('/patchAPI', params); return request.patch('/patchAPI', params);
} }
/* 测试状态码500失败 */
export function testFailedRequest() {
return request.get('/filedRequest');
}
/* 测试业务码500失败 */
export function testFailedResponse() {
return request.get('/filedResponse');
}
/* 测试token刷新接口 */
export function testUpdataToken() {
return request.get('/updataToken');
}
/* 测试token刷新接口 */
export function testFailedResponse_NT() {
return request.get('/failedResponse_NT');
}
/* mock方法测试 */ /* mock方法测试 */
export function fetchMock() { export function fetchMock() {
return mockRequest.post('/login'); return mockRequest.post('/login');
} }

View File

@ -1,5 +1,6 @@
import type { AxiosResponse, AxiosError } from 'axios'; import type { AxiosResponse, AxiosError, AxiosRequestConfig } from 'axios';
import { import {
ERROR_MSG_DURATION,
DEFAULT_REQUEST_ERROR_CODE, DEFAULT_REQUEST_ERROR_CODE,
DEFAULT_REQUEST_ERROR_MSG, DEFAULT_REQUEST_ERROR_MSG,
NETWORK_ERROR_CODE, NETWORK_ERROR_CODE,
@ -7,15 +8,22 @@ import {
REQUEST_TIMEOUT_CODE, REQUEST_TIMEOUT_CODE,
REQUEST_TIMEOUT_MSG, REQUEST_TIMEOUT_MSG,
ERROR_STATUS, ERROR_STATUS,
ERROR_NO_TIP_STATUS,
} from '@/config'; } from '@/config';
import { useAuthStore } from '@/store';
import { getRefreshToken } from '@/utils';
import { fetchUpdateToken } from '@/service';
import { setToken, setRefreshToken } from '@/utils';
type ErrorStatus = keyof typeof ERROR_STATUS; type ErrorStatus = keyof typeof ERROR_STATUS;
/** /**
* @description: axios或http错误 * @description: axios或http错误
* @param {AxiosError} err * @param {AxiosError} err
* @return {*} * @return {*}
*/ */
export function handleAxiosError(err: AxiosError) { export function handleAxiosError(err: AxiosError) {
const error = { const error: Service.RequestError = {
type: 'Axios', type: 'Axios',
code: DEFAULT_REQUEST_ERROR_CODE, code: DEFAULT_REQUEST_ERROR_CODE,
msg: DEFAULT_REQUEST_ERROR_MSG, msg: DEFAULT_REQUEST_ERROR_MSG,
@ -38,6 +46,8 @@ export function handleAxiosError(err: AxiosError) {
Object.assign(error, { code: errorCode, msg }); Object.assign(error, { code: errorCode, msg });
} }
showError(error);
return error; return error;
} }
@ -47,7 +57,7 @@ export function handleAxiosError(err: AxiosError) {
* @return {*} * @return {*}
*/ */
export function handleResponseError(response: AxiosResponse) { export function handleResponseError(response: AxiosResponse) {
const error = { const error: Service.RequestError = {
type: 'Axios', type: 'Axios',
code: DEFAULT_REQUEST_ERROR_CODE, code: DEFAULT_REQUEST_ERROR_CODE,
msg: DEFAULT_REQUEST_ERROR_MSG, msg: DEFAULT_REQUEST_ERROR_MSG,
@ -63,6 +73,8 @@ export function handleResponseError(response: AxiosResponse) {
Object.assign(error, { type: 'Response', code: errorCode, msg }); Object.assign(error, { type: 'Response', code: errorCode, msg });
} }
showError(error);
return error; return error;
} }
@ -72,13 +84,69 @@ export function handleResponseError(response: AxiosResponse) {
* @param {Service} config axios字段配置 * @param {Service} config axios字段配置
* @return {*} * @return {*}
*/ */
export function handleBusinessError(apiData: Record<string, any>, config: Service.BackendResultConfig) { export function handleBusinessError(data: Record<string, any>, config: Service.BackendResultConfig) {
const { codeKey, msgKey } = config; const { codeKey, msgKey } = config;
const error = { const error: Service.RequestError = {
type: 'Business', type: 'Business',
code: apiData[codeKey], code: data[codeKey],
msg: apiData[msgKey], msg: data[msgKey],
}; };
showError(error);
return error; return error;
} }
/**
* @description:
* @param {any} data
* @param {Service} error
* @return {*} result
*/
export async function handleServiceResult<T = any>(data: any, error: Service.RequestError | null) {
if (error) {
const fail: Service.FailedResult = {
error,
data: null,
};
return fail;
}
const success: Service.SuccessResult<T> = {
error: null,
data,
};
return success;
}
/**
* @description: token刷新
* @param {AxiosRequestConfig} config axios字段配置
* @return {*}
*/
export async function handleRefreshToken(config: AxiosRequestConfig) {
const { resetAuthStore } = useAuthStore();
const refreshToken = getRefreshToken();
const { data } = await fetchUpdateToken(refreshToken);
if (data) {
setRefreshToken(data.token);
setToken(data.refreshToken);
// 设置token
if (config.headers) {
typeof config.headers.set === 'function' && config.headers.set('Authorization', `Bearer ${data.token || ''}`);
}
return config;
}
resetAuthStore();
return null;
}
export function showError(error: Service.RequestError) {
// 如果error不需要提示,则跳过
const code = Number(error.code);
if (ERROR_NO_TIP_STATUS.includes(code)) return;
window.console.warn(error.code, error.msg);
window.$message?.error(error.msg, { duration: ERROR_MSG_DURATION });
}

View File

@ -1,7 +1,16 @@
import axios from 'axios'; import axios from 'axios';
import type { AxiosInstance, AxiosRequestConfig, AxiosResponse, AxiosError } from 'axios'; import type { AxiosInstance, AxiosRequestConfig, AxiosError } from 'axios';
import { getToken } from '@/utils'; import { getToken } from '@/utils';
import { handleAxiosError, handleResponseError, handleBusinessError } from './handle'; import { REFRESH_TOKEN_CODE } from '@/config';
import {
handleAxiosError,
handleResponseError,
handleBusinessError,
handleServiceResult,
handleRefreshToken,
} from './handle';
import { DEFAULT_AXIOS_OPTIONS, DEFAULT_BACKEND_OPTIONS } from '@/config';
/** /**
* @description: axios请求类 * @description: axios请求类
@ -14,26 +23,16 @@ export default class createAxiosInstance {
// 基础配置 // 基础配置
axiosConfig: AxiosRequestConfig = {}; axiosConfig: AxiosRequestConfig = {};
constructor( constructor(axiosConfig: AxiosRequestConfig, backendConfig: Service.BackendResultConfig) {
axiosConfig: AxiosRequestConfig,
backendConfig: Service.BackendResultConfig = {
codeKey: 'code',
dataKey: 'data',
msgKey: 'msg',
successCode: '200',
}
) {
this.backendConfig = backendConfig;
// 设置了axios实例上的一些默认配置,新配置会覆盖默认配置 // 设置了axios实例上的一些默认配置,新配置会覆盖默认配置
this.instance = axios.create({ timeout: 60000, ...axiosConfig }); this.instance = axios.create({ ...DEFAULT_AXIOS_OPTIONS, ...axiosConfig });
this.backendConfig = { ...DEFAULT_BACKEND_OPTIONS, ...backendConfig };
this.setInterceptor(); this.setInterceptor();
} }
// 设置类拦截器的函数 // 设置类拦截器的函数
setInterceptor() { setInterceptor() {
this.instance.interceptors.request.use( this.instance.interceptors.request.use(
async (config) => { (config) => {
const handleConfig = { ...config }; const handleConfig = { ...config };
if (handleConfig.headers) { if (handleConfig.headers) {
// 设置token // 设置token
@ -42,36 +41,41 @@ export default class createAxiosInstance {
} }
return handleConfig; return handleConfig;
}, },
(axiosError: AxiosError) => { (error: AxiosError) => {
const error = handleAxiosError(axiosError); const errorResult = handleAxiosError(error);
Promise.reject(error); return handleServiceResult(null, errorResult);
} }
); );
this.instance.interceptors.response.use( this.instance.interceptors.response.use(
async (response) => { async (response): Promise<any> => {
const { status } = response; const { status } = response;
if (status === 200) { if (status === 200) {
// 获取返回的数据 // 获取返回的数据
const apiData = response.data; const apiData = response.data;
const { codeKey, successCode } = this.backendConfig; const { codeKey, successCode, dataKey } = this.backendConfig;
// 请求成功 // 请求成功
if (apiData[codeKey] == successCode) { if (apiData[codeKey] == successCode) {
// return apiData[dataKey]; return handleServiceResult(apiData[dataKey], null);
return apiData; }
// token失效, 刷新token
if (REFRESH_TOKEN_CODE.includes(apiData[codeKey])) {
const config = await handleRefreshToken(response.config);
if (config) {
return this.instance.request(config);
}
} }
//TODO 添加刷新token的操作
// 业务请求失败 // 业务请求失败
const error = handleBusinessError(apiData, this.backendConfig); const errorResult = handleBusinessError(apiData, this.backendConfig);
return Promise.reject(error); return handleServiceResult(null, errorResult);
} }
// 接口请求失败 // 接口请求失败
const error = handleResponseError(response); const errorResult = handleResponseError(response);
return Promise.reject(error); return handleServiceResult(null, errorResult);
}, },
(axiosError: AxiosError) => { (error: AxiosError) => {
// 处理http常见错误进行全局提示等 // 处理http常见错误进行全局提示等
const error = handleAxiosError(axiosError); const errorResult = handleAxiosError(error);
return Promise.reject(error); return handleServiceResult(null, errorResult);
} }
); );
} }

View File

@ -1,101 +1,102 @@
import { defineStore } from 'pinia'; import { defineStore } from 'pinia';
import { fetchLogin, fetchUserInfo } from '@/service'; import { fetchLogin, fetchUserInfo } from '@/service';
import { setUserInfo, getUserInfo, getToken, setToken, clearAuthStorage } from '@/utils/auth'; import { setUserInfo, getUserInfo, getToken, setToken, clearAuthStorage, setRefreshToken } from '@/utils/auth';
import { router } from '@/router'; import { router } from '@/router';
import { useAppRouter } from '@/hooks'; import { useAppRouter } from '@/hooks';
import { unref } from 'vue'; import { unref } from 'vue';
import { useRouteStore } from './route'; import { useRouteStore } from './route';
export const useAuthStore = defineStore('auth-store', { export const useAuthStore = defineStore('auth-store', {
state: () => { state: () => {
return { return {
userInfo: getUserInfo(), userInfo: getUserInfo(),
token: getToken(), token: getToken(),
loginLoading: false, loginLoading: false,
}; };
}, },
getters: { getters: {
/** 是否登录 */ /** 是否登录 */
isLogin(state) { isLogin(state) {
return Boolean(state.token); return Boolean(state.token);
}, },
}, },
actions: { actions: {
/* 登录退出,重置用户信息等 */ /* 登录退出,重置用户信息等 */
resetAuthStore() { resetAuthStore() {
const route = unref(router.currentRoute); const route = unref(router.currentRoute);
const { toLogin } = useAppRouter(false); const { toLogin } = useAppRouter(false);
const { resetRouteStore } = useRouteStore(); const { resetRouteStore } = useRouteStore();
// 清除本地缓存 // 清除本地缓存
clearAuthStorage(); clearAuthStorage();
// 清空路由、菜单等数据 // 清空路由、菜单等数据
resetRouteStore(); resetRouteStore();
this.$reset(); this.$reset();
if (route.meta.requiresAuth) { if (route.meta.requiresAuth) {
toLogin(); toLogin();
} }
}, },
/* 用户登录 */ /* 用户登录 */
async login(userName: string, password: string) { async login(userName: string, password: string) {
this.loginLoading = true; this.loginLoading = true;
const { data } = await fetchLogin({ userName, password }); const { data } = await fetchLogin({ userName, password });
// 处理登录信息 // 处理登录信息
await this.handleAfterLogin(data); await this.handleAfterLogin(data);
this.loginLoading = false; this.loginLoading = false;
}, },
/* 登录后的处理函数 */ /* 登录后的处理函数 */
async handleAfterLogin(data: Auth.loginToken) { async handleAfterLogin(data: Auth.loginToken) {
// 将token和userInfo保存下来 // 将token和userInfo保存下来
const catchSuccess = await this.catchUserInfo(data); const catchSuccess = await this.catchUserInfo(data);
// 添加路由和菜单 // 添加路由和菜单
const { initAuthRoute } = useRouteStore(); const { initAuthRoute } = useRouteStore();
await initAuthRoute(); await initAuthRoute();
// 登录写入信息成功 // 登录写入信息成功
if (catchSuccess) { if (catchSuccess) {
// 进行重定向跳转 // 进行重定向跳转
const { toLoginRedirect } = useAppRouter(false); const { toLoginRedirect } = useAppRouter(false);
toLoginRedirect(); toLoginRedirect();
// 触发用户提示 // 触发用户提示
window.$notification?.success({ window.$notification?.success({
title: '登录成功!', title: '登录成功!',
content: `欢迎回来😊,${this.userInfo.realName}!`, content: `欢迎回来😊,${this.userInfo.realName}!`,
duration: 3000, duration: 3000,
}); });
return; return;
} }
// 如果不成功则重置存储 // 如果不成功则重置存储
this.resetAuthStore(); this.resetAuthStore();
// 登录失败提示 // 登录失败提示
window.$notification?.error({ window.$notification?.error({
title: '登录失败!', title: '登录失败!',
content: `验证失败,请检查账号密码`, content: `验证失败,请检查账号密码`,
duration: 3000, duration: 3000,
}); });
}, },
/* 缓存用户信息 */ /* 缓存用户信息 */
async catchUserInfo(userToken: Auth.loginToken) { async catchUserInfo(userToken: Auth.loginToken) {
let catchSuccess = false; let catchSuccess = false;
// 先存储token // 先存储token
const { token } = userToken; const { token, refreshToken } = userToken;
setToken(token); setToken(token);
setRefreshToken(refreshToken);
// 请求/存储用户信息 // 请求/存储用户信息
const { data } = await fetchUserInfo(); const { data } = await fetchUserInfo();
setUserInfo(data); setUserInfo(data);
// 再将token和userInfo初始化 // 再将token和userInfo初始化
this.userInfo = data; this.userInfo = data;
this.token = token; this.token = token;
catchSuccess = true; catchSuccess = true;
return catchSuccess; return catchSuccess;
}, },
}, },
}); });

View File

@ -1,62 +1,63 @@
/** 用户相关模块 */ /** 用户相关模块 */
declare namespace Auth { declare namespace Auth {
/** /**
* () * ()
* - super: () * - super: ()
* - admin: 管理员 * - admin: 管理员
* - user: 用户 * - user: 用户
* - custom: 自定义角色 * - custom: 自定义角色
*/ */
/** 用户信息 */ /** 用户信息 */
interface loginToken { interface loginToken {
token: string; token: string;
} refreshToken: string;
}
interface UserInfo { interface UserInfo {
/** 用户id */ /** 用户id */
userId: string; userId: string;
/** 用户名 */ /** 用户名 */
userName: string; userName: string;
/* 用户称呼 */ /* 用户称呼 */
realName: string; realName: string;
/* 用户头像 */ /* 用户头像 */
avatar: string; avatar: string;
/** 用户角色类型 */ /** 用户角色类型 */
role: RoleType; role: RoleType;
/* 密码 */ /* 密码 */
password: string; password: string;
} }
} }
/* 系统消息 */ /* 系统消息 */
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;
} }
} }
declare namespace CommonList { declare namespace CommonList {
interface UserList { interface UserList {
id: number; id: number;
name: string; name: string;
age: number; age: number;
gender: '0' | '1' | null; gender: '0' | '1' | null;
email: string; email: string;
address: string; address: string;
role: 'super' | 'admin' | 'user'; role: 'super' | 'admin' | 'user';
disabled: boolean; disabled: boolean;
} }
} }

View File

@ -1,21 +1,52 @@
/** 请求的相关类型 */ /** 请求的相关类型 */
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;
} }
/** 菜单项配置 */ /** 菜单项配置 */
type GlobalMenuOption = import('naive-ui').MenuOption & { type GlobalMenuOption = import('naive-ui').MenuOption & {
key: string; key: string;
label: string; label: string;
icon?: () => import('vue').VNodeChild; icon?: () => import('vue').VNodeChild;
children?: GlobalMenuOption[]; children?: GlobalMenuOption[];
}; };

View File

@ -1,35 +1,49 @@
import { setLocal, getLocal, removeLocal } from './storage'; import { loacl } from './storage';
import { EnumStorageKey } from '@/enum'; import { EnumStorageKey } from '@/enum';
const DURATION = 6 * 60 * 60; const DURATION = 6 * 60 * 60;
/* 获取当前token */ /* 获取当前token */
export function getToken() { export function getToken() {
return getLocal(EnumStorageKey.token); return loacl.get(EnumStorageKey.token);
} }
/* 设置token */ /* 设置token */
export function setToken(data: string) { export function setToken(data: string) {
setLocal(EnumStorageKey.token, data, DURATION); loacl.set(EnumStorageKey.token, data, DURATION);
} }
/* 移除token */ /* 移除token */
export function removeToken() { export function removeToken() {
removeLocal(EnumStorageKey.token); loacl.remove(EnumStorageKey.token);
}
/* 获取当前refreshToken */
export function getRefreshToken() {
return loacl.get(EnumStorageKey.refreshToken);
}
/* 设置refreshToken */
export function setRefreshToken(data: string) {
loacl.set(EnumStorageKey.refreshToken, data, DURATION);
}
/* 移除refreshToken */
export function removeRefreshToken() {
loacl.remove(EnumStorageKey.refreshToken);
} }
/* 获取用户详情 */ /* 获取用户详情 */
export function getUserInfo() { export function getUserInfo() {
return getLocal(EnumStorageKey.userInfo); return loacl.get(EnumStorageKey.userInfo);
} }
/* 设置用户详情 */ /* 设置用户详情 */
export function setUserInfo(data: any) { export function setUserInfo(data: any) {
setLocal(EnumStorageKey.userInfo, data); loacl.set(EnumStorageKey.userInfo, data);
} }
/* 移除用户详情 */ /* 移除用户详情 */
export function removeUserInfo() { export function removeUserInfo() {
removeLocal(EnumStorageKey.userInfo); loacl.remove(EnumStorageKey.userInfo);
} }
/** 去除用户相关缓存 */ /** 去除用户相关缓存 */
export function clearAuthStorage() { export function clearAuthStorage() {
removeToken(); removeToken();
removeUserInfo(); removeRefreshToken();
removeUserInfo();
} }

View File

@ -1,6 +1,3 @@
// 默认缓存期限为7天
const DEFAULT_CACHE_TIME = 60 * 60 * 24 * 7;
// 读取缓存前缀 // 读取缓存前缀
const prefix = import.meta.env.VITE_STORAGE_PREFIX as string; const prefix = import.meta.env.VITE_STORAGE_PREFIX as string;
@ -8,69 +5,81 @@ interface StorageData {
value: any; value: any;
expire: number | null; expire: number | null;
} }
/** /**
* LocalStorage部分操作 * LocalStorage部分操作
*/ */
export const setLocal = (key: string, value: unknown, expire: number | null = DEFAULT_CACHE_TIME): void => { function createLocalStorage() {
const storageData: StorageData = { // 默认缓存期限为7天
value, const DEFAULT_CACHE_TIME = 60 * 60 * 24 * 7;
expire: expire !== null ? new Date().getTime() + expire * 1000 : null,
};
const json = JSON.stringify(storageData);
localStorage.setItem(prefix + key, json);
};
export const getLocal = (key: string) => { function set(key: string, value: any, expire: number = DEFAULT_CACHE_TIME) {
const json = localStorage.getItem(prefix + key); const storageData: StorageData = {
if (!json) return null; value,
expire: new Date().getTime() + expire * 1000,
let storageData: StorageData | null = null; };
storageData = JSON.parse(json as string); const json = JSON.stringify(storageData);
window.localStorage.setItem(prefix + key, json);
if (storageData) {
const { value, expire } = storageData;
if (expire === null || expire >= Date.now()) {
return value;
}
} }
removeLocal(key);
return null;
};
export const removeLocal = (key: string): void => { function get(key: string) {
localStorage.removeItem(prefix + key); const json = window.localStorage.getItem(prefix + key);
}; if (!json) return null;
export const clearLocal = (): void => { let storageData: StorageData | null = null;
localStorage.clear(); storageData = JSON.parse(json as string);
};
if (storageData) {
const { value, expire } = storageData;
if (expire === null || expire >= Date.now()) {
return value;
}
}
loacl.remove(key);
return null;
}
function remove(key: string) {
window.localStorage.removeItem(prefix + key);
}
function clear() {
window.localStorage.clear();
}
return {
set,
get,
remove,
clear,
};
}
/** /**
* sessionStorage部分操作 * sessionStorage部分操作
*/ */
export function setSession(key: string, value: unknown) {
const json = JSON.stringify(value);
sessionStorage.setItem(prefix + key, json);
}
export function getSession<T>(key: string) { function createSessionStorage() {
const json = sessionStorage.getItem(prefix + key); function set(key: string, value: any) {
let data: T | null = null; const json = JSON.stringify(value);
if (json) { window.sessionStorage.setItem(prefix + key, json);
try { }
data = JSON.parse(json); function get<T>(key: string) {
} catch { const json = sessionStorage.getItem(prefix + key);
// 防止解析失败 let data: T | null = null;
} if (json) {
try {
data = JSON.parse(json);
} catch {
// 防止解析失败
}
}
return data;
}
function remove(key: string) {
window.sessionStorage.removeItem(prefix + key);
}
function clear() {
window.sessionStorage.clear();
} }
return data;
} }
export function removeSession(key: string) { export const loacl = createLocalStorage();
window.sessionStorage.removeItem(prefix + key); export const session = createSessionStorage();
}
export function clearSession() {
window.sessionStorage.clear();
}

View File

@ -1,33 +1,55 @@
<template> <template>
<n-grid :x-gap="16" :y-gap="16"> <n-grid
:x-gap="16"
:y-gap="16"
>
<n-gi :span="24"> <n-gi :span="24">
<n-card> <n-card>
<n-space justify="space-between"> <n-space justify="space-between">
<div class="flex-y-center"> <div class="flex-y-center">
<n-avatar round :size="64" :src="authStore.userInfo?.avatar" /> <n-avatar
round
:size="64"
:src="userInfo.avatar"
/>
<div class="pl-12px"> <div class="pl-12px">
<h3 class="text-18px font-semibold">您好{{ authStore.userInfo.realName }},今天又是充满活力的一天</h3> <h3 class="text-18px font-semibold">
<p class="leading-30px text-[#999]">今日多云转晴20 - 25</p> 您好{{ userInfo.realName }},今天又是充满活力的一天
</h3>
<p class="leading-30px text-[#999]">
今日多云转晴20 - 25
</p>
</div> </div>
</div> </div>
<n-row class="w-450px"> <n-row class="w-450px">
<n-col :span="10"> <n-col :span="10">
<n-statistic label="统计数据" :value="99"> <n-statistic
label="统计数据"
:value="99"
>
<template #prefix> <template #prefix>
<i-icon-park-outline-chart-histogram /> <i-icon-park-outline-chart-histogram />
</template> </template>
<template #suffix>/ 100</template> <template #suffix>
/ 100
</template>
</n-statistic> </n-statistic>
</n-col> </n-col>
<n-col :span="10"> <n-col :span="10">
<n-statistic label="活跃用户" value="34,123"> <n-statistic
label="活跃用户"
value="34,123"
>
<template #prefix> <template #prefix>
<i-icon-park-outline-customer /> <i-icon-park-outline-customer />
</template> </template>
</n-statistic> </n-statistic>
</n-col> </n-col>
<n-col :span="4"> <n-col :span="4">
<n-statistic label="待办" :value="18"> <n-statistic
label="待办"
:value="18"
>
<template #prefix> <template #prefix>
<i-icon-park-outline-list-checkbox /> <i-icon-park-outline-list-checkbox />
</template> </template>
@ -38,54 +60,108 @@
</n-card> </n-card>
</n-gi> </n-gi>
<n-gi :span="17"> <n-gi :span="17">
<n-space vertical :size="16"> <n-space
vertical
:size="16"
>
<n-card title="项目"> <n-card title="项目">
<template #header-extra><n-button type="primary" quaternary>更多</n-button></template> <template #header-extra>
<n-grid :x-gap="8" :y-gap="8"> <n-button
type="primary"
quaternary
>
更多
</n-button>
</template>
<n-grid
:x-gap="8"
:y-gap="8"
>
<n-gi :span="8"> <n-gi :span="8">
<n-card title="卡片" hoverable> <n-card
title="卡片"
hoverable
>
卡片内容 卡片内容
<template #action>#action</template> <template #action>
#action
</template>
</n-card> </n-card>
</n-gi> </n-gi>
<n-gi :span="8"> <n-gi :span="8">
<n-card title="卡片" hoverable> <n-card
title="卡片"
hoverable
>
卡片内容 卡片内容
<template #action>#action</template> <template #action>
#action
</template>
</n-card> </n-card>
</n-gi> </n-gi>
<n-gi :span="8"> <n-gi :span="8">
<n-card title="卡片" hoverable> <n-card
title="卡片"
hoverable
>
卡片内容 卡片内容
<template #action>#action</template> <template #action>
#action
</template>
</n-card> </n-card>
</n-gi> </n-gi>
<n-gi :span="8"> <n-gi :span="8">
<n-card title="卡片" hoverable> <n-card
title="卡片"
hoverable
>
卡片内容 卡片内容
<template #action>#action</template> <template #action>
#action
</template>
</n-card> </n-card>
</n-gi> </n-gi>
<n-gi :span="8"> <n-gi :span="8">
<n-card title="卡片" hoverable> <n-card
title="卡片"
hoverable
>
卡片内容 卡片内容
<template #action>#action</template> <template #action>
#action
</template>
</n-card> </n-card>
</n-gi> </n-gi>
<n-gi :span="8"> <n-gi :span="8">
<n-card title="卡片" hoverable> <n-card
title="卡片"
hoverable
>
卡片内容 卡片内容
<template #action>#action</template> <template #action>
#action
</template>
</n-card> </n-card>
</n-gi> </n-gi>
</n-grid> </n-grid>
</n-card> </n-card>
<n-card title="动态"> <n-card title="动态">
<template #header-extra><n-button type="primary" quaternary>更多</n-button></template> <template #header-extra>
<n-button
type="primary"
quaternary
>
更多
</n-button>
</template>
<n-list hoverable> <n-list hoverable>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-avatar round :size="48" :src="authStore.userInfo?.avatar" /> <n-avatar
round
:size="48"
:src="userInfo.avatar"
/>
</template> </template>
<n-thing <n-thing
title="客怎车" title="客怎车"
@ -95,7 +171,11 @@
</n-list-item> </n-list-item>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-avatar round :size="48" :src="authStore.userInfo?.avatar" /> <n-avatar
round
:size="48"
:src="userInfo.avatar"
/>
</template> </template>
<n-thing <n-thing
title="街健五大神技" title="街健五大神技"
@ -105,7 +185,11 @@
</n-list-item> </n-list-item>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-avatar round :size="48" :src="authStore.userInfo?.avatar" /> <n-avatar
round
:size="48"
:src="userInfo.avatar"
/>
</template> </template>
<n-thing <n-thing
title="天下岂有七十年太子乎" title="天下岂有七十年太子乎"
@ -115,7 +199,11 @@
</n-list-item> </n-list-item>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-avatar round :size="48" :src="authStore.userInfo?.avatar" /> <n-avatar
round
:size="48"
:src="userInfo.avatar"
/>
</template> </template>
<n-thing <n-thing
title="你干嘛~哈哈~哎哟~" title="你干嘛~哈哈~哎哟~"
@ -128,47 +216,146 @@
</n-space> </n-space>
</n-gi> </n-gi>
<n-gi :span="7"> <n-gi :span="7">
<n-space vertical :size="16"> <n-space
vertical
:size="16"
>
<n-card title="公告"> <n-card title="公告">
<template #header-extra><n-button type="primary" quaternary>更多</n-button></template> <template #header-extra>
<n-button
type="primary"
quaternary
>
更多
</n-button>
</template>
<n-list> <n-list>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-tag :bordered="false" type="info" size="small">通知</n-tag> <n-tag
:bordered="false"
type="info"
size="small"
>
通知
</n-tag>
</template> </template>
<n-button text>漂洋过海上大专</n-button> <n-button text>
漂洋过海上大专
</n-button>
</n-list-item> </n-list-item>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-tag :bordered="false" type="success" size="small">消息</n-tag> <n-tag
:bordered="false"
type="success"
size="small"
>
消息
</n-tag>
</template> </template>
<n-button text>你在玩很新的东西</n-button> <n-button text>
你在玩很新的东西
</n-button>
</n-list-item> </n-list-item>
<n-list-item> <n-list-item>
<template #prefix> <template #prefix>
<n-tag :bordered="false" type="warning" size="small">活动</n-tag> <n-tag
:bordered="false"
type="warning"
size="small"
>
活动
</n-tag>
</template> </template>
<n-button text>上岸第一剑先斩意中人</n-button> <n-button text>
上岸第一剑先斩意中人
</n-button>
</n-list-item> </n-list-item>
</n-list> </n-list>
</n-card> </n-card>
<n-card title="快捷入口"> <n-card title="快捷入口">
<n-grid :x-gap="8" :y-gap="8"> <n-grid
<n-gi :span="8"><n-card title="卡片" hoverable>卡片内容</n-card></n-gi> :x-gap="8"
<n-gi :span="8"><n-card title="卡片" hoverable>卡片内容</n-card></n-gi> :y-gap="8"
<n-gi :span="8"><n-card title="卡片" hoverable>卡片内容</n-card></n-gi> >
<n-gi :span="8"><n-card title="卡片" hoverable>卡片内容</n-card></n-gi> <n-gi :span="8">
<n-gi :span="8"><n-card title="卡片" hoverable>卡片内容</n-card></n-gi> <n-card
<n-gi :span="8"><n-card title="卡片" hoverable>卡片内容</n-card></n-gi> title="卡片"
hoverable
>
卡片内容
</n-card>
</n-gi>
<n-gi :span="8">
<n-card
title="卡片"
hoverable
>
卡片内容
</n-card>
</n-gi>
<n-gi :span="8">
<n-card
title="卡片"
hoverable
>
卡片内容
</n-card>
</n-gi>
<n-gi :span="8">
<n-card
title="卡片"
hoverable
>
卡片内容
</n-card>
</n-gi>
<n-gi :span="8">
<n-card
title="卡片"
hoverable
>
卡片内容
</n-card>
</n-gi>
<n-gi :span="8">
<n-card
title="卡片"
hoverable
>
卡片内容
</n-card>
</n-gi>
</n-grid> </n-grid>
</n-card> </n-card>
<n-card title="任务进度"> <n-card title="任务进度">
<n-timeline> <n-timeline>
<n-timeline-item content="啊" /> <n-timeline-item content="啊" />
<n-timeline-item type="success" title="成功" content="哪里成功" time="2018-04-03 20:46" /> <n-timeline-item
<n-timeline-item type="error" content="哪里错误" time="2018-04-03 20:46" /> type="success"
<n-timeline-item type="warning" title="警告" content="哪里警告" time="2018-04-03 20:46" /> title="成功"
<n-timeline-item type="info" title="信息" content="是的" time="2018-04-03 20:46" line-type="dashed" /> content="哪里成功"
time="2018-04-03 20:46"
/>
<n-timeline-item
type="error"
content="哪里错误"
time="2018-04-03 20:46"
/>
<n-timeline-item
type="warning"
title="警告"
content="哪里警告"
time="2018-04-03 20:46"
/>
<n-timeline-item
type="info"
title="信息"
content="是的"
time="2018-04-03 20:46"
line-type="dashed"
/>
<n-timeline-item content="啊" /> <n-timeline-item content="啊" />
</n-timeline> </n-timeline>
</n-card> </n-card>
@ -180,7 +367,7 @@
<script setup lang="ts"> <script setup lang="ts">
import { useAuthStore } from '@/store'; import { useAuthStore } from '@/store';
const authStore = useAuthStore(); const { userInfo } = useAuthStore();
</script> </script>
<style scoped></style> <style scoped></style>

View File

@ -1,13 +1,109 @@
<template> <template>
<div> <div>
<n-space> <n-space>
<n-button strong secondary type="success" @click="pinter">check env</n-button> <n-button
<n-button strong secondary type="success" @click="get">to get</n-button> strong
<n-button strong secondary type="success" @click="post">to post</n-button> secondary
<n-button strong secondary type="success" @click="delete2">to delete</n-button> type="success"
<n-button strong secondary type="success" @click="put">to put</n-button> @click="pinter"
<n-button strong secondary type="success" @click="patch">to patch</n-button> >
<n-button strong secondary type="success" @click="mock">to use mock</n-button> check env
</n-button>
<n-button
strong
secondary
type="success"
@click="get"
>
use online get
</n-button>
<n-button
strong
secondary
type="success"
@click="post"
>
use online post
</n-button>
<n-button
strong
secondary
type="success"
@click="delete2"
>
use online delete
</n-button>
<n-button
strong
secondary
type="success"
@click="put"
>
use online put
</n-button>
<n-button
strong
secondary
type="success"
@click="patch"
>
use online patch
</n-button>
<n-button
strong
secondary
type="success"
@click="mock"
>
to use mock
</n-button>
<n-button
strong
secondary
type="success"
@click="patch"
>
use online patch
</n-button>
<n-button
strong
secondary
type="success"
@click="mock"
>
to use mock
</n-button>
<n-button
strong
secondary
type="error"
@click="failedRequest"
>
请求失败
</n-button>
<n-button
strong
secondary
type="error"
@click="failedResponse"
>
响应失败
</n-button>
<n-button
strong
secondary
type="error"
@click="failedResponse_NT"
>
响应失败(无提示)
</n-button>
<n-button
strong
secondary
@click="updataToken"
>
测试刷新token接口
</n-button>
</n-space> </n-space>
{{ msg }} {{ msg }}
@ -15,50 +111,87 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { fetachGet, fetachPost, fetachDelete, fetachPut, fetachPatch, fetchMock } from '@/service'; import {
fetachGet,
fetachPost,
fetachDelete,
fetachPut,
fetachPatch,
fetchMock,
testFailedRequest,
testFailedResponse,
testFailedResponse_NT,
testUpdataToken,
} from '@/service';
import { ref } from 'vue'; import { ref } from 'vue';
const msg = ref(); const msg = ref();
const pinter = () => { const pinter = () => {
msg.value = import.meta.env; msg.value = import.meta.env;
}; };
const get = () => { const get = () => {
fetachGet().then((res) => { fetachGet().then((res) => {
msg.value = res; msg.value = res;
}); });
}; };
const delete2 = () => { const delete2 = () => {
fetachDelete().then((res) => { fetachDelete().then((res) => {
msg.value = res; msg.value = res;
}); });
}; };
const post = () => { const post = () => {
const params = { const params = {
data: '2022-2-2', data: '2022-2-2',
}; };
fetachPost(params).then((res) => { fetachPost(params).then((res) => {
msg.value = res; msg.value = res;
}); });
}; };
const put = () => { const put = () => {
const params = { const params = {
data: '2022-2-2', data: '2022-2-2',
}; };
fetachPut(params).then((res) => { fetachPut(params).then((res) => {
msg.value = res; msg.value = res;
}); });
}; };
const patch = () => { const patch = () => {
const params = { const params = {
data: '2022-2-2', data: '2022-2-2',
}; };
fetachPatch(params).then((res) => { fetachPatch(params).then((res) => {
msg.value = res; msg.value = res;
}); });
};
//
const failedRequest = () => {
testFailedRequest().then((res) => {
msg.value = res;
});
};
//
const failedResponse = () => {
testFailedResponse().then((res) => {
msg.value = res;
});
};
//
const failedResponse_NT = () => {
testFailedResponse_NT().then((res) => {
msg.value = res;
});
};
// token
const updataToken = () => {
testUpdataToken().then((res) => {
msg.value = res;
});
}; };
const mock = () => { const mock = () => {
fetchMock().then((res) => { fetchMock().then((res) => {
msg.value = res; msg.value = res;
}); });
}; };
</script> </script>