This commit is contained in:
XiaoDaiGua-Ray 2023-10-24 17:42:01 +08:00
parent 9e3f199d21
commit b05edfeaeb
34 changed files with 480 additions and 257 deletions

View File

@ -1,5 +1,24 @@
# CHANGE LOG # CHANGE LOG
## 4.2.6
### Feats
做了一点大的更新改动,支持小尺寸设备、显示器展示了。但是仅仅是在布局上做了一些兼容!
补充了一些代码的注释。
- RChart 改动
- 重构了 RChart 组件,现在支持自定义主题色了
- 支持配置 theme 为 default 属性,则可以启用 echart 默认样式
- 更新了 echart-themes 包中的说明文件
- 更改 canceler 方法名为 RequestCanceler
- 更改 app-config 包中的一些属性命名,现在将更加统一命名规则
### Fixes
- 修复 axios request error 状态时不能正确取消拦截器问题
## 4.2.5 ## 4.2.5
### Feats ### Feats

2
cfg.ts
View File

@ -51,7 +51,7 @@ const config: AppConfigExport = {
/** 配置首屏加载信息 */ /** 配置首屏加载信息 */
preloadingConfig: PRE_LOADING_CONFIG, preloadingConfig: PRE_LOADING_CONFIG,
/** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */ /** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */
appPrimaryColor: APP_THEME.APP_PRIMARY_COLOR, appPrimaryColor: APP_THEME.appPrimaryColor,
sideBarLogo: SIDE_BAR_LOGO, sideBarLogo: SIDE_BAR_LOGO,
/** /**
* *

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": false, "private": false,
"version": "4.2.5", "version": "4.2.6",
"type": "module", "type": "module",
"engines": { "engines": {
"node": ">=16.0.0", "node": ">=16.0.0",

View File

@ -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: 边栏将只会移动它的位置而不会改变宽度 * - transform: 边栏将只会移动它的位置而不会改变宽度
* - width: Sider * - width: Sider
* MENU_COLLAPSED_ICON_SIZE * menuCollapsedIconSize
* MENU_COLLAPSED_INDENT * menuCollapsedIndent
* MENU_ACCORDION * menuAccordion
*/ */
export const APP_MENU_CONFIG: Readonly<AppMenuConfig> = { export const APP_MENU_CONFIG: Readonly<AppMenuConfig> = {
MENU_COLLAPSED_WIDTH: 64, menuCollapsedWidth: 64,
MENU_COLLAPSED_MODE: 'width', menuCollapsedMode: 'width',
MENU_COLLAPSED_ICON_SIZE: 22, menuCollapsedIconSize: 22,
MENU_COLLAPSED_INDENT: 24, menuCollapsedIndent: 24,
MENU_ACCORDION: false, menuAccordion: false,
} }
/** /**

View File

@ -19,7 +19,7 @@ export const APP_THEME: AppTheme = {
* *
* RGBARGB * RGBARGB
*/ */
APP_THEME_COLOR: [ appThemeColors: [
'#2d8cf0', '#2d8cf0',
'#0960bd', '#0960bd',
'#536dfe', '#536dfe',
@ -30,7 +30,7 @@ export const APP_THEME: AppTheme = {
'#18A058', '#18A058',
], ],
/** 系统主题色 */ /** 系统主题色 */
APP_PRIMARY_COLOR: { appPrimaryColor: {
/** 主题色 */ /** 主题色 */
primaryColor: '#2d8cf0', primaryColor: '#2d8cf0',
/** 主题辅助色(用于整体 hover、active 等之类颜色) */ /** 主题辅助色(用于整体 hover、active 等之类颜色) */
@ -42,7 +42,7 @@ export const APP_THEME: AppTheme = {
* : <https://www.naiveui.com/zh-CN/dark/docs/customize-theme> * : <https://www.naiveui.com/zh-CN/dark/docs/customize-theme>
* *
* : * :
* - APP_PRIMARY_COLOR common * - appPrimaryColor common
* *
* , * ,
* , peers * , peers
@ -60,7 +60,7 @@ export const APP_THEME: AppTheme = {
* } * }
* ``` * ```
*/ */
APP_NAIVE_UI_THEME_OVERRIDES: {}, appNaiveUIThemeOverrides: {},
/** /**
* *
* echart * echart

View File

@ -37,7 +37,7 @@ export default class RequestCanceler {
* *
* @remark config request key * @remark config request key
*/ */
generateRequestKey(config: AppRawRequestConfig): string { generateRequestKey(config: AppRawRequestConfig) {
const { method, url } = config const { method, url } = config
return [ return [
@ -82,7 +82,6 @@ export default class RequestCanceler {
if (this.pendingRequest.has(requestKey)) { if (this.pendingRequest.has(requestKey)) {
this.pendingRequest.get(requestKey)!.abort() this.pendingRequest.get(requestKey)!.abort()
this.pendingRequest.delete(requestKey) this.pendingRequest.delete(requestKey)
} }
} }

View File

@ -20,7 +20,7 @@
* 使, * 使,
*/ */
import RequestCanceler from '@/axios/helper/canceler' import RequestCanceler from '@/axios/helper/RequestCanceler'
import { getAppEnvironment } from '@use-utils/hook' import { getAppEnvironment } from '@use-utils/hook'
import type { import type {

View File

@ -0,0 +1,35 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-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',
)
}

View File

@ -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 { appendRequestHeaders } from '@/axios/helper/axiosCopilot'
import { APP_CATCH_KEY } from '@/app-config/appConfig' import { APP_CATCH_KEY } from '@/app-config/appConfig'
import { getStorage } from '@/utils/cache' import { getStorage } from '@/utils/cache'
@ -28,8 +30,7 @@ import type {
BeforeFetchFunction, BeforeFetchFunction,
FetchErrorFunction, FetchErrorFunction,
} from '@/axios/type' } from '@/axios/type'
import type { Recordable } from '@/types/modules/helper'
const { setImplement } = useAxiosInterceptor()
/** /**
* *
@ -66,8 +67,14 @@ const injectRequestHeaders: BeforeFetchFunction<RequestInterceptorConfig> = (
]) ])
} }
/** 注入重复请求拦截器 */ /**
const injectCanceler: BeforeFetchFunction<RequestInterceptorConfig> = ( *
* @param ins
* @param mode
*
*
*/
const injectRequestCanceler: BeforeFetchFunction<RequestInterceptorConfig> = (
ins, ins,
mode, mode,
) => { ) => {
@ -75,9 +82,15 @@ const injectCanceler: BeforeFetchFunction<RequestInterceptorConfig> = (
axiosCanceler.addPendingRequest(ins) // 把当前的请求信息添加到 pendingRequest 表中 axiosCanceler.addPendingRequest(ins) // 把当前的请求信息添加到 pendingRequest 表中
} }
/** 请求发生错误示例 */ /**
const requestError: FetchErrorFunction<unknown> = (error, mode) => { *
console.log(error, mode) * @param error
* @param mode
*
*
*/
const requestErrorCanceler: FetchErrorFunction<Recordable> = (error, mode) => {
axiosCanceler.removePendingRequest(error)
} }
/** /**
@ -85,19 +98,12 @@ const requestError: FetchErrorFunction<unknown> = (error, mode) => {
* *
* *
*/ */
export const setupRequestInterceptor = () => { export default {
setImplement( // 请求正常
'implementRequestInterceptorArray', implementRequestInterceptorArray: [
[injectRequestHeaders, injectCanceler], injectRequestHeaders,
'ok', injectRequestCanceler,
) ],
} // 请求错误
implementRequestInterceptorErrorArray: [requestErrorCanceler],
/**
*
*
*
*/
export const setupRequestErrorInterceptor = () => {
setImplement('implementRequestInterceptorErrorArray', [requestError], 'error')
} }

View File

@ -0,0 +1,35 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-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',
)
}

View File

@ -16,19 +16,26 @@
* , * ,
* *
* , 便 * , 便
*
* injectResponseCanceler responseErrorCanceler axios response interceptor
*/ */
import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' import { axiosCanceler } from '@/axios/helper/interceptor'
import type { import type {
ResponseInterceptorConfig, ResponseInterceptorConfig,
BeforeFetchFunction, BeforeFetchFunction,
FetchErrorFunction, FetchErrorFunction,
} from '@/axios/type' } from '@/axios/type'
import type { Recordable } from '@/types/modules/helper'
const { setImplement } = useAxiosInterceptor() /**
*
/** 响应成功后移除缓存请求 url */ * @param ins
* @param mode
*
*
*/
const injectResponseCanceler: BeforeFetchFunction<ResponseInterceptorConfig> = ( const injectResponseCanceler: BeforeFetchFunction<ResponseInterceptorConfig> = (
ins, ins,
mode, mode,
@ -41,13 +48,10 @@ const injectResponseCanceler: BeforeFetchFunction<ResponseInterceptorConfig> = (
* @param error * @param error
* @param mode * @param mode
* *
* *
*
*
* ,
*/ */
const responseError: FetchErrorFunction<unknown> = (error, mode) => { const responseErrorCanceler: FetchErrorFunction<Recordable> = (error, mode) => {
console.log(error, mode) axiosCanceler.removePendingRequest(error.config)
} }
/** /**
@ -55,23 +59,9 @@ const responseError: FetchErrorFunction<unknown> = (error, mode) => {
* *
* *
*/ */
export const setupResponseInterceptor = () => { export default {
setImplement( // 响应正常
'implementResponseInterceptorArray', implementResponseInterceptorArray: [injectResponseCanceler],
[injectResponseCanceler], // 响应错误
'ok', implementResponseInterceptorErrorArray: [responseErrorCanceler],
)
}
/**
*
*
*
*/
export const setupResponseErrorInterceptor = () => {
setImplement(
'implementResponseInterceptorErrorArray',
[responseError],
'error',
)
} }

View File

@ -13,20 +13,20 @@
* *
* *
* , inject * , inject
* *
*/ */
import axios from 'axios' import axios from 'axios'
import { AXIOS_CONFIG } from '@/app-config/requestConfig' import { AXIOS_CONFIG } from '@/app-config/requestConfig'
import { useAxiosInterceptor, axiosCanceler } from '@/axios/helper/interceptor' import { useAxiosInterceptor } from '@/axios/helper/interceptor'
import { import {
setupResponseInterceptor, setupResponseInterceptor,
setupResponseErrorInterceptor, setupResponseErrorInterceptor,
} from '@/axios/inject/response/provide' } from '@/axios/inject/response'
import { import {
setupRequestInterceptor, setupRequestInterceptor,
setupRequestErrorInterceptor, setupRequestErrorInterceptor,
} from '@/axios/inject/request/provide' } from '@/axios/inject/request'
import type { AxiosInstanceExpand } from './type' import type { AxiosInstanceExpand } from './type'
@ -70,9 +70,6 @@ server.interceptors.response.use(
setupResponseErrorInterceptor() setupResponseErrorInterceptor()
fetchError('responseError', error, 'implementResponseInterceptorErrorArray') fetchError('responseError', error, 'implementResponseInterceptorErrorArray')
// 注销该失败请求的取消器
axiosCanceler.removePendingRequest(error.config || {})
return Promise.reject(error) return Promise.reject(error)
}, },
) )

View File

@ -69,7 +69,7 @@ export default defineComponent({
props, props,
setup(props, { expose }) { setup(props, { expose }) {
const settingStore = useSetting() const settingStore = useSetting()
const { themeValue } = storeToRefs(settingStore) const { themeValue: currentTheme } = storeToRefs(settingStore)
const rayChartRef = ref<HTMLElement>() // `echart` 容器实例 const rayChartRef = ref<HTMLElement>() // `echart` 容器实例
const rayChartWrapperRef = ref<HTMLElement>() const rayChartWrapperRef = ref<HTMLElement>()
const echartInstanceRef = ref<ECharts>() // `echart` 实例 const echartInstanceRef = ref<ECharts>() // `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` * @returns `chart options`
@ -171,7 +195,7 @@ export default defineComponent({
* *
* 使, `legend` * 使, `legend`
*/ */
const renderChart = (theme: ChartTheme = echartTheme) => { const renderChart = (theme: string = echartTheme) => {
/** 获取 dom 容器 */ /** 获取 dom 容器 */
const element = rayChartRef.value as HTMLElement 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` , * `chart` ,
@ -249,7 +255,10 @@ export default defineComponent({
/** 重置 echarts 尺寸 */ /** 重置 echarts 尺寸 */
const resizeChart = () => { const resizeChart = () => {
if (echartInstanceRef.value) { 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 return
} }
if (props.autoChangeTheme) { updateChartTheme()
/** 注册 echarts */
renderThemeChart(themeValue.value)
} else {
props.theme ? renderChart(`${echartTheme}-dark`) : renderChart()
}
/** 注册事件 */ /** 注册事件 */
if (props.autoResize) { if (props.autoResize) {
@ -296,8 +300,8 @@ export default defineComponent({
/** 监听全局主题变化, 然后重新渲染对应主题 echarts */ /** 监听全局主题变化, 然后重新渲染对应主题 echarts */
watch( watch(
() => themeValue.value, () => currentTheme.value,
(theme) => { () => {
/** /**
* *
* Q: 为什么需要重新卸载再渲染 * Q: 为什么需要重新卸载再渲染
@ -306,8 +310,7 @@ export default defineComponent({
*/ */
if (props.autoChangeTheme) { if (props.autoChangeTheme) {
destroyChart() destroyChart()
updateChartTheme()
renderThemeChart(theme)
} }
}, },
) )
@ -322,12 +325,7 @@ export default defineComponent({
() => props.showAria, () => props.showAria,
() => { () => {
destroyChart() destroyChart()
updateChartTheme()
if (props.autoChangeTheme || props.theme) {
themeValue.value ? renderChart(`${echartTheme}-dark`) : renderChart()
} else {
renderChart()
}
}, },
) )
@ -369,13 +367,11 @@ export default defineComponent({
/** 注册 echarts 组件与渲染器 */ /** 注册 echarts 组件与渲染器 */
await registerChartCore() await registerChartCore()
}) })
onMounted(() => { onMounted(() => {
nextTick(() => { nextTick(() => {
mount() mount()
}) })
}) })
onBeforeUnmount(() => { onBeforeUnmount(() => {
unmount() unmount()
watchCallback?.() watchCallback?.()

View File

@ -47,6 +47,7 @@ const props = {
canvasRender: { canvasRender: {
/** /**
* *
* @deprecated
* `chart` , 使 `canvas` * `chart` , 使 `canvas`
* *
* , `SVGRenderer` * , `SVGRenderer`
@ -91,8 +92,12 @@ const props = {
default: null, default: null,
}, },
theme: { theme: {
type: [String, Object] as PropType<ChartTheme>, /**
default: '', *
* chart theme
*/
type: String as PropType<ChartTheme>,
default: null,
}, },
autoChangeTheme: { autoChangeTheme: {
/** /**

View File

@ -43,7 +43,12 @@ export type AutoResize =
height: number height: number
} }
export type ChartTheme = 'macarons-dark' | string | object | 'macarons' export type ChartTheme =
| 'macarons-dark'
| 'macarons'
| 'default'
| string
| null
export interface RayChartInst { export interface RayChartInst {
/** /**

View File

@ -7,4 +7,14 @@
1. 配置、选择主题 1. 配置、选择主题
2. 点击下载主题 2. 点击下载主题
3. 选择 json 类型,然后复制 3. 选择 json 类型,然后复制
4. 在 @/echart-themes 包中创建对应的 json 文件,文件名为主题名称 4. 在 src/echart-themes 包中创建对应的 json 文件,文件名为主题名称
## 注意
### 一份主题
如果有且仅有一份 echart theme则会视为明暗主题色都共用一套主题色。
### 两份主题
下载好的主题应该分为xxx 与 xxx-dark 两份。这样模板会自动根据配置主题色切换明暗主题。

View File

@ -20,6 +20,7 @@
/** 全局响应式变量 */ /** 全局响应式变量 */
const variableState = reactive({ const variableState = reactive({
globalSpinning: false, globalSpinning: false,
globalDrawerValue: false,
}) })
type VariableStateKey = keyof typeof variableState type VariableStateKey = keyof typeof variableState

View File

@ -12,5 +12,6 @@
import { useI18n, t } from './useI18n' import { useI18n, t } from './useI18n'
import { useVueRouter } from '../web/useVueRouter' import { useVueRouter } from '../web/useVueRouter'
import { useDayjs } from '../web/useDayjs' import { useDayjs } from '../web/useDayjs'
import { useDevice } from './useDevice'
export { useI18n, useVueRouter, useDayjs, t } export { useI18n, useVueRouter, useDayjs, t, useDevice }

View File

@ -0,0 +1,32 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-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,
}
}

6
src/icons/menu.svg Normal file
View File

@ -0,0 +1,6 @@
<svg t="1698137684048" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="3267" width="64" height="64">
<path
d="M495.65696 499.15904H250.80832C115.79392 499.15904 5.95968 389.3248 5.95968 254.3104S115.79392 9.46176 250.80832 9.46176s244.8384 109.83424 244.8384 244.84864c0 1.1776-0.03072 2.34496-0.07168 3.51232l-0.04096 1.52576 0.12288 239.81056zM250.80832 50.42176c-112.42496 0-203.88864 91.46368-203.88864 203.88864s91.46368 203.88864 203.88864 203.88864h203.8784l-0.1024-199.22944 0.11264-4.6592c-0.01024-112.42496-91.46368-203.88864-203.88864-203.88864zM776.00768 499.15904H531.15904l0.12288-240.20992-0.04096-1.11616a99.95264 99.95264 0 0 1-0.07168-3.51232c0-135.0144 109.83424-244.84864 244.8384-244.84864s244.84864 109.824 244.84864 244.8384-109.83424 244.84864-244.84864 244.84864z m-203.86816-40.96h203.86816c112.42496 0 203.88864-91.46368 203.88864-203.88864S888.44288 50.42176 776.00768 50.42176c-112.42496 0-203.8784 91.46368-203.8784 203.88864l0.11264 4.2496-0.1024 199.63904zM247.98208 1024C112.97792 1024 3.14368 914.15552 3.14368 779.15136s109.83424-244.8384 244.84864-244.8384H492.8512l-0.13312 240.20992 0.04096 1.00352c0.04096 1.19808 0.08192 2.4064 0.08192 3.62496C492.83072 914.15552 382.99648 1024 247.98208 1024z m0-448.72704c-112.42496 0-203.88864 91.46368-203.88864 203.8784C44.10368 891.57632 135.55712 983.04 247.98208 983.04s203.88864-91.46368 203.88864-203.88864l-0.11264-4.17792 0.1024-199.70048h-203.8784zM773.20192 1024c-135.0144 0-244.84864-109.84448-244.84864-244.84864 0-1.28 0.04096-2.53952 0.08192-3.79904l0.0512-1.37216-0.14336-239.6672h244.85888c135.0144 0 244.8384 109.83424 244.8384 244.8384S908.20608 1024 773.20192 1024z m-203.8784-448.72704l0.11264 199.22944-0.12288 4.64896C569.31328 891.57632 660.76672 983.04 773.20192 983.04c112.42496 0 203.8784-91.46368 203.8784-203.88864 0-112.42496-91.46368-203.8784-203.8784-203.8784h-203.8784z"
fill="currentColor" p-id="3268"></path>
</svg>

After

Width:  |  Height:  |  Size: 1.9 KiB

View File

@ -0,0 +1,7 @@
.n-drawer.app-menu__drawer {
width: auto !important;
& .n-layout-sider {
height: 100%;
}
}

View File

@ -9,11 +9,15 @@
* @remark * @remark
*/ */
import { NMenu, NLayoutSider } from 'naive-ui' import './index.scss'
import { NMenu, NLayoutSider, NDrawer } from 'naive-ui'
import SiderBarLogo from './components/SiderBarLogo/index' import SiderBarLogo from './components/SiderBarLogo/index'
import { useMenu } from '@/store' import { useMenu } from '@/store'
import { APP_MENU_CONFIG } from '@/app-config/appConfig' 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 { MenuInst } from 'naive-ui'
import type { NaiveMenuOptions } from '@/types/modules/component' import type { NaiveMenuOptions } from '@/types/modules/component'
@ -35,11 +39,21 @@ const LayoutMenu = defineComponent({
return menuStore.menuKey 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 modelMenuOptions = computed(() => menuStore.options)
const modelCollapsed = computed(() => menuStore.collapsed) const modelCollapsed = computed(() => menuStore.collapsed)
const { isTabletOrSmaller } = useDevice()
const modelGlobalDrawerValue = computed({
get: () => globalVariableToRefs('globalDrawerValue').value,
set: (val) => {
setVariable('globalDrawerValue', val)
},
})
const showMenuOption = () => { const showMenuOption = () => {
const key = modelMenuKey.value as string const key = modelMenuKey.value as string
@ -49,42 +63,55 @@ const LayoutMenu = defineComponent({
}) })
} }
return { const BaseicMenu = () => (
modelMenuKey,
changeMenuModelValue,
modelMenuOptions,
modelCollapsed,
collapsedMenu,
menuRef,
}
},
render() {
return (
<NLayoutSider <NLayoutSider
bordered bordered
showTrigger showTrigger={!isTabletOrSmaller.value}
collapseMode={APP_MENU_CONFIG.MENU_COLLAPSED_MODE} collapseMode={APP_MENU_CONFIG.menuCollapsedMode}
collapsedWidth={APP_MENU_CONFIG.MENU_COLLAPSED_WIDTH} collapsedWidth={APP_MENU_CONFIG.menuCollapsedWidth}
onUpdateCollapsed={this.collapsedMenu.bind(this)} onUpdateCollapsed={collapsedMenu.bind(this)}
nativeScrollbar={false} nativeScrollbar={false}
> >
<SiderBarLogo collapsed={this.modelCollapsed} /> <SiderBarLogo collapsed={modelCollapsed.value} />
<NMenu <NMenu
ref="menuRef" ref="menuRef"
class="r-menu--app" class="r-menu--app"
v-model:value={this.modelMenuKey} v-model:value={modelMenuKey.value}
options={this.modelMenuOptions as NaiveMenuOptions[]} options={modelMenuOptions.value as NaiveMenuOptions[]}
indent={APP_MENU_CONFIG.MENU_COLLAPSED_INDENT} indent={APP_MENU_CONFIG.menuCollapsedIndent}
collapsed={this.modelCollapsed} collapsed={modelCollapsed.value}
collapsedIconSize={APP_MENU_CONFIG.MENU_COLLAPSED_ICON_SIZE} collapsedIconSize={APP_MENU_CONFIG.menuCollapsedIconSize}
collapsedWidth={APP_MENU_CONFIG.MENU_COLLAPSED_WIDTH} collapsedWidth={APP_MENU_CONFIG.menuCollapsedWidth}
onUpdateValue={(key, op) => { onUpdateValue={(key, op) => {
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}
/> />
</NLayoutSider> </NLayoutSider>
) )
return {
menuRef,
isTabletOrSmaller,
BaseicMenu,
modelGlobalDrawerValue,
}
},
render() {
const { isTabletOrSmaller, BaseicMenu } = this
return !isTabletOrSmaller ? (
<BaseicMenu />
) : (
<NDrawer
class="app-menu__drawer"
v-model:show={this.modelGlobalDrawerValue}
placement="left"
displayDirective="show"
>
<BaseicMenu />
</NDrawer>
)
}, },
}) })

View File

@ -21,6 +21,7 @@
import { NDropdown, NBreadcrumb, NBreadcrumbItem } from 'naive-ui' import { NDropdown, NBreadcrumb, NBreadcrumbItem } from 'naive-ui'
import { useMenu } from '@/store' import { useMenu } from '@/store'
import { useDevice } from '@/hooks/web/index'
import type { DropdownOption } from 'naive-ui' import type { DropdownOption } from 'naive-ui'
import type { import type {
@ -29,7 +30,7 @@ import type {
AppMenuKey, AppMenuKey,
} from '@/types/modules/app' } from '@/types/modules/app'
const Breadcrumb = defineComponent({ export default defineComponent({
name: 'RBreadcrumb', name: 'RBreadcrumb',
setup() { setup() {
const menuStore = useMenu() const menuStore = useMenu()
@ -37,15 +38,13 @@ const Breadcrumb = defineComponent({
const { changeMenuModelValue } = menuStore const { changeMenuModelValue } = menuStore
const { breadcrumbOptions } = storeToRefs(menuStore) const { breadcrumbOptions } = storeToRefs(menuStore)
const modelBreadcrumbOptions = computed(() => breadcrumbOptions.value) const modelBreadcrumbOptions = computed(() => breadcrumbOptions.value)
const { isTabletOrSmaller } = useDevice()
const handleDropdownSelect = ( const dropdownSelect = (key: string | number, option: DropdownOption) => {
key: string | number,
option: DropdownOption,
) => {
changeMenuModelValue(key, option as unknown as AppMenuOption) changeMenuModelValue(key, option as unknown as AppMenuOption)
} }
const handleBreadcrumbItemClick = (option: AppMenuOption) => { const breadcrumbItemClick = (option: AppMenuOption) => {
if (!option.children?.length) { if (!option.children?.length) {
const { meta = {} } = option const { meta = {} } = option
@ -57,24 +56,29 @@ const Breadcrumb = defineComponent({
return { return {
modelBreadcrumbOptions, modelBreadcrumbOptions,
handleDropdownSelect, dropdownSelect,
handleBreadcrumbItemClick, breadcrumbItemClick,
isTabletOrSmaller,
} }
}, },
render() { render() {
return ( const { isTabletOrSmaller } = this
return isTabletOrSmaller ? (
<div></div>
) : (
<NBreadcrumb> <NBreadcrumb>
{this.modelBreadcrumbOptions.map((curr) => ( {this.modelBreadcrumbOptions.map((curr) => (
<NBreadcrumbItem <NBreadcrumbItem
key={curr.key} key={curr.key}
onClick={this.handleBreadcrumbItemClick.bind(this, curr)} onClick={this.breadcrumbItemClick.bind(this, curr)}
> >
<NDropdown <NDropdown
labelField="breadcrumbLabel" labelField="breadcrumbLabel"
options={ options={
curr.children && curr.children?.length > 1 ? curr.children : [] curr.children && curr.children?.length > 1 ? curr.children : []
} }
onSelect={this.handleDropdownSelect.bind(this)} onSelect={this.dropdownSelect.bind(this)}
> >
{{ {{
default: () => ( default: () => (
@ -92,5 +96,3 @@ const Breadcrumb = defineComponent({
) )
}, },
}) })
export default Breadcrumb

View File

@ -18,11 +18,12 @@ import { on, off, queryElements, addClass, removeClass } from '@/utils/element'
import { debounce } from 'lodash-es' import { debounce } from 'lodash-es'
import { useMenu } from '@/store' import { useMenu } from '@/store'
import { validMenuItemShow } from '@/router/helper/routerCopilot' import { validMenuItemShow } from '@/router/helper/routerCopilot'
import { useDevice } from '@/hooks/web/index'
import type { AppRouteMeta } from '@/router/type' import type { AppRouteMeta } from '@/router/type'
import type { AppMenuOption } from '@/types/modules/app' import type { AppMenuOption } from '@/types/modules/app'
const GlobalSeach = defineComponent({ export default defineComponent({
name: 'GlobalSeach', name: 'GlobalSeach',
props: { props: {
show: { show: {
@ -76,6 +77,7 @@ const GlobalSeach = defineComponent({
let searchElementIndex = 0 let searchElementIndex = 0
/** 缓存索引 */ /** 缓存索引 */
let preSearchElementIndex = searchElementIndex let preSearchElementIndex = searchElementIndex
const { isTabletOrSmaller } = useDevice()
/** 初始化一些值 */ /** 初始化一些值 */
const resetSearchSomeValue = () => { const resetSearchSomeValue = () => {
@ -240,13 +242,18 @@ const GlobalSeach = defineComponent({
autoFouceSearchItem() autoFouceSearchItem()
} }
watchEffect(() => {
if (isTabletOrSmaller.value) {
modelShow.value = false
}
})
onMounted(() => { onMounted(() => {
on(window, 'keydown', (e: Event) => { on(window, 'keydown', (e: Event) => {
registerArouseKeyboard(e as KeyboardEvent) registerArouseKeyboard(e as KeyboardEvent)
registerChangeSearchElementIndex(e as KeyboardEvent) registerChangeSearchElementIndex(e as KeyboardEvent)
}) })
}) })
onBeforeUnmount(() => { onBeforeUnmount(() => {
off(window, 'keydown', (e: Event) => { off(window, 'keydown', (e: Event) => {
registerArouseKeyboard(e as KeyboardEvent) registerArouseKeyboard(e as KeyboardEvent)
@ -261,10 +268,15 @@ const GlobalSeach = defineComponent({
handleSearchMenuOptions: debounce(handleSearchMenuOptions, 300), handleSearchMenuOptions: debounce(handleSearchMenuOptions, 300),
handleSearchItemClick, handleSearchItemClick,
RenderPreIcon, RenderPreIcon,
isTabletOrSmaller,
} }
}, },
render() { render() {
return ( const { isTabletOrSmaller } = this
return isTabletOrSmaller ? (
<div></div>
) : (
<NModal v-model:show={this.modelShow} transform-origin="center" show> <NModal v-model:show={this.modelShow} transform-origin="center" show>
<div class="global-seach global-seach--dark global-seach--light"> <div class="global-seach global-seach--dark global-seach--light">
<div class="global-seach__wrapper"> <div class="global-seach__wrapper">
@ -339,5 +351,3 @@ const GlobalSeach = defineComponent({
) )
}, },
}) })
export default GlobalSeach

View File

@ -113,7 +113,7 @@ const SettingDrawer = defineComponent({
{t('headerSettingOptions.ThemeOptions.PrimaryColorConfig')} {t('headerSettingOptions.ThemeOptions.PrimaryColorConfig')}
</NDivider> </NDivider>
<NColorPicker <NColorPicker
swatches={APP_THEME.APP_THEME_COLOR} swatches={APP_THEME.appThemeColors}
v-model:value={this.primaryColorOverride.common!.primaryColor} v-model:value={this.primaryColorOverride.common!.primaryColor}
onUpdateValue={this.changePrimaryColor.bind(this)} onUpdateValue={this.changePrimaryColor.bind(this)}
/> />

View File

@ -32,8 +32,10 @@ import { LOCAL_OPTIONS } from '@/app-config/localConfig'
import { useAvatarOptions, avatarDropdownClick } from './hook' import { useAvatarOptions, avatarDropdownClick } from './hook'
import { useI18n } from '@/hooks/web/index' import { useI18n } from '@/hooks/web/index'
import { useFullscreen } from 'vue-hooks-plus' 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({ const SiderBar = defineComponent({
name: 'SiderBar', name: 'SiderBar',
@ -53,55 +55,76 @@ const SiderBar = defineComponent({
display: 'flex', display: 'flex',
} }
const globalSearchShown = ref(false) const globalSearchShown = ref(false)
const { isTabletOrSmaller } = useDevice()
const globalDrawerValue = globalVariableToRefs('globalDrawerValue')
/** /**
* *
* *
*/ */
const leftIconOptions = computed(() => [ const leftIconOptions = computed(() => {
{ const options: LeftIconOptions[] = [
name: 'reload', {
size: 18, name: 'reload',
tooltip: t('headerTooltip.Reload'), size: 18,
iconClass: computed(() => tooltip: t('headerTooltip.Reload'),
!reloadRouteSwitch.value ? 'ray-icon__reload--loading' : '', iconClass: computed(() =>
), !reloadRouteSwitch.value ? 'ray-icon__reload--loading' : '',
}, ),
]) },
]
if (isTabletOrSmaller.value) {
options[0] = {
name: 'menu',
size: 18,
}
}
return options
})
/** /**
* *
* *
*/ */
const rightTooltipIconOptions = computed(() => [ const rightTooltipIconOptions = computed(() => {
{ const options = [
name: 'search', {
size: 18, name: 'search',
tooltip: t('headerTooltip.Search'), size: 18,
eventKey: 'search', tooltip: t('headerTooltip.Search'),
}, eventKey: 'search',
{ },
name: 'fullscreen', {
size: 18, name: 'fullscreen',
tooltip: computed(() => size: 18,
isFullscreen.value tooltip: computed(() =>
? t('headerTooltip.CancelFullScreen') isFullscreen.value
: t('headerTooltip.FullScreen'), ? t('headerTooltip.CancelFullScreen')
), : t('headerTooltip.FullScreen'),
eventKey: 'screen', ),
}, eventKey: 'screen',
{ },
name: 'github', {
size: 18, name: 'github',
tooltip: t('headerTooltip.Github'), size: 18,
eventKey: 'github', tooltip: t('headerTooltip.Github'),
}, eventKey: 'github',
{ },
name: 'setting', {
size: 18, name: 'setting',
tooltip: t('headerTooltip.Setting'), size: 18,
eventKey: 'setting', tooltip: t('headerTooltip.Setting'),
}, eventKey: 'setting',
]) },
]
if (isTabletOrSmaller.value) {
options.shift()
}
return options
})
const iconEventMap: IconEventMapOptions = { const iconEventMap: IconEventMapOptions = {
// 刷新组件重新加载,手动设置 800ms loading 时长 // 刷新组件重新加载,手动设置 800ms loading 时长
reload: () => { reload: () => {
@ -124,26 +147,46 @@ const SiderBar = defineComponent({
lock: () => { lock: () => {
changeSwitcher(true, 'lockScreenSwitch') changeSwitcher(true, 'lockScreenSwitch')
}, },
menu: () => {
setVariable('globalDrawerValue', !globalDrawerValue.value)
},
} }
const handleIconClick = (key: IconEventMap) => { const toolIconClick = (key: IconEventMap) => {
iconEventMap[key]?.() iconEventMap[key]?.()
} }
const LeftToolIcon = (props: (typeof leftIconOptions.value)[0]) => {
const { iconClass, name, size } = props
return (
<RIcon
customClassName={`${isRef(iconClass) ? iconClass.value : iconClass}`}
name={name}
size={size}
cursor="pointer"
onClick={toolIconClick.bind(this, name)}
/>
)
}
return { return {
leftIconOptions, leftIconOptions,
rightTooltipIconOptions, rightTooltipIconOptions,
t, t,
handleIconClick, toolIconClick,
showSettings, showSettings,
updateLocale, updateLocale,
spaceItemStyle, spaceItemStyle,
drawerPlacement, drawerPlacement,
breadcrumbSwitch, breadcrumbSwitch,
globalSearchShown, globalSearchShown,
LeftToolIcon,
} }
}, },
render() { render() {
const { LeftToolIcon } = this
return ( return (
<NLayoutHeader class="layout-header" bordered> <NLayoutHeader class="layout-header" bordered>
<GlobalSeach v-model:show={this.globalSearchShown} /> <GlobalSeach v-model:show={this.globalSearchShown} />
@ -157,26 +200,18 @@ const SiderBar = defineComponent({
wrapItem={false} wrapItem={false}
itemStyle={this.spaceItemStyle} itemStyle={this.spaceItemStyle}
> >
{this.leftIconOptions.map((curr) => ( {this.leftIconOptions.map((curr) =>
<NTooltip> curr.tooltip ? (
{{ <NTooltip>
trigger: () => ( {{
<RIcon trigger: () => <LeftToolIcon {...curr} />,
customClassName={`${ default: () => curr.tooltip,
isRef(curr.iconClass) }}
? curr.iconClass.value </NTooltip>
: curr.iconClass ) : (
}`} <LeftToolIcon {...curr} />
name={curr.name} ),
size={curr.size} )}
cursor="pointer"
onClick={this.handleIconClick.bind(this, curr.name)}
/>
),
default: () => curr.tooltip,
}}
</NTooltip>
))}
{this.breadcrumbSwitch ? <Breadcrumb /> : null} {this.breadcrumbSwitch ? <Breadcrumb /> : null}
</NSpace> </NSpace>
<NSpace <NSpace
@ -190,7 +225,7 @@ const SiderBar = defineComponent({
tooltipText={ tooltipText={
isRef(curr.tooltip) ? curr.tooltip.value : curr.tooltip isRef(curr.tooltip) ? curr.tooltip.value : curr.tooltip
} }
onClick={this.handleIconClick.bind(this, curr.name)} onClick={this.toolIconClick.bind(this, curr.name)}
/> />
))} ))}
<NDropdown <NDropdown

View File

@ -1,4 +1,5 @@
import type { DropdownOption } from 'naive-ui' import type { DropdownOption } from 'naive-ui'
import type { ComputedRef } from 'vue'
export interface IconEventMapOptions { export interface IconEventMapOptions {
[propName: string]: (...args: unknown[]) => unknown [propName: string]: (...args: unknown[]) => unknown
@ -20,3 +21,10 @@ export interface IconOptions {
eventKey?: string eventKey?: string
dropdown?: IconDropdownOptions dropdown?: IconDropdownOptions
} }
export interface LeftIconOptions {
name: string
size: number
tooltip?: string
iconClass?: ComputedRef
}

View File

@ -126,7 +126,7 @@ export const naiveLocales = (key: string) => {
* @remak , `main.ts` , `i18n` * @remak , `main.ts` , `i18n`
*/ */
export const getAppDefaultLanguage = () => { export const getAppDefaultLanguage = () => {
const language = getStorage<string>( const language = getStorage(
APP_CATCH_KEY.localeLanguage, APP_CATCH_KEY.localeLanguage,
'localStorage', 'localStorage',
SYSTEM_DEFAULT_LOCAL, SYSTEM_DEFAULT_LOCAL,

View File

@ -163,7 +163,7 @@ export const hasMenuIcon = (option: AppMenuOption) => {
RIcon, RIcon,
{ {
name: meta!.icon as string, name: meta!.icon as string,
size: APP_MENU_CONFIG.MENU_COLLAPSED_ICON_SIZE, size: APP_MENU_CONFIG.menuCollapsedIconSize,
cursor: 'pointer', cursor: 'pointer',
}, },
{}, {},

View File

@ -22,7 +22,7 @@ export const useSetting = defineStore(
const settingState = reactive<SettingState>({ const settingState = reactive<SettingState>({
drawerPlacement: 'right', drawerPlacement: 'right',
primaryColorOverride: { primaryColorOverride: {
...APP_THEME.APP_NAIVE_UI_THEME_OVERRIDES, ...APP_THEME.appNaiveUIThemeOverrides,
common: { common: {
primaryColor: primaryColor, // 主题色 primaryColor: primaryColor, // 主题色
primaryColorHover: primaryColor, primaryColorHover: primaryColor,

View File

@ -3,11 +3,11 @@ import type { CreateAxiosDefaults } from 'axios'
export type CollapsedMode = 'transform' | 'width' export type CollapsedMode = 'transform' | 'width'
export interface AppMenuConfig { export interface AppMenuConfig {
MENU_COLLAPSED_WIDTH: number menuCollapsedWidth: number
MENU_COLLAPSED_MODE: CollapsedMode menuCollapsedMode: CollapsedMode
MENU_COLLAPSED_ICON_SIZE: number menuCollapsedIconSize: number
MENU_COLLAPSED_INDENT: number menuCollapsedIndent: number
MENU_ACCORDION: boolean menuAccordion: boolean
} }
export interface AppKeepAlive { export interface AppKeepAlive {

View File

@ -76,8 +76,8 @@ export interface AppConfig {
export type AppConfigExport = Config & UserConfigExport export type AppConfigExport = Config & UserConfigExport
export interface AppTheme { export interface AppTheme {
APP_THEME_COLOR: string[] appThemeColors: string[]
APP_PRIMARY_COLOR: AppPrimaryColor appPrimaryColor: AppPrimaryColor
APP_NAIVE_UI_THEME_OVERRIDES: GlobalThemeOverrides appNaiveUIThemeOverrides: GlobalThemeOverrides
echartTheme: string echartTheme: string
} }

View File

@ -9,8 +9,6 @@
* @remark * @remark
*/ */
/** vue3 项目里建议直接用 vueuse useStorage 方法 */
import type { StorageLike, RemoveStorageKey } from '@/types/modules/utils' import type { StorageLike, RemoveStorageKey } from '@/types/modules/utils'
/** /**
@ -40,15 +38,13 @@ function setStorage<T = unknown>(
} }
} }
/** 重载函数 getStorage */ function getStorage<T = unknown>(
function getStorage<T>(
key: string, key: string,
storageType: StorageLike, storageType: StorageLike,
defaultValue: T, defaultValue: T,
): T ): T
/** 重载函数 getStorage */ function getStorage<T = unknown>(
function getStorage<T>(
key: string, key: string,
storageType?: StorageLike, storageType?: StorageLike,
defaultValue?: T, defaultValue?: T,
@ -59,7 +55,7 @@ function getStorage<T>(
* @param key key * @param key key
* @returns * @returns
*/ */
function getStorage<T>( function getStorage<T = unknown>(
key: string, key: string,
storageType: StorageLike = 'sessionStorage', storageType: StorageLike = 'sessionStorage',
defaultValue?: T, defaultValue?: T,

View File

@ -242,22 +242,23 @@ const Echart = defineComponent({
<NCard title="chart 组件"> <NCard title="chart 组件">
<ul> <ul>
<li> <li>
<h3> 200*200 </h3> <h3>1. 200*200 </h3>
</li> </li>
<li> <li>
<h3> <h3>
autoChangeThemeRayTemplate 2. autoChangeTheme
false APP_THEME.echartTheme RayTemplate
</h3> </h3>
</li> </li>
<li> <li>
<h3> watchOptions</h3> <h3>3. watchOptions</h3>
</li> </li>
<li> <li>
<h3> animation</h3> <h3>4. animation</h3>
</li> </li>
<li> <li>
<h3> setChartOptions </h3> <h3>5. setChartOptions </h3>
</li> </li>
</ul> </ul>
</NCard> </NCard>
@ -277,11 +278,11 @@ const Echart = defineComponent({
showAria={this.chartAria} showAria={this.chartAria}
/> />
</div> </div>
<NH2></NH2> <NH2></NH2>
<div class="chart--container"> <div class="chart--container">
<RChart <RChart
autoChangeTheme={false} autoChangeTheme={false}
theme="dark" theme="default"
options={this.baseOptions} options={this.baseOptions}
/> />
</div> </div>