From 75cd4c5a9ffd524248e67e1a4700400c47fdf286 Mon Sep 17 00:00:00 2001 From: ray_wuhao <443547225@qq.com> Date: Tue, 6 Jun 2023 15:39:05 +0800 Subject: [PATCH] =?UTF-8?q?v3.3.1=20=E7=BB=86=E8=8A=82=E8=A1=A5=E5=85=85?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .gitignore | 2 + CHANGELOG.md | 29 ++++++ components.d.ts | 1 - package.json | 2 +- src/axios/helper/axiosCopilot.ts | 33 +++++++ src/axios/helper/interceptor.ts | 93 +++++++++++++++---- src/axios/inject/requestInject.ts | 48 ++++++++++ src/axios/inject/responseInject.ts | 32 +++++++ src/axios/instance.ts | 45 ++++----- src/axios/type.ts | 46 +++++---- .../AppComponents/AppLockScreen/index.scss | 4 + .../AppComponents/AppLockScreen/index.tsx | 40 +++++--- .../src/components/TableSize/index.scss | 12 ++- .../src/components/TableSize/index.tsx | 5 +- .../Components/GlobalSeach/index.scss | 32 ++++--- .../SiderBar/Components/GlobalSeach/index.tsx | 2 +- src/styles/mixins.scss | 7 ++ src/types/utils.d.ts | 5 +- src/utils/cache.ts | 13 +++ 19 files changed, 347 insertions(+), 104 deletions(-) create mode 100644 src/axios/helper/axiosCopilot.ts create mode 100644 src/axios/inject/requestInject.ts create mode 100644 src/axios/inject/responseInject.ts diff --git a/.gitignore b/.gitignore index ec7bcf40..dbf0d6ab 100644 --- a/.gitignore +++ b/.gitignore @@ -14,6 +14,8 @@ dist dist/ *.local visualizer.* +components.d.ts +auto-imports.d.ts # Editor directories and files .idea diff --git a/CHANGELOG.md b/CHANGELOG.md index c5be564f..5ea80c3c 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,34 @@ # CHANGE LOG +## 3.3.1 + +### Feats + +- 新增 useAppTheme sass 方法 + +```ts +useAppTheme key 类型: 'dark' | 'light' +``` + +```scss +// 暗色主题 +.demo--dark { + @include useAppTheme('dark') { + color: #ffffff; + } +} +// 明亮主题 +.demo--light { + @include useAppTheme('light') { + color: #000000; + } +} +``` + +- 一些细节优化 +- axios 拦截器与 axios instance 进行独立(现在不再 instance.ts 文件中编写拦截器相关逻辑),拦截器逻辑放在 inject 包中 +- 一些 bug 修复 + ## 3.3.0 ### 特征 diff --git a/components.d.ts b/components.d.ts index 11f5125e..903cc398 100644 --- a/components.d.ts +++ b/components.d.ts @@ -7,7 +7,6 @@ export {} declare module '@vue/runtime-core' { export interface GlobalComponents { - RayTransitionComponent: typeof import('./src/components/RayTransitionComponent/TransitionComponent.vue')['default'] RouterLink: typeof import('vue-router')['RouterLink'] RouterView: typeof import('vue-router')['RouterView'] TransitionComponent: typeof import('./src/components/RayTransitionComponent/TransitionComponent.vue')['default'] diff --git a/package.json b/package.json index c19d6ae3..8ca1a156 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "ray-template", "private": false, - "version": "3.3.1", + "version": "3.3.0", "type": "module", "scripts": { "dev": "vite", diff --git a/src/axios/helper/axiosCopilot.ts b/src/axios/helper/axiosCopilot.ts new file mode 100644 index 00000000..fae3a3a8 --- /dev/null +++ b/src/axios/helper/axiosCopilot.ts @@ -0,0 +1,33 @@ +/** + * + * @author Ray + * + * @date 2023-06-02 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** axios 拦截器工具 */ + +import type { RawAxiosRequestHeaders, AxiosRequestConfig } from 'axios' +import type { RequestHeaderOptions } from '../type' + +/** + * + * @param instance axios instance + * @param options axios headers options + * + * @remark 自定义 `axios` 请求头配置 + */ +export const appendRequestHeaders = ( + instance: AxiosRequestConfig, + options: RequestHeaderOptions[], +) => { + const requestHeaders = instance.headers as RawAxiosRequestHeaders + + options.forEach((curr) => { + requestHeaders[curr.key] = curr.value + }) +} diff --git a/src/axios/helper/interceptor.ts b/src/axios/helper/interceptor.ts index fae3a3a8..cd3a7ab2 100644 --- a/src/axios/helper/interceptor.ts +++ b/src/axios/helper/interceptor.ts @@ -2,32 +2,91 @@ * * @author Ray * - * @date 2023-06-02 + * @date 2023-06-05 * * @workspace ray-template * * @remark 今天也是元气满满撸代码的一天 */ -/** axios 拦截器工具 */ - -import type { RawAxiosRequestHeaders, AxiosRequestConfig } from 'axios' -import type { RequestHeaderOptions } from '../type' - /** * - * @param instance axios instance - * @param options axios headers options + * axios 拦截器注入 * - * @remark 自定义 `axios` 请求头配置 + * 请求拦截器、响应拦截器 + * 暴露启动方法调用所有已注册方法 + * + * 为什么要把 create 方法拆成两个, emmmmmm 我也不知道 */ -export const appendRequestHeaders = ( - instance: AxiosRequestConfig, - options: RequestHeaderOptions[], -) => { - const requestHeaders = instance.headers as RawAxiosRequestHeaders - options.forEach((curr) => { - requestHeaders[curr.key] = curr.value - }) +import RequestCanceler from '@/axios/helper/canceler' + +import type { + RequestInterceptorConfig, + ResponseInterceptorConfig, + ImplementKey, + ImplementQueue, +} from '@/axios/type' + +const axiosFetchInstance = { + requestInstance: null as RequestInterceptorConfig | null, + responseInstance: null as ResponseInterceptorConfig | null, +} +const implement: ImplementQueue = { + implementRequestInterceptorArray: [], + implementResponseInterceptorArray: [], +} +export const axiosCanceler = new RequestCanceler() + +export const useAxiosInterceptor = () => { + const getImplementKey = (key: ImplementKey) => { + return key === 'requestInstance' + ? 'implementRequestInterceptorArray' + : 'implementResponseInterceptorArray' + } + + const createRequestAxiosInstance = (instance: RequestInterceptorConfig) => { + axiosFetchInstance['requestInstance'] = instance + } + + const createResponseAxiosInstance = (instance: ResponseInterceptorConfig) => { + axiosFetchInstance['responseInstance'] = instance + } + + /** 请求前, 执行队列所有方法 */ + const beforeAxiosFetch = (key: ImplementKey) => { + const funcArr = implement[getImplementKey(key)] + + funcArr?.forEach((curr) => { + if (typeof curr === 'function') { + curr() + } + }) + } + + /** 设置拦截器队列 */ + const setImplementQueue = (func: AnyFunc[], key: ImplementKey) => { + if (func && key) { + implement[getImplementKey(key)] = func + } + } + + /** 获取拦截器队列 */ + const getImplementQueue = (key: ImplementKey) => { + return implement[getImplementKey(key)] + } + + /** 获取请求实例或者响应实例 */ + const getAxiosFetchInstance = (key: ImplementKey) => { + return axiosFetchInstance[key] + } + + return { + createRequestAxiosInstance, + createResponseAxiosInstance, + beforeAxiosFetch, + setImplementQueue, + getImplementQueue, + getAxiosFetchInstance, + } } diff --git a/src/axios/inject/requestInject.ts b/src/axios/inject/requestInject.ts new file mode 100644 index 00000000..51482602 --- /dev/null +++ b/src/axios/inject/requestInject.ts @@ -0,0 +1,48 @@ +/** + * + * @author Ray + * + * @date 2023-06-06 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** 请求拦截器入口 */ + +import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' +import { appendRequestHeaders } from '@/axios/helper/axiosCopilot' + +import type { RequestInterceptorConfig } from '@/axios/type' + +const { getAxiosFetchInstance, setImplementQueue } = useAxiosInterceptor() + +/** 注入请求头信息 */ +const injectRequestHeaders = () => { + const request = getAxiosFetchInstance( + 'requestInstance', + ) as RequestInterceptorConfig + + appendRequestHeaders(request, [ + { + key: 'X-TOKEN', + value: 'token', + }, + ]) +} + +/** 注入重复请求拦截器 */ +const injectCanceler = () => { + const request = getAxiosFetchInstance( + 'requestInstance', + ) as RequestInterceptorConfig + + axiosCanceler.removePendingRequest(request) // 检查是否存在重复请求, 若存在则取消已发的请求 + axiosCanceler.addPendingRequest(request) // 把当前的请求信息添加到 pendingRequest 表中 +} + +/** 注册请求拦截器 */ +export const setupRequestInterceptor = () => { + setImplementQueue([injectRequestHeaders, injectCanceler], 'requestInstance') +} diff --git a/src/axios/inject/responseInject.ts b/src/axios/inject/responseInject.ts new file mode 100644 index 00000000..41e20341 --- /dev/null +++ b/src/axios/inject/responseInject.ts @@ -0,0 +1,32 @@ +/** + * + * @author Ray + * + * @date 2023-06-06 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** 响应拦截器入口 */ + +import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' + +import type { ResponseInterceptorConfig } from '@/axios/type' + +const { getAxiosFetchInstance, setImplementQueue } = useAxiosInterceptor() + +/** 响应成功后移除缓存请求 url */ +const injectResponseCanceler = () => { + const response = getAxiosFetchInstance( + 'responseInstance', + ) as ResponseInterceptorConfig + + axiosCanceler.removePendingRequest(response.config) +} + +/** 注册响应拦截器 */ +export const setupResponseInterceptor = () => { + setImplementQueue([injectResponseCanceler], 'responseInstance') +} diff --git a/src/axios/instance.ts b/src/axios/instance.ts index b4621c79..f4b32b60 100644 --- a/src/axios/instance.ts +++ b/src/axios/instance.ts @@ -12,43 +12,30 @@ /** * * 请求, 响应拦截器 - * - * 可在此实现共享的基础配置 + * 如果有需要拓展拦截器, 请在 inject 目录下参照示例方法继续拓展 + * 该页面不做改动与配置 */ import axios from 'axios' -import { getDetermineEnv } from '@use-utils/hook' -import RequestCanceler from './helper/canceler' import { AXIOS_CONFIG } from '@/appConfig/requestConfig' -import { appendRequestHeaders } from './helper/interceptor' +import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' +import { setupRequestInterceptor } from '@/axios/inject/requestInject' +import { setupResponseInterceptor } from '@/axios/inject/responseInject' import type { AxiosInstanceExpand } from './type' -const canceler = new RequestCanceler() - const server: AxiosInstanceExpand = axios.create(AXIOS_CONFIG) +const { + createRequestAxiosInstance, + createResponseAxiosInstance, + beforeAxiosFetch, +} = useAxiosInterceptor() server.interceptors.request.use( (request) => { - const { MODE } = getDetermineEnv() - - if (MODE === 'development') { - // TODO: 开发环境 - } else if (MODE === 'production') { - // TODO: 生产环境 - } else if (MODE === 'test') { - // TODO: 测试环境 - } - - appendRequestHeaders(request, [ - { - key: 'X-TOKEN', - value: 'token', - }, - ]) // 自定义请求头 - - canceler.removePendingRequest(request) // 检查是否存在重复请求, 若存在则取消已发的请求 - canceler.addPendingRequest(request) // 把当前的请求信息添加到 pendingRequest 表中 + createRequestAxiosInstance(request) + setupRequestInterceptor() + beforeAxiosFetch('requestInstance') return request }, @@ -59,14 +46,16 @@ server.interceptors.request.use( server.interceptors.response.use( (response) => { - canceler.removePendingRequest(response.config) + createResponseAxiosInstance(response) + setupResponseInterceptor() + beforeAxiosFetch('responseInstance') const { data } = response return Promise.resolve(data) }, (error) => { - canceler.removePendingRequest(error.config || {}) + axiosCanceler.removePendingRequest(error.config || {}) return Promise.reject(error) }, diff --git a/src/axios/type.ts b/src/axios/type.ts index 22714fd4..a3d780b0 100644 --- a/src/axios/type.ts +++ b/src/axios/type.ts @@ -1,9 +1,12 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ import type { AxiosHeaders, AxiosRequestConfig, HeadersDefaults, AxiosDefaults, Axios, + InternalAxiosRequestConfig, + AxiosResponse, } from 'axios' export type AxiosHeaderValue = @@ -20,56 +23,50 @@ export interface RequestHeaderOptions { } export interface AxiosInstanceExpand extends Axios { - (config: AxiosRequestConfig): Promise - ( - url: string, - config?: AxiosRequestConfig, - ): Promise + (config: AxiosRequestConfig): Promise + (url: string, config?: AxiosRequestConfig): Promise getUri(config?: AxiosRequestConfig): string - request(config: AxiosRequestConfig): Promise - get( + request(config: AxiosRequestConfig): Promise + get(url: string, config?: AxiosRequestConfig): Promise + delete( url: string, config?: AxiosRequestConfig, ): Promise - delete( + head( url: string, config?: AxiosRequestConfig, ): Promise - head( + options( url: string, config?: AxiosRequestConfig, ): Promise - options( - url: string, - config?: AxiosRequestConfig, - ): Promise - post( + post( url: string, data?: D, config?: AxiosRequestConfig, ): Promise - put( + put( url: string, data?: D, config?: AxiosRequestConfig, ): Promise - patch( + patch( url: string, data?: D, config?: AxiosRequestConfig, ): Promise - postForm( + postForm( url: string, data?: D, config?: AxiosRequestConfig, ): Promise - putForm( + putForm( url: string, data?: D, config?: AxiosRequestConfig, ): Promise - patchForm( + patchForm( url: string, data?: D, config?: AxiosRequestConfig, @@ -81,3 +78,14 @@ export interface AxiosInstanceExpand extends Axios { } } } + +export type RequestInterceptorConfig = InternalAxiosRequestConfig + +export type ResponseInterceptorConfig = AxiosResponse + +export type ImplementKey = 'requestInstance' | 'responseInstance' + +export interface ImplementQueue { + implementRequestInterceptorArray: AnyFunc[] + implementResponseInterceptorArray: AnyFunc[] +} diff --git a/src/components/AppComponents/AppLockScreen/index.scss b/src/components/AppComponents/AppLockScreen/index.scss index 468c3fea..e4bff73d 100644 --- a/src/components/AppComponents/AppLockScreen/index.scss +++ b/src/components/AppComponents/AppLockScreen/index.scss @@ -58,6 +58,10 @@ & .current-date span { font-size: 1.5rem; } + + // & .current-year span { + // font-size: 0.75rem; + // } } } } diff --git a/src/components/AppComponents/AppLockScreen/index.tsx b/src/components/AppComponents/AppLockScreen/index.tsx index 1220e6f5..0bcd70ff 100644 --- a/src/components/AppComponents/AppLockScreen/index.tsx +++ b/src/components/AppComponents/AppLockScreen/index.tsx @@ -9,6 +9,16 @@ * @remark 今天也是元气满满撸代码的一天 */ +/** + * + * 这里没有做解锁密码校验, 只要符合校验规则值皆可 + * 可以根据需求自行更改 + * + * @deprecated + * 后期该组件会进行破坏性更新, 请注意版本的更新 + * 会将该组件的锁屏、解锁操作拆分, 使其更加合理 + */ + import './index.scss' import { NModal, NInput, NForm, NFormItem, NButton, NSpace } from 'naive-ui' @@ -35,19 +45,19 @@ const LockScreen = defineComponent({ const { changeSwitcher } = settingStore const { logout } = signinStore - const TIME_FORMAT = 'HH:mm' + const HH_MM_FORMAT = 'HH:mm' const AM_PM_FORMAT = 'A' - const YEAR_FORMAT = 'YY-MM-DD' - const DATE_FORMAT = 'dddd' + const YY_MM_DD_FORMAT = 'YY年MM月DD日' + const DDD_FORMAT = 'ddd' const state = reactive({ lockCondition: { pwd: null, }, - time: dayjs().format(TIME_FORMAT), - second: dayjs().locale('en').format(AM_PM_FORMAT), - year: dayjs().format(YEAR_FORMAT), - date: dayjs().format(DATE_FORMAT), + HH_MM: dayjs().format(HH_MM_FORMAT), + AM_PM: dayjs().locale('en').format(AM_PM_FORMAT), + YY_MM_DD: dayjs().format(YY_MM_DD_FORMAT), + DDD: dayjs().format(DDD_FORMAT), }) const rules = { pwd: { @@ -77,12 +87,12 @@ const LockScreen = defineComponent({ } const dayInterval = setInterval(() => { - state.time = dayjs().format(TIME_FORMAT) - state.second = dayjs().format(AM_PM_FORMAT) + state.HH_MM = dayjs().format(HH_MM_FORMAT) + state.AM_PM = dayjs().format(AM_PM_FORMAT) }, 60_000) const yearInterval = setInterval(() => { - state.year = dayjs().format(YEAR_FORMAT) - state.date = dayjs().format(DATE_FORMAT) + state.YY_MM_DD = dayjs().format(YY_MM_DD_FORMAT) + state.DDD = dayjs().format(DDD_FORMAT) }, 86_400_000) const handleBackToSignin = () => { @@ -190,8 +200,8 @@ const LockScreen = defineComponent({
-
{this.time?.split(':')[0]}
-
{this.time?.split(':')[1]}
+
{this.HH_MM?.split(':')[0]}
+
{this.HH_MM?.split(':')[1]}
@@ -233,10 +243,10 @@ const LockScreen = defineComponent({
diff --git a/src/components/RayTable/src/components/TableSize/index.scss b/src/components/RayTable/src/components/TableSize/index.scss index 1c07c8c5..4fcbd871 100644 --- a/src/components/RayTable/src/components/TableSize/index.scss +++ b/src/components/RayTable/src/components/TableSize/index.scss @@ -35,11 +35,13 @@ } } -.ray-template--dark { - & .table-size__dropdown-wrapper { - & .dropdown-item:hover { - background-color: var(--ray-theme-primary-fade-color); - color: var(--ray-theme-primary-color); +.ray-table__table-size--dark { + @include useAppTheme("dark") { + & .table-size__dropdown-wrapper { + & .dropdown-item:hover { + background-color: var(--ray-theme-primary-fade-color); + color: var(--ray-theme-primary-color); + } } } } diff --git a/src/components/RayTable/src/components/TableSize/index.tsx b/src/components/RayTable/src/components/TableSize/index.tsx index cfa373f8..4d23e8ea 100644 --- a/src/components/RayTable/src/components/TableSize/index.tsx +++ b/src/components/RayTable/src/components/TableSize/index.tsx @@ -92,7 +92,10 @@ const TableSize = defineComponent({ ), default: () => ( - +
{this.sizeOptions.map((curr) => ( diff --git a/src/layout/components/SiderBar/Components/GlobalSeach/index.scss b/src/layout/components/SiderBar/Components/GlobalSeach/index.scss index 24e7dede..cb994426 100644 --- a/src/layout/components/SiderBar/Components/GlobalSeach/index.scss +++ b/src/layout/components/SiderBar/Components/GlobalSeach/index.scss @@ -67,29 +67,33 @@ $globalSearchWidth: 650px; } } -.ray-template--dark { - & .global-seach__card { - background-color: #242424; +.global-seach--dark { + @include useAppTheme("dark") { + & .global-seach__card { + background-color: #242424; - & .global-seach__card-content .content-item { - background-color: #2f2f2f; + & .global-seach__card-content .content-item { + background-color: #2f2f2f; - &:hover { - background-color: var(--ray-theme-primary-fade-color); + &:hover { + background-color: var(--ray-theme-primary-fade-color); + } } } } } -.ray-template--light { - & .global-seach__card { - background-color: #f9f9f9; +.global-seach--light { + @include useAppTheme("light") { + & .global-seach__card { + background-color: #f9f9f9; - & .global-seach__card-content .content-item { - background-color: #ffffff; + & .global-seach__card-content .content-item { + background-color: #ffffff; - &:hover { - background-color: var(--ray-theme-primary-fade-color); + &:hover { + background-color: var(--ray-theme-primary-fade-color); + } } } } diff --git a/src/layout/components/SiderBar/Components/GlobalSeach/index.tsx b/src/layout/components/SiderBar/Components/GlobalSeach/index.tsx index cba4248e..4441bf9c 100644 --- a/src/layout/components/SiderBar/Components/GlobalSeach/index.tsx +++ b/src/layout/components/SiderBar/Components/GlobalSeach/index.tsx @@ -139,7 +139,7 @@ const GlobalSeach = defineComponent({ render() { return ( -
+
diff --git a/src/styles/mixins.scss b/src/styles/mixins.scss index 5e20ba15..28aeffd1 100644 --- a/src/styles/mixins.scss +++ b/src/styles/mixins.scss @@ -40,3 +40,10 @@ cursor: pointer; } } + +// 根据主题切换样式 +@mixin useAppTheme($theme) { + body[class="ray-template--#{$theme}"] & { + @content; + } +} diff --git a/src/types/utils.d.ts b/src/types/utils.d.ts index 4193149d..074fc41b 100644 --- a/src/types/utils.d.ts +++ b/src/types/utils.d.ts @@ -1,3 +1,4 @@ +/* eslint-disable @typescript-eslint/no-explicit-any */ export {} import type CryptoJS from 'crypto-js' @@ -32,7 +33,7 @@ export global { declare type CipherParams = CryptoJS.lib.CipherParams - declare type VoidFunc = (...args: unknown[]) => void + declare type VoidFunc = (...args: any[]) => void - declare type AnyFunc = (...args: unknown[]) => unknown + declare type AnyFunc = (...args: any[]) => any } diff --git a/src/utils/cache.ts b/src/utils/cache.ts index dada6182..4dd34cdf 100644 --- a/src/utils/cache.ts +++ b/src/utils/cache.ts @@ -1,3 +1,16 @@ +/** + * + * @author Ray + * + * @date 2023-06-05 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** vue3 项目里建议直接用 vueuse useStorage 方法 */ + /** * * @param key 需要设置的key