diff --git a/CHANGELOG.md b/CHANGELOG.md index efb66837..dc414e7a 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,24 @@ # CHANGE LOG +## 4.2.6 + +### Feats + +做了一点大的更新改动,支持小尺寸设备、显示器展示了。但是仅仅是在布局上做了一些兼容! + +补充了一些代码的注释。 + +- RChart 改动 + - 重构了 RChart 组件,现在支持自定义主题色了 + - 支持配置 theme 为 default 属性,则可以启用 echart 默认样式 + - 更新了 echart-themes 包中的说明文件 +- 更改 canceler 方法名为 RequestCanceler +- 更改 app-config 包中的一些属性命名,现在将更加统一命名规则 + +### Fixes + +- 修复 axios request error 状态时不能正确取消拦截器问题 + ## 4.2.5 ### Feats diff --git a/cfg.ts b/cfg.ts index 83526643..c48f1cb3 100644 --- a/cfg.ts +++ b/cfg.ts @@ -51,7 +51,7 @@ const config: AppConfigExport = { /** 配置首屏加载信息 */ preloadingConfig: PRE_LOADING_CONFIG, /** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */ - appPrimaryColor: APP_THEME.APP_PRIMARY_COLOR, + appPrimaryColor: APP_THEME.appPrimaryColor, sideBarLogo: SIDE_BAR_LOGO, /** * diff --git a/package.json b/package.json index cf23a26e..d285601f 100644 --- a/package.json +++ b/package.json @@ -1,7 +1,7 @@ { "name": "ray-template", "private": false, - "version": "4.2.5", + "version": "4.2.6", "type": "module", "engines": { "node": ">=16.0.0", diff --git a/src/app-config/appConfig.ts b/src/app-config/appConfig.ts index 1e0cac46..157d8506 100644 --- a/src/app-config/appConfig.ts +++ b/src/app-config/appConfig.ts @@ -76,21 +76,21 @@ export const SIDE_BAR_LOGO: LayoutSideBarLogo | undefined = { * * 系统菜单折叠配置 * - * MENU_COLLAPSED_WIDTH 配置仅当 MENU_COLLAPSED_MODE 为 width 风格时才有效 + * menuCollapsedWidth 配置仅当 menuCollapsedMode 为 width 风格时才有效 * - * MENU_COLLAPSED_MODE: + * menuCollapsedMode: * - transform: 边栏将只会移动它的位置而不会改变宽度 * - width: Sider 的内容宽度将会被实际改变 - * MENU_COLLAPSED_ICON_SIZE 配置菜单未折叠时图标的大小 - * MENU_COLLAPSED_INDENT 配置菜单每级的缩进 - * MENU_ACCORDION 手风琴模式 + * menuCollapsedIconSize 配置菜单未折叠时图标的大小 + * menuCollapsedIndent 配置菜单每级的缩进 + * menuAccordion 手风琴模式 */ export const APP_MENU_CONFIG: Readonly = { - MENU_COLLAPSED_WIDTH: 64, - MENU_COLLAPSED_MODE: 'width', - MENU_COLLAPSED_ICON_SIZE: 22, - MENU_COLLAPSED_INDENT: 24, - MENU_ACCORDION: false, + menuCollapsedWidth: 64, + menuCollapsedMode: 'width', + menuCollapsedIconSize: 22, + menuCollapsedIndent: 24, + menuAccordion: false, } /** diff --git a/src/app-config/designConfig.ts b/src/app-config/designConfig.ts index 85676825..845ed7b2 100644 --- a/src/app-config/designConfig.ts +++ b/src/app-config/designConfig.ts @@ -19,7 +19,7 @@ export const APP_THEME: AppTheme = { * 系统主题颜色预设色盘 * 支持 RGBA、RGB、十六进制 */ - APP_THEME_COLOR: [ + appThemeColors: [ '#2d8cf0', '#0960bd', '#536dfe', @@ -30,7 +30,7 @@ export const APP_THEME: AppTheme = { '#18A058', ], /** 系统主题色 */ - APP_PRIMARY_COLOR: { + appPrimaryColor: { /** 主题色 */ primaryColor: '#2d8cf0', /** 主题辅助色(用于整体 hover、active 等之类颜色) */ @@ -42,7 +42,7 @@ export const APP_THEME: AppTheme = { * 官网文档地址: * * 注意: - * - APP_PRIMARY_COLOR common 配置优先级大于该配置 + * - appPrimaryColor common 配置优先级大于该配置 * * 如果需要定制化整体组件样式, 配置示例 * 具体自行查看官网, 还有模式更佳丰富的 peers 主题变量配置 @@ -60,7 +60,7 @@ export const APP_THEME: AppTheme = { * } * ``` */ - APP_NAIVE_UI_THEME_OVERRIDES: {}, + appNaiveUIThemeOverrides: {}, /** * * 配置 echart 主题颜色 diff --git a/src/axios/helper/canceler.ts b/src/axios/helper/RequestCanceler.ts similarity index 97% rename from src/axios/helper/canceler.ts rename to src/axios/helper/RequestCanceler.ts index 5651e99d..2e6f33d9 100644 --- a/src/axios/helper/canceler.ts +++ b/src/axios/helper/RequestCanceler.ts @@ -37,7 +37,7 @@ export default class RequestCanceler { * * @remark 将当前请求 config 生成 request key */ - generateRequestKey(config: AppRawRequestConfig): string { + generateRequestKey(config: AppRawRequestConfig) { const { method, url } = config return [ @@ -82,7 +82,6 @@ export default class RequestCanceler { if (this.pendingRequest.has(requestKey)) { this.pendingRequest.get(requestKey)!.abort() - this.pendingRequest.delete(requestKey) } } diff --git a/src/axios/helper/interceptor.ts b/src/axios/helper/interceptor.ts index d4b4532e..a5f967cd 100644 --- a/src/axios/helper/interceptor.ts +++ b/src/axios/helper/interceptor.ts @@ -20,7 +20,7 @@ * 所以在使用的时候, 需要按照约定格式进行参数传递 */ -import RequestCanceler from '@/axios/helper/canceler' +import RequestCanceler from '@/axios/helper/RequestCanceler' import { getAppEnvironment } from '@use-utils/hook' import type { diff --git a/src/axios/inject/request/index.ts b/src/axios/inject/request/index.ts new file mode 100644 index 00000000..150314e9 --- /dev/null +++ b/src/axios/inject/request/index.ts @@ -0,0 +1,35 @@ +/** + * + * @author Ray + * + * @date 2023-10-23 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import { useAxiosInterceptor } from '@/axios/helper/interceptor' +import implement from './provider' + +const { setImplement } = useAxiosInterceptor() + +export const setupRequestInterceptor = () => { + const { implementRequestInterceptorArray } = implement + + setImplement( + 'implementRequestInterceptorArray', + implementRequestInterceptorArray, + 'ok', + ) +} + +export const setupRequestErrorInterceptor = () => { + const { implementRequestInterceptorErrorArray } = implement + + setImplement( + 'implementRequestInterceptorErrorArray', + implementRequestInterceptorErrorArray, + 'error', + ) +} diff --git a/src/axios/inject/request/provide.ts b/src/axios/inject/request/provider.ts similarity index 67% rename from src/axios/inject/request/provide.ts rename to src/axios/inject/request/provider.ts index 1b06c3c9..449af2c5 100644 --- a/src/axios/inject/request/provide.ts +++ b/src/axios/inject/request/provider.ts @@ -16,9 +16,11 @@ * 在内部执行方法中, 已经做了边界处理 * * 提供两个工具方法, 方便类型推导 + * + * 其中 injectRequestCanceler requestErrorCanceler 方法为 axios request interceptor 方法 */ -import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' +import { axiosCanceler } from '@/axios/helper/interceptor' import { appendRequestHeaders } from '@/axios/helper/axiosCopilot' import { APP_CATCH_KEY } from '@/app-config/appConfig' import { getStorage } from '@/utils/cache' @@ -28,8 +30,7 @@ import type { BeforeFetchFunction, FetchErrorFunction, } from '@/axios/type' - -const { setImplement } = useAxiosInterceptor() +import type { Recordable } from '@/types/modules/helper' /** * @@ -66,8 +67,14 @@ const injectRequestHeaders: BeforeFetchFunction = ( ]) } -/** 注入重复请求拦截器 */ -const injectCanceler: BeforeFetchFunction = ( +/** + * + * @param ins 当前请求实例 + * @param mode 当前环境 + * + * 移除请求拦截器与注入请求拦截器 + */ +const injectRequestCanceler: BeforeFetchFunction = ( ins, mode, ) => { @@ -75,9 +82,15 @@ const injectCanceler: BeforeFetchFunction = ( axiosCanceler.addPendingRequest(ins) // 把当前的请求信息添加到 pendingRequest 表中 } -/** 请求发生错误示例 */ -const requestError: FetchErrorFunction = (error, mode) => { - console.log(error, mode) +/** + * + * @param error 请求错误信息 + * @param mode 当前环境 + * + * 请求错误时候,移除请求拦截器 + */ +const requestErrorCanceler: FetchErrorFunction = (error, mode) => { + axiosCanceler.removePendingRequest(error) } /** @@ -85,19 +98,12 @@ const requestError: FetchErrorFunction = (error, mode) => { * 注册请求拦截器 * 请注意执行顺序 */ -export const setupRequestInterceptor = () => { - setImplement( - 'implementRequestInterceptorArray', - [injectRequestHeaders, injectCanceler], - 'ok', - ) -} - -/** - * - * 注册请求错误拦截器 - * 请注意执行顺序 - */ -export const setupRequestErrorInterceptor = () => { - setImplement('implementRequestInterceptorErrorArray', [requestError], 'error') +export default { + // 请求正常 + implementRequestInterceptorArray: [ + injectRequestHeaders, + injectRequestCanceler, + ], + // 请求错误 + implementRequestInterceptorErrorArray: [requestErrorCanceler], } diff --git a/src/axios/inject/response/index.ts b/src/axios/inject/response/index.ts new file mode 100644 index 00000000..ef268787 --- /dev/null +++ b/src/axios/inject/response/index.ts @@ -0,0 +1,35 @@ +/** + * + * @author Ray + * + * @date 2023-10-23 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import { useAxiosInterceptor } from '@/axios/helper/interceptor' +import implement from './provider' + +const { setImplement } = useAxiosInterceptor() + +export const setupResponseInterceptor = () => { + const { implementResponseInterceptorArray } = implement + + setImplement( + 'implementResponseInterceptorArray', + implementResponseInterceptorArray, + 'ok', + ) +} + +export const setupResponseErrorInterceptor = () => { + const { implementResponseInterceptorErrorArray } = implement + + setImplement( + 'implementResponseInterceptorErrorArray', + implementResponseInterceptorErrorArray, + 'error', + ) +} diff --git a/src/axios/inject/response/provide.ts b/src/axios/inject/response/provider.ts similarity index 50% rename from src/axios/inject/response/provide.ts rename to src/axios/inject/response/provider.ts index 34eba431..3fbcb5e1 100644 --- a/src/axios/inject/response/provide.ts +++ b/src/axios/inject/response/provider.ts @@ -16,19 +16,26 @@ * 在内部执行方法中, 已经做了边界处理 * * 提供两个工具方法, 方便类型推导 + * + * 其中 injectResponseCanceler responseErrorCanceler 方法是注入的 axios response interceptor 方法 */ -import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' +import { axiosCanceler } from '@/axios/helper/interceptor' import type { ResponseInterceptorConfig, BeforeFetchFunction, FetchErrorFunction, } from '@/axios/type' +import type { Recordable } from '@/types/modules/helper' -const { setImplement } = useAxiosInterceptor() - -/** 响应成功后移除缓存请求 url */ +/** + * + * @param ins 当前响应实例 + * @param mode 当前环境 + * + * 响应成功后注销请求取消器 + */ const injectResponseCanceler: BeforeFetchFunction = ( ins, mode, @@ -41,13 +48,10 @@ const injectResponseCanceler: BeforeFetchFunction = ( * @param error 错误信息 * @param mode 当前环境 * - * 你可以在响应错误的时候做一些什么 - * 这里不做具体演示 - * - * 方法执行时会有两个参数, 可以根据报错信息与环境定做一些处理 + * 注销失败请求取消器 */ -const responseError: FetchErrorFunction = (error, mode) => { - console.log(error, mode) +const responseErrorCanceler: FetchErrorFunction = (error, mode) => { + axiosCanceler.removePendingRequest(error.config) } /** @@ -55,23 +59,9 @@ const responseError: FetchErrorFunction = (error, mode) => { * 注册响应拦截器 * 请注意执行顺序 */ -export const setupResponseInterceptor = () => { - setImplement( - 'implementResponseInterceptorArray', - [injectResponseCanceler], - 'ok', - ) -} - -/** - * - * 注册响应错误拦截器 - * 请注意执行顺序 - */ -export const setupResponseErrorInterceptor = () => { - setImplement( - 'implementResponseInterceptorErrorArray', - [responseError], - 'error', - ) +export default { + // 响应正常 + implementResponseInterceptorArray: [injectResponseCanceler], + // 响应错误 + implementResponseInterceptorErrorArray: [responseErrorCanceler], } diff --git a/src/axios/instance.ts b/src/axios/instance.ts index e968087a..923d200d 100644 --- a/src/axios/instance.ts +++ b/src/axios/instance.ts @@ -13,20 +13,20 @@ * * 请求拦截器与响应拦截器 * 如果有需要拓展拦截器, 请在 inject 目录下参照示例方法继续拓展 - * 该页面不做改动与配置 + * 该页面不应该做过多的改动与配置 */ import axios from 'axios' import { AXIOS_CONFIG } from '@/app-config/requestConfig' -import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' +import { useAxiosInterceptor } from '@/axios/helper/interceptor' import { setupResponseInterceptor, setupResponseErrorInterceptor, -} from '@/axios/inject/response/provide' +} from '@/axios/inject/response' import { setupRequestInterceptor, setupRequestErrorInterceptor, -} from '@/axios/inject/request/provide' +} from '@/axios/inject/request' import type { AxiosInstanceExpand } from './type' @@ -70,9 +70,6 @@ server.interceptors.response.use( setupResponseErrorInterceptor() fetchError('responseError', error, 'implementResponseInterceptorErrorArray') - // 注销该失败请求的取消器 - axiosCanceler.removePendingRequest(error.config || {}) - return Promise.reject(error) }, ) diff --git a/src/components/RChart/index.tsx b/src/components/RChart/index.tsx index 610ec779..d1d836ab 100644 --- a/src/components/RChart/index.tsx +++ b/src/components/RChart/index.tsx @@ -69,7 +69,7 @@ export default defineComponent({ props, setup(props, { expose }) { const settingStore = useSetting() - const { themeValue } = storeToRefs(settingStore) + const { themeValue: currentTheme } = storeToRefs(settingStore) const rayChartRef = ref() // `echart` 容器实例 const rayChartWrapperRef = ref() const echartInstanceRef = ref() // `echart` 实例 @@ -130,6 +130,30 @@ export default defineComponent({ } } + /** + * + * 更新 chart 主题 + */ + const updateChartTheme = () => { + if (props.theme === 'default') { + props.autoChangeTheme ? renderChart('dark') : renderChart('') + + return + } + + if (!props.theme) { + const theme = props.autoChangeTheme + ? currentTheme.value + ? `${echartTheme}-dark` + : echartTheme + : echartTheme + + renderChart(theme) + } else { + renderChart(props.theme) + } + } + /** * * @returns `chart options` @@ -171,7 +195,7 @@ export default defineComponent({ * 缓存两个实例 * 直接使用响应式代理实例会出现诡异的问题, 例如 `legend` 点击时报错 */ - const renderChart = (theme: ChartTheme = echartTheme) => { + const renderChart = (theme: string = echartTheme) => { /** 获取 dom 容器 */ const element = rayChartRef.value as HTMLElement /** 获取配置项 */ @@ -217,24 +241,6 @@ export default defineComponent({ } } - /** - * - * @param bool 渲染带有主题色的可视化图 - * - * 区别自动跟随模板主题切换与指定主题切换 - */ - const renderThemeChart = (bool?: boolean) => { - if (props.autoChangeTheme) { - bool ? renderChart(`${echartTheme}-dark`) : renderChart() - - return - } - - if (!props.theme) { - renderChart() - } - } - /** * * 销毁 `chart` 实例, 释放资源 @@ -249,7 +255,10 @@ export default defineComponent({ /** 重置 echarts 尺寸 */ const resizeChart = () => { if (echartInstanceRef.value) { - echartInstanceRef.value.resize() + try { + echartInstanceRef.value.resize() + // eslint-disable-next-line no-empty + } catch (e) {} } } @@ -263,12 +272,7 @@ export default defineComponent({ return } - if (props.autoChangeTheme) { - /** 注册 echarts */ - renderThemeChart(themeValue.value) - } else { - props.theme ? renderChart(`${echartTheme}-dark`) : renderChart() - } + updateChartTheme() /** 注册事件 */ if (props.autoResize) { @@ -296,8 +300,8 @@ export default defineComponent({ /** 监听全局主题变化, 然后重新渲染对应主题 echarts */ watch( - () => themeValue.value, - (theme) => { + () => currentTheme.value, + () => { /** * * Q: 为什么需要重新卸载再渲染 @@ -306,8 +310,7 @@ export default defineComponent({ */ if (props.autoChangeTheme) { destroyChart() - - renderThemeChart(theme) + updateChartTheme() } }, ) @@ -322,12 +325,7 @@ export default defineComponent({ () => props.showAria, () => { destroyChart() - - if (props.autoChangeTheme || props.theme) { - themeValue.value ? renderChart(`${echartTheme}-dark`) : renderChart() - } else { - renderChart() - } + updateChartTheme() }, ) @@ -369,13 +367,11 @@ export default defineComponent({ /** 注册 echarts 组件与渲染器 */ await registerChartCore() }) - onMounted(() => { nextTick(() => { mount() }) }) - onBeforeUnmount(() => { unmount() watchCallback?.() diff --git a/src/components/RChart/props.ts b/src/components/RChart/props.ts index def980bb..66cadbef 100644 --- a/src/components/RChart/props.ts +++ b/src/components/RChart/props.ts @@ -47,6 +47,7 @@ const props = { canvasRender: { /** * + * @deprecated * `chart` 渲染器, 默认使用 `canvas` * * 考虑到打包体积与大多数业务场景缘故, 暂时移除 `SVGRenderer` 渲染器的默认导入 @@ -91,8 +92,12 @@ const props = { default: null, }, theme: { - type: [String, Object] as PropType, - default: '', + /** + * + * 手动指定 chart theme + */ + type: String as PropType, + default: null, }, autoChangeTheme: { /** diff --git a/src/components/RChart/type.ts b/src/components/RChart/type.ts index f7866896..931ccc06 100644 --- a/src/components/RChart/type.ts +++ b/src/components/RChart/type.ts @@ -43,7 +43,12 @@ export type AutoResize = height: number } -export type ChartTheme = 'macarons-dark' | string | object | 'macarons' +export type ChartTheme = + | 'macarons-dark' + | 'macarons' + | 'default' + | string + | null export interface RayChartInst { /** diff --git a/src/echart-themes/README.md b/src/echart-themes/README.md index 29b455c3..9143a1e0 100644 --- a/src/echart-themes/README.md +++ b/src/echart-themes/README.md @@ -7,4 +7,14 @@ 1. 配置、选择主题 2. 点击下载主题 3. 选择 json 类型,然后复制 -4. 在 @/echart-themes 包中创建对应的 json 文件,文件名为主题名称 +4. 在 src/echart-themes 包中创建对应的 json 文件,文件名为主题名称 + +## 注意 + +### 一份主题 + +如果有且仅有一份 echart theme,则会视为明暗主题色都共用一套主题色。 + +### 两份主题 + +下载好的主题应该分为:xxx 与 xxx-dark 两份。这样模板会自动根据配置主题色切换明暗主题。 diff --git a/src/hooks/variable/useGlobalVariable.ts b/src/hooks/variable/useGlobalVariable.ts index c110e1ea..e4a9731b 100644 --- a/src/hooks/variable/useGlobalVariable.ts +++ b/src/hooks/variable/useGlobalVariable.ts @@ -20,6 +20,7 @@ /** 全局响应式变量 */ const variableState = reactive({ globalSpinning: false, + globalDrawerValue: false, }) type VariableStateKey = keyof typeof variableState diff --git a/src/hooks/web/index.ts b/src/hooks/web/index.ts index 8e86afc1..7e6a9ca5 100644 --- a/src/hooks/web/index.ts +++ b/src/hooks/web/index.ts @@ -12,5 +12,6 @@ import { useI18n, t } from './useI18n' import { useVueRouter } from '../web/useVueRouter' import { useDayjs } from '../web/useDayjs' +import { useDevice } from './useDevice' -export { useI18n, useVueRouter, useDayjs, t } +export { useI18n, useVueRouter, useDayjs, t, useDevice } diff --git a/src/hooks/web/useDevice.ts b/src/hooks/web/useDevice.ts new file mode 100644 index 00000000..a0df97a4 --- /dev/null +++ b/src/hooks/web/useDevice.ts @@ -0,0 +1,32 @@ +/** + * + * @author Ray + * + * @date 2023-10-24 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** + * + * 检测当前尺寸是否为平板或者更小 + */ + +import { useWindowSize } from '@vueuse/core' + +export function useDevice() { + const { width, height } = useWindowSize() + const isTabletOrSmaller = ref(false) + + watchEffect(() => { + isTabletOrSmaller.value = width.value <= 768 + }) + + return { + width, + height, + isTabletOrSmaller, + } +} diff --git a/src/icons/menu.svg b/src/icons/menu.svg new file mode 100644 index 00000000..cd8ec04d --- /dev/null +++ b/src/icons/menu.svg @@ -0,0 +1,6 @@ + + + \ No newline at end of file diff --git a/src/layout/components/Menu/index.scss b/src/layout/components/Menu/index.scss new file mode 100644 index 00000000..abd0df87 --- /dev/null +++ b/src/layout/components/Menu/index.scss @@ -0,0 +1,7 @@ +.n-drawer.app-menu__drawer { + width: auto !important; + + & .n-layout-sider { + height: 100%; + } +} diff --git a/src/layout/components/Menu/index.tsx b/src/layout/components/Menu/index.tsx index d68c5815..b6ee0298 100644 --- a/src/layout/components/Menu/index.tsx +++ b/src/layout/components/Menu/index.tsx @@ -9,11 +9,15 @@ * @remark 今天也是元气满满撸代码的一天 */ -import { NMenu, NLayoutSider } from 'naive-ui' +import './index.scss' + +import { NMenu, NLayoutSider, NDrawer } from 'naive-ui' import SiderBarLogo from './components/SiderBarLogo/index' import { useMenu } from '@/store' import { APP_MENU_CONFIG } from '@/app-config/appConfig' +import { useDevice } from '@/hooks/web/index' +import { globalVariableToRefs, setVariable } from '@/hooks/variable/index' import type { MenuInst } from 'naive-ui' import type { NaiveMenuOptions } from '@/types/modules/component' @@ -35,11 +39,21 @@ const LayoutMenu = defineComponent({ return menuStore.menuKey }, - // eslint-disable-next-line @typescript-eslint/no-empty-function - set: () => {}, + set: () => { + if (isTabletOrSmaller.value) { + setVariable('globalDrawerValue', false) + } + }, }) const modelMenuOptions = computed(() => menuStore.options) const modelCollapsed = computed(() => menuStore.collapsed) + const { isTabletOrSmaller } = useDevice() + const modelGlobalDrawerValue = computed({ + get: () => globalVariableToRefs('globalDrawerValue').value, + set: (val) => { + setVariable('globalDrawerValue', val) + }, + }) const showMenuOption = () => { const key = modelMenuKey.value as string @@ -49,42 +63,55 @@ const LayoutMenu = defineComponent({ }) } - return { - modelMenuKey, - changeMenuModelValue, - modelMenuOptions, - modelCollapsed, - collapsedMenu, - menuRef, - } - }, - render() { - return ( + const BaseicMenu = () => ( - + { - this.changeMenuModelValue(key, op as unknown as AppMenuOption) + changeMenuModelValue(key, op as unknown as AppMenuOption) }} - accordion={APP_MENU_CONFIG.MENU_ACCORDION} + accordion={APP_MENU_CONFIG.menuAccordion} /> ) + + return { + menuRef, + isTabletOrSmaller, + BaseicMenu, + modelGlobalDrawerValue, + } + }, + render() { + const { isTabletOrSmaller, BaseicMenu } = this + + return !isTabletOrSmaller ? ( + + ) : ( + + + + ) }, }) diff --git a/src/layout/components/SiderBar/components/Breadcrumb/index.tsx b/src/layout/components/SiderBar/components/Breadcrumb/index.tsx index 835a8646..40a17487 100644 --- a/src/layout/components/SiderBar/components/Breadcrumb/index.tsx +++ b/src/layout/components/SiderBar/components/Breadcrumb/index.tsx @@ -21,6 +21,7 @@ import { NDropdown, NBreadcrumb, NBreadcrumbItem } from 'naive-ui' import { useMenu } from '@/store' +import { useDevice } from '@/hooks/web/index' import type { DropdownOption } from 'naive-ui' import type { @@ -29,7 +30,7 @@ import type { AppMenuKey, } from '@/types/modules/app' -const Breadcrumb = defineComponent({ +export default defineComponent({ name: 'RBreadcrumb', setup() { const menuStore = useMenu() @@ -37,15 +38,13 @@ const Breadcrumb = defineComponent({ const { changeMenuModelValue } = menuStore const { breadcrumbOptions } = storeToRefs(menuStore) const modelBreadcrumbOptions = computed(() => breadcrumbOptions.value) + const { isTabletOrSmaller } = useDevice() - const handleDropdownSelect = ( - key: string | number, - option: DropdownOption, - ) => { + const dropdownSelect = (key: string | number, option: DropdownOption) => { changeMenuModelValue(key, option as unknown as AppMenuOption) } - const handleBreadcrumbItemClick = (option: AppMenuOption) => { + const breadcrumbItemClick = (option: AppMenuOption) => { if (!option.children?.length) { const { meta = {} } = option @@ -57,24 +56,29 @@ const Breadcrumb = defineComponent({ return { modelBreadcrumbOptions, - handleDropdownSelect, - handleBreadcrumbItemClick, + dropdownSelect, + breadcrumbItemClick, + isTabletOrSmaller, } }, render() { - return ( + const { isTabletOrSmaller } = this + + return isTabletOrSmaller ? ( +
+ ) : ( {this.modelBreadcrumbOptions.map((curr) => ( 1 ? curr.children : [] } - onSelect={this.handleDropdownSelect.bind(this)} + onSelect={this.dropdownSelect.bind(this)} > {{ default: () => ( @@ -92,5 +96,3 @@ const Breadcrumb = defineComponent({ ) }, }) - -export default Breadcrumb diff --git a/src/layout/components/SiderBar/components/GlobalSeach/index.tsx b/src/layout/components/SiderBar/components/GlobalSeach/index.tsx index 0e7405ed..c51d8a83 100644 --- a/src/layout/components/SiderBar/components/GlobalSeach/index.tsx +++ b/src/layout/components/SiderBar/components/GlobalSeach/index.tsx @@ -18,11 +18,12 @@ import { on, off, queryElements, addClass, removeClass } from '@/utils/element' import { debounce } from 'lodash-es' import { useMenu } from '@/store' import { validMenuItemShow } from '@/router/helper/routerCopilot' +import { useDevice } from '@/hooks/web/index' import type { AppRouteMeta } from '@/router/type' import type { AppMenuOption } from '@/types/modules/app' -const GlobalSeach = defineComponent({ +export default defineComponent({ name: 'GlobalSeach', props: { show: { @@ -76,6 +77,7 @@ const GlobalSeach = defineComponent({ let searchElementIndex = 0 /** 缓存索引 */ let preSearchElementIndex = searchElementIndex + const { isTabletOrSmaller } = useDevice() /** 初始化一些值 */ const resetSearchSomeValue = () => { @@ -240,13 +242,18 @@ const GlobalSeach = defineComponent({ autoFouceSearchItem() } + watchEffect(() => { + if (isTabletOrSmaller.value) { + modelShow.value = false + } + }) + onMounted(() => { on(window, 'keydown', (e: Event) => { registerArouseKeyboard(e as KeyboardEvent) registerChangeSearchElementIndex(e as KeyboardEvent) }) }) - onBeforeUnmount(() => { off(window, 'keydown', (e: Event) => { registerArouseKeyboard(e as KeyboardEvent) @@ -261,10 +268,15 @@ const GlobalSeach = defineComponent({ handleSearchMenuOptions: debounce(handleSearchMenuOptions, 300), handleSearchItemClick, RenderPreIcon, + isTabletOrSmaller, } }, render() { - return ( + const { isTabletOrSmaller } = this + + return isTabletOrSmaller ? ( +
+ ) : (
@@ -339,5 +351,3 @@ const GlobalSeach = defineComponent({ ) }, }) - -export default GlobalSeach diff --git a/src/layout/components/SiderBar/components/SettingDrawer/index.tsx b/src/layout/components/SiderBar/components/SettingDrawer/index.tsx index f68bef75..5a76565b 100644 --- a/src/layout/components/SiderBar/components/SettingDrawer/index.tsx +++ b/src/layout/components/SiderBar/components/SettingDrawer/index.tsx @@ -113,7 +113,7 @@ const SettingDrawer = defineComponent({ {t('headerSettingOptions.ThemeOptions.PrimaryColorConfig')} diff --git a/src/layout/components/SiderBar/index.tsx b/src/layout/components/SiderBar/index.tsx index 77dd6fa7..affd766c 100644 --- a/src/layout/components/SiderBar/index.tsx +++ b/src/layout/components/SiderBar/index.tsx @@ -32,8 +32,10 @@ import { LOCAL_OPTIONS } from '@/app-config/localConfig' import { useAvatarOptions, avatarDropdownClick } from './hook' import { useI18n } from '@/hooks/web/index' import { useFullscreen } from 'vue-hooks-plus' +import { useDevice } from '@/hooks/web/index' +import { globalVariableToRefs, setVariable } from '@/hooks/variable/index' -import type { IconEventMapOptions, IconEventMap } from './type' +import type { LeftIconOptions, IconEventMapOptions, IconEventMap } from './type' const SiderBar = defineComponent({ name: 'SiderBar', @@ -53,55 +55,76 @@ const SiderBar = defineComponent({ display: 'flex', } const globalSearchShown = ref(false) + const { isTabletOrSmaller } = useDevice() + const globalDrawerValue = globalVariableToRefs('globalDrawerValue') /** * * 顶部左边操作栏 */ - const leftIconOptions = computed(() => [ - { - name: 'reload', - size: 18, - tooltip: t('headerTooltip.Reload'), - iconClass: computed(() => - !reloadRouteSwitch.value ? 'ray-icon__reload--loading' : '', - ), - }, - ]) + const leftIconOptions = computed(() => { + const options: LeftIconOptions[] = [ + { + name: 'reload', + size: 18, + tooltip: t('headerTooltip.Reload'), + iconClass: computed(() => + !reloadRouteSwitch.value ? 'ray-icon__reload--loading' : '', + ), + }, + ] + + if (isTabletOrSmaller.value) { + options[0] = { + name: 'menu', + size: 18, + } + } + + return options + }) /** * * 顶部右边提示框操作栏 */ - const rightTooltipIconOptions = computed(() => [ - { - name: 'search', - size: 18, - tooltip: t('headerTooltip.Search'), - eventKey: 'search', - }, - { - name: 'fullscreen', - size: 18, - tooltip: computed(() => - isFullscreen.value - ? t('headerTooltip.CancelFullScreen') - : t('headerTooltip.FullScreen'), - ), - eventKey: 'screen', - }, - { - name: 'github', - size: 18, - tooltip: t('headerTooltip.Github'), - eventKey: 'github', - }, - { - name: 'setting', - size: 18, - tooltip: t('headerTooltip.Setting'), - eventKey: 'setting', - }, - ]) + const rightTooltipIconOptions = computed(() => { + const options = [ + { + name: 'search', + size: 18, + tooltip: t('headerTooltip.Search'), + eventKey: 'search', + }, + { + name: 'fullscreen', + size: 18, + tooltip: computed(() => + isFullscreen.value + ? t('headerTooltip.CancelFullScreen') + : t('headerTooltip.FullScreen'), + ), + eventKey: 'screen', + }, + { + name: 'github', + size: 18, + tooltip: t('headerTooltip.Github'), + eventKey: 'github', + }, + { + name: 'setting', + size: 18, + tooltip: t('headerTooltip.Setting'), + eventKey: 'setting', + }, + ] + + if (isTabletOrSmaller.value) { + options.shift() + } + + return options + }) const iconEventMap: IconEventMapOptions = { // 刷新组件重新加载,手动设置 800ms loading 时长 reload: () => { @@ -124,26 +147,46 @@ const SiderBar = defineComponent({ lock: () => { changeSwitcher(true, 'lockScreenSwitch') }, + menu: () => { + setVariable('globalDrawerValue', !globalDrawerValue.value) + }, } - const handleIconClick = (key: IconEventMap) => { + const toolIconClick = (key: IconEventMap) => { iconEventMap[key]?.() } + const LeftToolIcon = (props: (typeof leftIconOptions.value)[0]) => { + const { iconClass, name, size } = props + + return ( + + ) + } + return { leftIconOptions, rightTooltipIconOptions, t, - handleIconClick, + toolIconClick, showSettings, updateLocale, spaceItemStyle, drawerPlacement, breadcrumbSwitch, globalSearchShown, + LeftToolIcon, } }, render() { + const { LeftToolIcon } = this + return ( @@ -157,26 +200,18 @@ const SiderBar = defineComponent({ wrapItem={false} itemStyle={this.spaceItemStyle} > - {this.leftIconOptions.map((curr) => ( - - {{ - trigger: () => ( - - ), - default: () => curr.tooltip, - }} - - ))} + {this.leftIconOptions.map((curr) => + curr.tooltip ? ( + + {{ + trigger: () => , + default: () => curr.tooltip, + }} + + ) : ( + + ), + )} {this.breadcrumbSwitch ? : null} ))} unknown @@ -20,3 +21,10 @@ export interface IconOptions { eventKey?: string dropdown?: IconDropdownOptions } + +export interface LeftIconOptions { + name: string + size: number + tooltip?: string + iconClass?: ComputedRef +} diff --git a/src/locales/helper.ts b/src/locales/helper.ts index 7f8334f3..56333ffa 100644 --- a/src/locales/helper.ts +++ b/src/locales/helper.ts @@ -126,7 +126,7 @@ export const naiveLocales = (key: string) => { * @remak 未避免出现加载语言错误问题, 故而在 `main.ts` 注册时, 应优先加载 `i18n` 避免出现该问题 */ export const getAppDefaultLanguage = () => { - const language = getStorage( + const language = getStorage( APP_CATCH_KEY.localeLanguage, 'localStorage', SYSTEM_DEFAULT_LOCAL, diff --git a/src/store/modules/menu/helper.ts b/src/store/modules/menu/helper.ts index 252f5b58..f244e0f4 100644 --- a/src/store/modules/menu/helper.ts +++ b/src/store/modules/menu/helper.ts @@ -163,7 +163,7 @@ export const hasMenuIcon = (option: AppMenuOption) => { RIcon, { name: meta!.icon as string, - size: APP_MENU_CONFIG.MENU_COLLAPSED_ICON_SIZE, + size: APP_MENU_CONFIG.menuCollapsedIconSize, cursor: 'pointer', }, {}, diff --git a/src/store/modules/setting/index.ts b/src/store/modules/setting/index.ts index c8842cae..974a7480 100644 --- a/src/store/modules/setting/index.ts +++ b/src/store/modules/setting/index.ts @@ -22,7 +22,7 @@ export const useSetting = defineStore( const settingState = reactive({ drawerPlacement: 'right', primaryColorOverride: { - ...APP_THEME.APP_NAIVE_UI_THEME_OVERRIDES, + ...APP_THEME.appNaiveUIThemeOverrides, common: { primaryColor: primaryColor, // 主题色 primaryColorHover: primaryColor, diff --git a/src/types/modules/appConfig.ts b/src/types/modules/appConfig.ts index d5226619..38fb5548 100644 --- a/src/types/modules/appConfig.ts +++ b/src/types/modules/appConfig.ts @@ -3,11 +3,11 @@ import type { CreateAxiosDefaults } from 'axios' export type CollapsedMode = 'transform' | 'width' export interface AppMenuConfig { - MENU_COLLAPSED_WIDTH: number - MENU_COLLAPSED_MODE: CollapsedMode - MENU_COLLAPSED_ICON_SIZE: number - MENU_COLLAPSED_INDENT: number - MENU_ACCORDION: boolean + menuCollapsedWidth: number + menuCollapsedMode: CollapsedMode + menuCollapsedIconSize: number + menuCollapsedIndent: number + menuAccordion: boolean } export interface AppKeepAlive { diff --git a/src/types/modules/cfg.ts b/src/types/modules/cfg.ts index f1384207..b81c79c3 100644 --- a/src/types/modules/cfg.ts +++ b/src/types/modules/cfg.ts @@ -76,8 +76,8 @@ export interface AppConfig { export type AppConfigExport = Config & UserConfigExport export interface AppTheme { - APP_THEME_COLOR: string[] - APP_PRIMARY_COLOR: AppPrimaryColor - APP_NAIVE_UI_THEME_OVERRIDES: GlobalThemeOverrides + appThemeColors: string[] + appPrimaryColor: AppPrimaryColor + appNaiveUIThemeOverrides: GlobalThemeOverrides echartTheme: string } diff --git a/src/utils/cache.ts b/src/utils/cache.ts index e9cfc13e..0a3c8f80 100644 --- a/src/utils/cache.ts +++ b/src/utils/cache.ts @@ -9,8 +9,6 @@ * @remark 今天也是元气满满撸代码的一天 */ -/** vue3 项目里建议直接用 vueuse useStorage 方法 */ - import type { StorageLike, RemoveStorageKey } from '@/types/modules/utils' /** @@ -40,15 +38,13 @@ function setStorage( } } -/** 重载函数 getStorage */ -function getStorage( +function getStorage( key: string, storageType: StorageLike, defaultValue: T, ): T -/** 重载函数 getStorage */ -function getStorage( +function getStorage( key: string, storageType?: StorageLike, defaultValue?: T, @@ -59,7 +55,7 @@ function getStorage( * @param key 需要获取目标缓存的key * @returns 获取缓存值 */ -function getStorage( +function getStorage( key: string, storageType: StorageLike = 'sessionStorage', defaultValue?: T, diff --git a/src/views/demo/echart/index.tsx b/src/views/demo/echart/index.tsx index 9e157a28..1560c980 100644 --- a/src/views/demo/echart/index.tsx +++ b/src/views/demo/echart/index.tsx @@ -242,22 +242,23 @@ const Echart = defineComponent({
  • -

    当未获取到宽高时,组件会默认以 200*200 尺寸填充。

    +

    1. 当未获取到宽高时,组件会默认以 200*200 尺寸填充。

  • - 默认启用 autoChangeTheme,自动监听模板主题变化(RayTemplate + 2. 默认启用 autoChangeTheme,自动监听模板主题变化,如果设置为 + false 则为 APP_THEME.echartTheme 配置项为渲染结果(RayTemplate 独有)

  • -

    默认启用 watchOptions,自动监听配置项变化

    +

    3. 默认启用 watchOptions,自动监听配置项变化

  • -

    默认启用 animation,强制启用渲染过渡动画

    +

    4. 默认启用 animation,强制启用渲染过渡动画

  • -

    配置 setChartOptions 属性,可以定制化合并模式

    +

    5. 配置 setChartOptions 属性,可以定制化合并模式

@@ -277,11 +278,11 @@ const Echart = defineComponent({ showAria={this.chartAria} />
- 不跟随主题切换的暗色主题可视化图 + 不跟随主题切换的暗色主题可视化图,并且手动指定原始主题色