feat: 完善部分代码

This commit is contained in:
h_mo 2024-06-17 23:56:24 +08:00
parent bbbdaf7f1d
commit 862ee17c88
17 changed files with 200 additions and 151 deletions

View File

@ -29,6 +29,8 @@ export default antfu({
'style/brace-style': ['error', '1tbs', { allowSingleLine: true }], 'style/brace-style': ['error', '1tbs', { allowSingleLine: true }],
'vue/script-indent': ['error', 2, { baseIndent: 0 }], 'vue/script-indent': ['error', 2, { baseIndent: 0 }],
'no-console': process.env.NODE_ENV === 'production' ? 'error' : 'off', 'no-console': process.env.NODE_ENV === 'production' ? 'error' : 'off',
'no-debugger': process.env.NODE_ENV === 'production' ? 'error' : 'off',
'no-useless-catch': 'off',
}, },
env: { env: {
node: true, node: true,

View File

@ -1,10 +1,13 @@
<script setup lang="ts"> <script setup lang="ts">
import { onHide, onLaunch, onShow } from '@dcloudio/uni-app'; import { onHide, onLaunch, onShow } from '@dcloudio/uni-app';
import { useUserStore } from '@/stores/modules/user';
onLaunch(() => { onLaunch(() => {
console.log('App Launch'); console.log('App Launch');
}); });
onShow(() => { onShow(() => {
const userStore = useUserStore();
userStore.initUserInfo();
console.log('App Show'); console.log('App Show');
}); });
onHide(() => { onHide(() => {

View File

@ -7,7 +7,8 @@
"navigationBarTitleText": "Home" "navigationBarTitleText": "Home"
}, },
"meta": { "meta": {
"tabBar": true "tabBar": true,
"ignoreAuth": true
} }
}, },
{ {
@ -17,7 +18,8 @@
"navigationBarTitleText": "Demo" "navigationBarTitleText": "Demo"
}, },
"meta": { "meta": {
"tabBar": true "tabBar": true,
"ignoreAuth": true
} }
}, },
{ {

View File

@ -1,9 +1,41 @@
<script lang="ts" setup> <script lang="ts" setup>
import BasicButton from '@/components/BasicButton/index.vue';
import { useUserStore } from '@/stores/modules/user';
const router = useRouter();
const userStore = useUserStore();
const { loggedIn, userInfo } = storeToRefs(userStore);
function handleJump(url: string) {
router.push(url);
}
//
function handleLoginOut() {
userStore.logout();
}
</script> </script>
<template> <template>
<view>关于</view> <view class="text-md pt-36">
<view v-if="loggedIn" class="text-center">
<image class="h-56 w-56" :src="userInfo?.avatar" />
<view class="mt-2">
{{ userInfo?.nickname }}
</view>
</view>
<view class="mt-6 flex flex-col gap-y-xl justify-center items-center">
<BasicButton @click="handleJump('/pages/log/index?id=4345&title=log&word=关键词')">
log
</BasicButton>
<BasicButton v-if="loggedIn" @click="handleLoginOut">
登出
</BasicButton>
<BasicButton v-else @click="handleJump('/pages/login/index')">
登入
</BasicButton>
</view>
</view>
</template> </template>
<style lang="scss" scoped> <style lang="scss" scoped>

View File

@ -1,16 +1,12 @@
<script lang="ts" setup> <script lang="ts" setup>
import { ref } from 'vue';
const demo = ref('Demo'); const demo = ref('Demo');
</script> </script>
<template> <template>
<view>{{ demo }}</view> <view class="pt-36 text-lg font-medium flex justify-center items-center">
{{ demo }}
</view>
</template> </template>
<style lang="scss" scoped> <style lang="scss" scoped>
.container {
padding: 128rpx 0;
text-align: center;
}
</style> </style>

View File

@ -1,24 +1,19 @@
<script setup lang="ts"> <script setup lang="ts">
const router = useRouter(); import { getRandomIcon } from '@/utils/character';
import { platform } from '@/utils/platform';
function onClick() { const logo = getRandomIcon();
router.push({ name: 'Login' }); const appTitle = 'uniapp-vue3';
}
function goLog() {
router.push({ name: 'Log' });
}
</script> </script>
<template> <template>
<view class="border border-blue border-solid text-mini text-primary center h-44"> <view class="pt-36 flex flex-col gap-y-2 items-center">
home <image :src="logo" class="h-56 w-56" alt="" mode="widthFix" />
<view class="text-xl font-semibold">
{{ appTitle }}
</view>
<view>当前平台{{ platform }}</view>
</view> </view>
<button @click="onClick">
登录
</button>
<button @click="goLog">
Log
</button>
</template> </template>
<style lang="scss"> <style lang="scss">

View File

@ -9,10 +9,12 @@ const userStore = useUserStore();
<template> <template>
<view class="text-center"> <view class="text-center">
登录后访问log <view class="mt-36 text-center">
登录后访问log
</view>
<image class="my-4 h-48 w-48" :src="userStore.userInfo?.avatar" mode="aspectFit" lazy-load="false" binderror="" bindload="" />
<view>{{ userStore.userInfo?.nickname }}</view>
</view> </view>
<image class="" :src="userStore.userInfo?.avatar" mode="aspectFit" lazy-load="false" binderror="" bindload="" />
<view>用户昵称{{ userStore.userInfo?.nickname }}</view>
</template> </template>
<style lang="scss" scoped></style> <style lang="scss" scoped></style>

View File

@ -1,4 +1,5 @@
import type { Router } from 'uni-mini-router/lib/interfaces'; import type { Router } from 'uni-mini-router/lib/interfaces';
import { isLogin } from '@/utils/auth';
export function createRouterGuard(router: Router) { export function createRouterGuard(router: Router) {
createBeforeEachGuard(router); createBeforeEachGuard(router);
@ -6,37 +7,38 @@ export function createRouterGuard(router: Router) {
} }
function createBeforeEachGuard(router: Router) { function createBeforeEachGuard(router: Router) {
router.beforeEach((_1, _2, next) => { router.beforeEach((to, _, next) => {
// const authStore = useAuthStore(); console.log('beforeEach', to);
// if (to && to?.meta?.ignoreAuth) { const _isLogin = isLogin();
// // 如果目标路由忽略验证直接跳转 if (to && to?.meta?.ignoreAuth) {
// next(); // 如果目标路由忽略验证直接跳转
// } else if (!authStore.isLogin && to && to.name !== 'Login') { next();
// // 如果没有登录且目标路由不是登录页面则跳转到登录页面 } else if (!_isLogin && to && to.name !== 'Login') {
// // 将目标路由和参数传入登录页面,登录成功后直接跳转到目标路由,优化体验 // 如果没有登录且目标路由不是登录页面则跳转到登录页面
// next({ name: 'Login', params: { redirect: to.name!, ...to.query }, navType: 'push' }); // 将目标路由和参数传入登录页面,登录成功后直接跳转到目标路由,优化体验
// } else if (authStore.isLogin && to && to.name === 'Login') { next({ name: 'Login', params: { redirect: to.name!, ...to.query }, navType: 'push' });
// // 如果已经登录且目标页面是登录页面则跳转至首页 } else if (_isLogin && to && to.name === 'Login') {
// next({ name: 'Home', navType: 'replaceAll' }); // 如果已经登录且目标页面是登录页面则跳转至首页
// } else { next({ name: 'Home', navType: 'replaceAll' });
// next(); } else {
// } next();
}
next(); next();
}); });
} }
function createAfterEachGuard(router: Router) { function createAfterEachGuard(router: Router) {
router.afterEach((_) => { router.afterEach((to) => {
// if (to && to?.meta?.ignoreAuth) if (to && to?.meta?.ignoreAuth)
// return; return;
// const authStore = useAuthStore(); const _isLogin = isLogin();
// if (!authStore.isLogin && to && to.name !== 'Login') { if (!_isLogin && to && to.name !== 'Login') {
// // 如果没有登录且目标路由不是登录页面则跳转到登录页面 // 如果没有登录且目标路由不是登录页面则跳转到登录页面
// router.push({ name: 'Login', params: { ...to.query } }); router.push({ name: 'Login', params: { ...to.query } });
// } else if (authStore.isLogin && to && to.name === 'Login') { } else if (_isLogin && to && to.name === 'Login') {
// // 如果已经登录且目标页面是登录页面则跳转至首页 // 如果已经登录且目标页面是登录页面则跳转至首页
// router.replaceAll({ name: 'Home' }); router.replaceAll({ name: 'Home' });
// } }
console.log('afterEach', _); console.log('afterEach', to);
}); });
} }

View File

@ -9,7 +9,11 @@ const REFRESH_TOKEN = '/refresh/token';
* @param params * @param params
*/ */
export function login(params: LoginParams) { export function login(params: LoginParams) {
return request.Post<LoginModel>(LOGIN, params); return request.Post<LoginModel>(LOGIN, params, {
meta: {
ignoreAuth: true,
},
});
} }
/** /**

View File

@ -1,68 +0,0 @@
import { defineStore } from 'pinia';
import { useRequest } from 'alova';
import { getCache, removeCache, setCache } from '@/utils/cache';
import { TOKEN_KEY } from '@/enums/cacheEnum';
import { login as loginApi, logout as logoutApi } from '@/services/api/auth';
const authenticationScheme = 'Bearer';
export const useAuthStore = defineStore('AuthStore', () => {
const token = ref<string | null>(null);
const initToken = () => {
token.value = getCache<string>(TOKEN_KEY) || null;
};
function setToken(value: string | null) {
setCache(TOKEN_KEY, value);
token.value = value;
}
const getAuthorization = computed(() => {
return token.value ? `${authenticationScheme} ${token.value}` : '';
});
// 登录
const { send: sendLogin } = useRequest(loginApi, { immediate: false });
const login = async (params: LoginParams) => {
try {
const res = await sendLogin(params);
setToken(res.token);
} catch (error) {
console.log(error);
}
};
// 登出
const { send: sendLogout } = useRequest(logoutApi, { immediate: false });
async function logout() {
try {
await sendLogout();
removeCache(TOKEN_KEY);
token.value = null;
} catch (err: any) {
console.error(err);
}
}
// 刷新token
async function refreshToken() {
try {
// const res = await refreshToken();
// setToken(res.data.access_token);
// return res.data;
} catch (err: any) {
console.error(err);
}
}
return {
token,
initToken,
setToken,
getAuthorization,
login,
logout,
refreshToken,
};
});

View File

@ -1,36 +1,69 @@
import { defineStore } from 'pinia'; import { defineStore } from 'pinia';
import { useRequest } from 'alova'; import { useRequest } from 'alova';
import { useAuthStore } from './auth';
import { getUserInfoApi } from '@/services/api/user'; import { getUserInfoApi } from '@/services/api/user';
import type { UserInfoModel } from '@/services/model/userModel'; import type { UserInfoModel } from '@/services/model/userModel';
import { login as loginApi } from '@/services/api/auth';
import { getToken, isLogin, setToken } from '@/utils/auth';
import { removeCache } from '@/utils/cache';
import { TOKEN_KEY } from '@/enums/cacheEnum';
export const useUserStore = defineStore('UserStore', () => { export const useUserStore = defineStore('UserStore', () => {
const token = ref<string | null>(null);
const userInfo = ref<UserInfoModel | null>(null); const userInfo = ref<UserInfoModel | null>(null);
const authStore = useAuthStore(); // 初始化
function initUserInfo() {
if (isLogin()) {
token.value = getToken();
getUserInfo();
}
}
const { send: _getUserInfo } = useRequest(getUserInfoApi, { initialData: null, immediate: false }); // 是否登录
const loggedIn = computed(() => !!token.value);
// 登录
const { send: sendLogin } = useRequest(loginApi, { immediate: false });
async function login(params: LoginParams) { async function login(params: LoginParams) {
try { try {
await authStore.login(params); const res = await sendLogin(params);
token.value = res.token;
setToken(res.token);
await getUserInfo(); await getUserInfo();
} catch (error) { } catch (error) {
console.log(error); throw error;
} }
} }
// 获取用户信息 // 获取用户信息
const { send: _getUserInfo } = useRequest(getUserInfoApi, { initialData: null, immediate: false });
async function getUserInfo() { async function getUserInfo() {
try { try {
userInfo.value = await _getUserInfo(); userInfo.value = await _getUserInfo();
} catch (error) { } catch (error) {
console.log(error); throw error;
}
}
// 登出
// const { send: sendLogout } = useRequest(logoutApi, { immediate: false });
async function logout() {
try {
// await sendLogout();
removeCache(TOKEN_KEY);
userInfo.value = null;
token.value = null;
} catch (err: any) {
throw err;
} }
} }
return { return {
userInfo, userInfo,
loggedIn,
login, login,
logout,
getUserInfo, getUserInfo,
initUserInfo,
}; };
}); });

View File

@ -1,4 +1,26 @@
import { getCache } from '@/utils/cache'; import { getCache, setCache } from '@/utils/cache';
import { TOKEN_KEY } from '@/enums/cacheEnum'; import { TOKEN_KEY } from '@/enums/cacheEnum';
export const TOKEN = () => getCache<string>(TOKEN_KEY) || undefined; const authenticationScheme = 'Bearer';
export function getToken() {
return getCache<string>(TOKEN_KEY) || null;
}
export function getAuthorization() {
const token = getToken();
return token ? `${authenticationScheme} ${token}` : null;
}
export function setToken(token: string) {
return setCache(TOKEN_KEY, token);
}
export function removeToken() {
return setCache(TOKEN_KEY, null);
}
// 是否登录
export function isLogin() {
return !!getToken();
}

View File

@ -1,3 +1,6 @@
import { random } from 'lodash-es';
import multiavatar from '@multiavatar/multiavatar';
const CHS_RANGE_START = 0x4E00; // 简体中文编码范围开始 const CHS_RANGE_START = 0x4E00; // 简体中文编码范围开始
const CHS_RANGE_END = 0x9FA5; // 简体中文编码范围结束 const CHS_RANGE_END = 0x9FA5; // 简体中文编码范围结束
@ -19,3 +22,13 @@ export function getRandomChsString(length: number) {
} }
return result; return result;
} }
/**
* svg
*/
export function getRandomIcon() {
const svgCode = multiavatar(getRandomChsString(random(16, 32)));
return `data:image/svg+xml;charset=utf-8,${encodeURIComponent(
svgCode,
)}`;
}

View File

@ -70,7 +70,7 @@ export function isUseMock(): boolean {
* @example: * @example:
*/ */
export function getBaseUrl(): string { export function getBaseUrl(): string {
return (isH5 && isDevMode()) ? getEnvValue<string>('VITE_PROXY_PREFIX') : getEnvValue<string>('VITE_BASE_URL'); return (isH5() && isDevMode()) ? getEnvValue<string>('VITE_PROXY_PREFIX') : getEnvValue<string>('VITE_BASE_URL');
} }
/** /**
@ -79,5 +79,5 @@ export function getBaseUrl(): string {
* @example: * @example:
*/ */
export function getUploadUrl(): string { export function getUploadUrl(): string {
return (isH5 && isDevMode()) ? getEnvValue<string>('VITE_UPLOAD_PROXY_PREFIX') : getEnvValue<string>('VITE_UPLOAD_URL'); return (isH5() && isDevMode()) ? getEnvValue<string>('VITE_UPLOAD_PROXY_PREFIX') : getEnvValue<string>('VITE_UPLOAD_URL');
} }

View File

@ -4,13 +4,13 @@ import { assign } from 'lodash-es';
import { checkStatus } from './checkStatus'; import { checkStatus } from './checkStatus';
import { getBaseUrl, isUseMock } from '@/utils/env'; import { getBaseUrl, isUseMock } from '@/utils/env';
import { mockAdapter } from '@/mock'; import { mockAdapter } from '@/mock';
import { useAuthStore } from '@/stores/modules/auth';
import { ContentTypeEnum, ResultEnum } from '@/enums/httpEnum'; import { ContentTypeEnum, ResultEnum } from '@/enums/httpEnum';
import type { API } from '@/services/model/baseModel'; import type { API } from '@/services/model/baseModel';
import { getAuthorization } from '@/utils/auth';
const BASE_URL = getBaseUrl(); const BASE_URL = getBaseUrl();
const HEADER = { const ContentType = {
'Content-Type': ContentTypeEnum.JSON, 'Content-Type': ContentTypeEnum.JSON,
'Accept': 'application/json, text/plain, */*', 'Accept': 'application/json, text/plain, */*',
}; };
@ -29,8 +29,14 @@ const alovaInstance = createAlova({
}), }),
timeout: 5000, timeout: 5000,
beforeRequest: (method) => { beforeRequest: (method) => {
const authStore = useAuthStore(); method.config.headers = assign(method.config.headers, ContentType);
method.config.headers = assign(method.config.headers, HEADER, authStore.getAuthorization); const { config } = method;
const ignoreAuth = !config.meta?.ignoreAuth;
const authorization = ignoreAuth ? getAuthorization() : null;
if (ignoreAuth && !authorization) {
throw new Error('[请求错误]:未登录');
}
method.config.headers.authorization = getAuthorization();
}, },
responded: { responded: {
/** /**
@ -52,16 +58,16 @@ const alovaInstance = createAlova({
return data as any; return data as any;
} }
checkStatus(statusCode, message || ''); checkStatus(statusCode, message || '');
throw new Error(`[请求错误]${message}`); throw new Error(`请求错误[${code}]${message}`);
} }
throw new Error(`[请求错误]${errMsg}`); throw new Error(`请求错误[${statusCode}]${errMsg}`);
}, },
/** /**
* *
*/ */
onError: (err) => { onError: (err) => {
throw new Error(`[请求错误]${err}`); throw new Error(`请求错误:${err}`);
}, },
}, },
}); });

View File

@ -2,14 +2,16 @@
* @description * @description
*/ */
const platform = PLATFORM; export const platform = PLATFORM;
const isH5 = platform === 'h5';
const isApp = platform === 'app';
const isMp = platform.startsWith('mp-');
export { export function isH5() {
platform, return platform === 'h5';
isH5, }
isApp,
isMp, export function isApp() {
}; return platform === 'app';
}
export function isMp() {
return platform.startsWith('mp-');
}

View File

@ -62,6 +62,9 @@ export default defineConfig(async ({ mode }) => {
{ {
'uni-mini-router': ['useRouter', 'useRoute'], 'uni-mini-router': ['useRouter', 'useRoute'],
}, },
{
alova: ['useRequest'],
},
], ],
dts: 'typings/auto-imports.d.ts', dts: 'typings/auto-imports.d.ts',
eslintrc: { eslintrc: {