This commit is contained in:
ray_wuhao 2023-04-17 12:37:25 +08:00
parent 2d360f392e
commit c179929e16
27 changed files with 586 additions and 128 deletions

View File

@ -1,5 +1,19 @@
# CHANGE LOG # CHANGE LOG
## 3.1.7
### Fixes
- 修复默认获取容器可视区域高度问题
### Feats
- 修改 Menu 菜单过滤逻辑,现在如果权限不匹配或者设置了 hidden 属性,则会被过滤掉
- 移除 $activedColor 全局 sass 变量,使用 --ray-theme-primary-color 替代
- 新增路由菜单检索功能
- 移除 App.tsx 中同步主题方法,改为使用 cfg 配置并且使用 ejs 注入
- 移除 MenuTag 默认主题色,现在会以当前主题色为主色
## 3.1.6 ## 3.1.6
### Fixes ### Fixes
@ -17,6 +31,9 @@
- 现在可以直接配置首屏加载动画一些信息(cfg.ts) - 现在可以直接配置首屏加载动画一些信息(cfg.ts)
- 新增对于 ejs 支持 - 新增对于 ejs 支持
- 补充一些细节注释 - 补充一些细节注释
- 新增 RayChart 组件 loading、loadingOptions 属性配置
- 新增反转色模式
- 修改 Menu 菜单过滤逻辑,现在如果权限不匹配或者设置了 hidden 属性,则会被过滤掉
## 3.1.5 ## 3.1.5

9
cfg.ts
View File

@ -55,8 +55,13 @@ const config: AppConfigExport = {
tagColor: '#ff6700', tagColor: '#ff6700',
titleColor: '#2d8cf0', titleColor: '#2d8cf0',
}, },
/** 默认主题色 */ /** 默认主题色(不可省略, 必填), 也用于 ejs 注入 */
primaryColor: '#2d8cf0', appPrimaryColor: {
/** 主题色 */
primaryColor: '#2d8cf0',
/** 主题辅助色(用于整体 hover、active 等之类颜色) */
primaryFadeColor: 'rgba(45, 140, 240, 0.25)',
},
/** /**
* *
* *

View File

@ -10,6 +10,8 @@
:root { :root {
--preloading-tag-color: <%= preloadingConfig.tagColor %>; --preloading-tag-color: <%= preloadingConfig.tagColor %>;
--preloading-title-color: <%= preloadingConfig.titleColor %>; --preloading-title-color: <%= preloadingConfig.titleColor %>;
--ray-theme-primary-fade-color: <%= appPrimaryColor.primaryFadeColor %>;
--ray-theme-primary-color: <%= appPrimaryColor.primaryColor %>;
} }
#pre-loading-animation { #pre-loading-animation {

View File

@ -22,7 +22,8 @@
"Setting": "Setting", "Setting": "Setting",
"Github": "Github", "Github": "Github",
"FullScreen": "Full Screen", "FullScreen": "Full Screen",
"CancelFullScreen": "Cancel Full Screen" "CancelFullScreen": "Cancel Full Screen",
"Search": "Search"
}, },
"LayoutHeaderSettingOptions": { "LayoutHeaderSettingOptions": {
"Title": "Configuration", "Title": "Configuration",
@ -31,7 +32,8 @@
"Dark": "Dark", "Dark": "Dark",
"Light": "Light", "Light": "Light",
"PrimaryColorConfig": "Primary Color" "PrimaryColorConfig": "Primary Color"
} },
"InterfaceDisplay": "Interface Display"
}, },
"LoginModule": { "LoginModule": {
"Register": "Register", "Register": "Register",

View File

@ -22,7 +22,8 @@
"Setting": "设置", "Setting": "设置",
"Github": "Github", "Github": "Github",
"FullScreen": "全屏", "FullScreen": "全屏",
"CancelFullScreen": "退出全屏" "CancelFullScreen": "退出全屏",
"Search": "搜索"
}, },
"LayoutHeaderSettingOptions": { "LayoutHeaderSettingOptions": {
"Title": "项目配置", "Title": "项目配置",
@ -31,7 +32,8 @@
"Dark": "暗色", "Dark": "暗色",
"Light": "明亮", "Light": "明亮",
"PrimaryColorConfig": "主题色" "PrimaryColorConfig": "主题色"
} },
"InterfaceDisplay": "界面显示"
}, },
"LoginModule": { "LoginModule": {
"Register": "注册", "Register": "注册",

View File

@ -1,7 +1,7 @@
{ {
"name": "ray-template", "name": "ray-template",
"private": true, "private": true,
"version": "3.1.6", "version": "3.1.7",
"type": "module", "type": "module",
"scripts": { "scripts": {
"dev": "vite", "dev": "vite",

View File

@ -14,21 +14,6 @@ const App = defineComponent({
const { themeValue } = storeToRefs(settingStore) const { themeValue } = storeToRefs(settingStore)
/** 同步主题色变量至 body, 如果未获取到缓存值则已默认值填充 */
const syncPrimaryColorToBody = () => {
const { primaryColor } = __APP_CFG__ // 默认主题色
const body = document.body
const primaryColorOverride = getCache('piniaSettingStore', 'localStorage')
const _p = get(
primaryColorOverride,
'primaryColorOverride.common.primaryColor',
)
/** 设置全局主题色 css 变量 */
body.style.setProperty('--ray-theme-primary-color', _p || primaryColor)
}
/** 隐藏加载动画 */ /** 隐藏加载动画 */
const hiddenLoadingAnimation = () => { const hiddenLoadingAnimation = () => {
/** pre-loading-animation 是默认 id */ /** pre-loading-animation 是默认 id */
@ -41,7 +26,6 @@ const App = defineComponent({
} }
} }
syncPrimaryColorToBody()
hiddenLoadingAnimation() hiddenLoadingAnimation()
/** 切换主题时, 同步更新 body class 以便于进行自定义 css 配置 */ /** 切换主题时, 同步更新 body class 以便于进行自定义 css 配置 */

View File

@ -27,7 +27,6 @@ import { cloneDeep, debounce } from 'lodash-es'
import { on, off, addStyle } from '@/utils/element' import { on, off, addStyle } from '@/utils/element'
import type { PropType } from 'vue' import type { PropType } from 'vue'
// import type { DebouncedFuncLeading } from 'lodash-es'
export type AutoResize = export type AutoResize =
| boolean | boolean
@ -59,7 +58,7 @@ export type ChartTheme = 'dark' | '' | object
* *
* 便使, , * 便使, ,
*/ */
export const loadingOptions = (options: LoadingOptions) => export const loadingOptions = (options?: LoadingOptions) =>
Object.assign( Object.assign(
{}, {},
{ {
@ -189,6 +188,16 @@ const RayChart = defineComponent({
type: Boolean, type: Boolean,
default: true, default: true,
}, },
loading: {
/** 加载动画 */
type: Boolean,
default: false,
},
loadingOptions: {
/** 配置加载动画样式 */
type: Object as PropType<LoadingOptions>,
default: () => loadingOptions(),
},
}, },
setup(props) { setup(props) {
const settingStore = useSetting() const settingStore = useSetting()
@ -196,7 +205,7 @@ const RayChart = defineComponent({
const rayChartRef = ref<HTMLElement>() // `echart` 容器实例 const rayChartRef = ref<HTMLElement>() // `echart` 容器实例
const echartInstanceRef = ref<EChartsInstance>() // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题 const echartInstanceRef = ref<EChartsInstance>() // `echart` 拷贝实例, 解决直接使用响应式实例带来的问题
let echartInstance: EChartsInstance // `echart` 实例 let echartInstance: EChartsInstance // `echart` 实例
let resizeDebounce: AnyFunc // resize 防抖方法 let resizeDebounce: AnyFunc // resize 防抖方法
const cssVarsRef = computed(() => { const cssVarsRef = computed(() => {
const cssVars = { const cssVars = {
@ -206,13 +215,15 @@ const RayChart = defineComponent({
return cssVars return cssVars
}) })
const modelLoadingOptions = computed(() =>
loadingOptions(props.loadingOptions),
)
/** /**
* *
* `echart` , , * `echart` , ,
* *
* `echart` * `echart`
*
* *
*/ */
const registerChartCore = async () => { const registerChartCore = async () => {
@ -400,6 +411,16 @@ const RayChart = defineComponent({
}, },
) )
/** 显示/隐藏加载动画 */
watch(
() => props.loading,
(newData) => {
newData
? echartInstance?.showLoading(modelLoadingOptions.value)
: echartInstance?.hideLoading()
},
)
/** 监听 options 变化 */ /** 监听 options 变化 */
if (props.watchOptions) { if (props.watchOptions) {
watch( watch(
@ -466,7 +487,6 @@ export default RayChart
* `chart` , 使, 使 `use` * `chart` , 使, 使 `use`
* *
* 预引入: 柱状图, 线, , k线图, * 预引入: 柱状图, 线, , k线图,
*
* 预引入: 提示框, , , , * 预引入: 提示框, , , ,
* *
* , `setOption` * , `setOption`

View File

@ -58,7 +58,7 @@
& .draggable-item__icon { & .draggable-item__icon {
&.draggable-item__icon--actived { &.draggable-item__icon--actived {
color: $activedColor; color: var(--ray-theme-primary-color);
} }
} }

View File

@ -27,7 +27,7 @@
&.dropdown-item--active, &.dropdown-item--active,
&:hover { &:hover {
background-color: $hoverLightBackgroundColor; background-color: $hoverLightBackgroundColor;
color: $activedColor; color: var(--ray-theme-primary-color);
} }
} }
} }
@ -39,7 +39,7 @@
& .table-size__dropdown-wrapper { & .table-size__dropdown-wrapper {
& .dropdown-item:hover { & .dropdown-item:hover {
background-color: $hoverDarkBackgroundColor; background-color: $hoverDarkBackgroundColor;
color: $activedColor; color: var(--ray-theme-primary-color);
} }
} }
} }

17
src/icons/AA_READMEmd Normal file
View File

@ -0,0 +1,17 @@
## 说明
该文件包属于全局 `svg icon`,配合 `RayIcon` 组件使用。
## TIP
添加新的 `svg` 图标时,应该注意图标自带 `fill` 属性的管理。如果自带了 `fill` 属性的图标,则会导致使用组件 `color` 属性失效的问题。所以如果是需要动态使用 `css` 属性控制样式的图标,应该去掉其 `fill` 属性或者配置为 `fill = currentColor`。
```html
<svg fill="currentColor"></svg>
```
## 使用方法
- 导入 `svg` 图标
- 命名(`命名必须全局唯一,并且尽量避免使用特殊符号`
- 导入 `RayIcon` 组件,配置 `name` 属性即可将 `svg` 作为图标使用

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

@ -0,0 +1,6 @@
<svg t="1681648912704" class="icon" viewBox="0 0 1024 1024" version="1.1"
xmlns="http://www.w3.org/2000/svg" p-id="8713" width="64" height="64">
<path
d="M907.4 906.4c-29 29-76 29-105 0L657.7 761.7c-57.6 38.4-126.8 60.9-201.3 60.9-200.5 0-363.1-162.6-363.1-363.2S255.9 96.2 456.5 96.2s363.2 162.6 363.2 363.2c0 72.4-21.2 139.9-57.7 196.5l145.5 145.5c28.9 29 28.9 76-0.1 105zM456.4 231C330.3 231 228 333.3 228 459.4c0 126.1 102.3 228.4 228.4 228.4s228.4-102.3 228.4-228.4C684.9 333.3 582.6 231 456.4 231z m118.1 379.4c-1.4 2.1-3.5 3.6-6.2 4.2-5.5 1.3-11.1-2.2-12.3-7.7-1.2-5.2 1.7-10.3 6.7-12 59-46.5 80.8-126.4 53.6-196.1-0.7-1.4-1-2.9-1-4.5 0-5.7 4.6-10.3 10.3-10.3 4.2 0 7.8 2.5 9.4 6.1h0.1c30.7 78.3 6.1 168.4-60.6 220.3z"
p-id="8714" fill="currentColor"></path>
</svg>

After

Width:  |  Height:  |  Size: 784 B

View File

@ -9,6 +9,15 @@
* @remark * @remark
*/ */
/**
*
* :
* - 关闭全部: 关闭所有标签页, rootRoute.path
* - 关闭右侧: 关闭右侧所有标签, ,
* - 关闭左侧: 关闭左侧所有标签, ,
* - 关闭其他: 关闭其他所有标签, ,
*/
import './index.scss' import './index.scss'
import { NScrollbar, NTag, NSpace, NLayoutHeader, NDropdown } from 'naive-ui' import { NScrollbar, NTag, NSpace, NLayoutHeader, NDropdown } from 'naive-ui'
@ -391,7 +400,7 @@ const MenuTag = defineComponent({
this.modelMenuTagOptions.length > 1 this.modelMenuTagOptions.length > 1
} }
onClose={() => this.closeCurrentMenuTag(idx)} onClose={() => this.closeCurrentMenuTag(idx)}
type={curr.key === this.menuKey ? 'success' : 'info'} type={curr.key === this.menuKey ? 'primary' : 'default'}
onClick={this.handleTagClick.bind(this, curr)} onClick={this.handleTagClick.bind(this, curr)}
bordered={false} bordered={false}
onContextmenu={this.handleContextMenu.bind(this, idx)} onContextmenu={this.handleContextMenu.bind(this, idx)}

View File

@ -0,0 +1,89 @@
.global-seach {
& .global-seach__wrapper {
box-sizing: border-box;
& .global-seach__card {
width: 650px;
height: 600px;
border-radius: 6px;
padding: 12px;
& .ray-icon {
color: var(--ray-theme-primary-color);
}
& .global-seach__card-header {
margin-bottom: 12px;
}
& .global-seach__card-content {
height: calc(100% - 98px);
& .content-item {
padding: 12px;
border-radius: 6px;
cursor: pointer;
transition: background-color 0.3s var(--r-bezier);
& .content-item-icon {
@include flexCenter;
}
}
}
& .global-seach__card-footer {
width: 100%;
& .card-footer__tip-wrapper {
display: flex;
align-items: center;
margin-top: 24px;
& .tip-wrapper-item {
display: flex;
align-items: center;
& .item-icon {
display: flex;
align-items: center;
margin-right: 4px;
& span {
color: var(--ray-theme-primary-color);
}
}
}
}
}
}
}
}
.ray-template--dark {
& .global-seach__card {
background-color: #242424;
& .global-seach__card-content .content-item {
background-color: #2f2f2f;
&:hover {
// background-color: $hoverDarkBackgroundColor;
background-color: var(--ray-theme-primary-fade-color);
}
}
}
}
.ray-template--light {
& .global-seach__card {
background-color: #f9f9f9;
& .global-seach__card-content .content-item {
background-color: #ffffff;
&:hover {
background-color: $hoverLightBackgroundColor;
}
}
}
}

View File

@ -0,0 +1,208 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-04-16
*
* @workspace ray-template
*
* @remark
*/
import './index.scss'
import { NInput, NModal, NScrollbar, NSpace } from 'naive-ui'
import RayIcon from '@/components/RayIcon/index'
import { on, off } from '@/utils/element'
import { debounce } from 'lodash-es'
import { useMenu } from '@/store'
import { validRole } from '@/router/basic'
import type { MenuOption } from 'naive-ui'
import type { RouteMeta } from 'vue-router'
const GlobalSeach = defineComponent({
name: 'GlobalSeach',
props: {
show: {
type: Boolean,
default: false,
},
},
emits: ['update:show'],
setup(props, { emit }) {
const menuStore = useMenu()
const { menuModelValueChange } = menuStore
const modelShow = computed({
get: () => props.show,
set: (val) => {
emit('update:show', val)
if (!val) {
state.searchOptions = []
state.searchValue = null
}
},
})
const modelMenuOptions = computed(() => menuStore.options)
const state = reactive({
searchValue: null,
searchOptions: [] as IMenuOptions[],
})
const tiptextOptions = [
{
icon: 'cmd / ctrl + k',
label: '唤起',
plain: true,
},
{
icon: 'esc',
label: '关闭',
plain: true,
},
]
/** 按下 ctrl + k 或者 command + k 激活搜索栏 */
const registerKeyboard = (e: Event) => {
const _e = e as KeyboardEvent
if ((_e.ctrlKey || _e.metaKey) && _e.key === 'k') {
modelShow.value = true
}
}
/** 根据输入值模糊检索菜单 */
const handleSearchMenuOptions = (value: string) => {
const arr: IMenuOptions[] = []
const filterArr = (options: IMenuOptions[]) => {
options.forEach((curr) => {
if (curr.children?.length) {
filterArr(curr.children)
}
/** 处理菜单名与输入值, 不区分大小写 */
const _breadcrumbLabel = curr.breadcrumbLabel?.toLocaleLowerCase()
const _value = String(value).toLocaleLowerCase()
if (
_breadcrumbLabel?.includes(_value) &&
validRole(curr) &&
!curr.children?.length
) {
arr.push(curr)
}
})
}
if (value) {
filterArr(modelMenuOptions.value)
state.searchOptions = arr
} else {
state.searchOptions = []
}
}
const handleSearchItemClick = (option: MenuOption) => {
const meta = option.meta as RouteMeta
/** 如果配置站外跳转则不会关闭搜索框 */
if (meta.windowOpen) {
window.open(meta.windowOpen)
} else {
modelShow.value = false
menuModelValueChange(option.key as string, option)
}
}
onMounted(() => {
on(window, 'keydown', registerKeyboard)
})
onBeforeUnmount(() => {
off(window, 'keydown', registerKeyboard)
})
return {
...toRefs(state),
modelShow,
tiptextOptions,
handleSearchMenuOptions: debounce(handleSearchMenuOptions, 300),
handleSearchItemClick,
}
},
render() {
return (
<NModal v-model:show={this.modelShow} transform-origin="center" show>
<div class="global-seach">
<div class="global-seach__wrapper">
<div class="global-seach__card">
<div class="global-seach__card-header">
<NInput
size="large"
v-model:value={this.searchValue}
onInput={this.handleSearchMenuOptions.bind(this)}
>
{{
prefix: () => <RayIcon name="search" size="24" />,
}}
</NInput>
</div>
<NScrollbar class="global-seach__card-content">
<NSpace vertical wrapItem={false} size={[8, 8]}>
{this.searchOptions.map((curr) => (
<NSpace
align="center"
wrapItem={false}
class="content-item"
onClick={this.handleSearchItemClick.bind(this, curr)}
>
<div class="content-item-icon">
{curr?.meta?.icon ? (
<RayIcon name={curr.meta.icon} size="24" />
) : (
<RayIcon name="table" size="24" />
)}
</div>
<div class="content-item-label">
{curr.breadcrumbLabel}
</div>
</NSpace>
))}
</NSpace>
</NScrollbar>
<div class="global-seach__card-footer">
<NSpace
class="card-footer__tip-wrapper"
align="center"
wrapItem={false}
size={[24, 8]}
>
{this.tiptextOptions.map((curr) => (
<div class="tip-wrapper-item">
<div class="item-icon">
{curr.plain ? (
<span>{curr.icon}</span>
) : (
<RayIcon name={curr.icon} size="18" />
)}
</div>
<div class="item-laebl">{curr.label}</div>
</div>
))}
</NSpace>
</div>
</div>
</div>
</div>
</NModal>
)
},
})
export default GlobalSeach

View File

@ -43,6 +43,7 @@ const SettingDrawer = defineComponent({
primaryColorOverride, primaryColorOverride,
menuTagSwitch, menuTagSwitch,
breadcrumbSwitch, breadcrumbSwitch,
invertSwitch,
} = storeToRefs(settingStore) } = storeToRefs(settingStore)
const modelShow = computed({ const modelShow = computed({
@ -66,6 +67,7 @@ const SettingDrawer = defineComponent({
menuTagSwitch, menuTagSwitch,
changeSwitcher, changeSwitcher,
breadcrumbSwitch, breadcrumbSwitch,
invertSwitch,
} }
}, },
render() { render() {
@ -91,9 +93,11 @@ const SettingDrawer = defineComponent({
v-model:value={this.primaryColorOverride.common!.primaryColor} v-model:value={this.primaryColorOverride.common!.primaryColor}
onUpdateValue={this.changePrimaryColor.bind(this)} onUpdateValue={this.changePrimaryColor.bind(this)}
/> />
<NDivider titlePlacement="center"></NDivider> <NDivider titlePlacement="center">
{t('LayoutHeaderSettingOptions.InterfaceDisplay')}
</NDivider>
<NDescriptions labelPlacement="left" column={1}> <NDescriptions labelPlacement="left" column={1}>
<NDescriptionsItem label="显示多标签"> <NDescriptionsItem label="多标签">
<NSwitch <NSwitch
v-model:value={this.menuTagSwitch} v-model:value={this.menuTagSwitch}
onUpdateValue={(bool: boolean) => onUpdateValue={(bool: boolean) =>
@ -101,7 +105,7 @@ const SettingDrawer = defineComponent({
} }
/> />
</NDescriptionsItem> </NDescriptionsItem>
<NDescriptionsItem label="显示面包屑"> <NDescriptionsItem label="面包屑">
<NSwitch <NSwitch
v-model:value={this.breadcrumbSwitch} v-model:value={this.breadcrumbSwitch}
onUpdateValue={(bool: boolean) => onUpdateValue={(bool: boolean) =>
@ -109,6 +113,14 @@ const SettingDrawer = defineComponent({
} }
/> />
</NDescriptionsItem> </NDescriptionsItem>
<NDescriptionsItem label="反转色">
<NSwitch
v-model:value={this.invertSwitch}
onUpdateValue={(bool: boolean) =>
this.changeSwitcher(bool, 'invertSwitch')
}
/>
</NDescriptionsItem>
</NDescriptions> </NDescriptions>
</NSpace> </NSpace>
</NDrawerContent> </NDrawerContent>

View File

@ -16,6 +16,7 @@ import RayIcon from '@/components/RayIcon/index'
import RayTooltipIcon from '@/components/RayTooltipIcon/index' import RayTooltipIcon from '@/components/RayTooltipIcon/index'
import SettingDrawer from './components/SettingDrawer/index' import SettingDrawer from './components/SettingDrawer/index'
import Breadcrumb from './components/Breadcrumb/index' import Breadcrumb from './components/Breadcrumb/index'
import GlobalSeach from './components/GlobalSeach/index'
import { useSetting } from '@/store' import { useSetting } from '@/store'
import { useSignin } from '@/store' import { useSignin } from '@/store'
@ -42,12 +43,14 @@ const SiderBar = defineComponent({
const { t } = useI18n() const { t } = useI18n()
const { updateLocale, changeSwitcher } = settingStore const { updateLocale, changeSwitcher } = settingStore
const { logout } = signinStore const { logout } = signinStore
const { drawerPlacement, breadcrumbSwitch } = storeToRefs(settingStore) const { drawerPlacement, breadcrumbSwitch } = storeToRefs(settingStore)
const showSettings = ref(false) const showSettings = ref(false)
const person = getCache('person') const person = getCache('person')
const spaceItemStyle = { const spaceItemStyle = {
display: 'flex', display: 'flex',
} }
const globalSearchShown = ref(false)
/** /**
* *
@ -65,6 +68,12 @@ const SiderBar = defineComponent({
* *
*/ */
const rightTooltipIconOptions = [ const rightTooltipIconOptions = [
{
name: 'search',
size: 18,
tooltip: 'LayoutHeaderTooltipOptions.Search',
eventKey: 'search',
},
{ {
name: 'fullscreen', name: 'fullscreen',
size: 18, size: 18,
@ -103,6 +112,9 @@ const SiderBar = defineComponent({
window.$message.warning('您的浏览器不支持全屏~') window.$message.warning('您的浏览器不支持全屏~')
} }
}, },
search: () => {
globalSearchShown.value = true
},
} }
const handleIconClick = (key: IconEventMap) => { const handleIconClick = (key: IconEventMap) => {
@ -137,11 +149,13 @@ const SiderBar = defineComponent({
spaceItemStyle, spaceItemStyle,
drawerPlacement, drawerPlacement,
breadcrumbSwitch, breadcrumbSwitch,
globalSearchShown,
} }
}, },
render() { render() {
return ( return (
<NLayoutHeader class="layout-header" bordered> <NLayoutHeader class="layout-header" bordered>
<GlobalSeach v-model:show={this.globalSearchShown} />
<NSpace <NSpace
class="layout-header__method" class="layout-header__method"
align="center" align="center"

View File

@ -8,6 +8,14 @@
& .layout-content__router-view { & .layout-content__router-view {
height: var(--layout-content-height); height: var(--layout-content-height);
padding: calc($layoutRouterViewContainer / 2); padding: calc($layoutRouterViewContainer / 2);
& .n-scrollbar-container {
height: 100%;
& .n-scrollbar-content {
height: 100%;
}
}
} }
& .layout-footer { & .layout-footer {

View File

@ -9,6 +9,23 @@
* @remark * @remark
*/ */
/**
*
*
*
* Naive UI Spin
*
* 使
* 1. import { useSpin } from '@/spin'
* 2. useSpin(true) | useSpin(false)
*
*
*
*
* 1. ,
* 2. 使
*/
import { NSpin } from 'naive-ui' import { NSpin } from 'naive-ui'
import { spinProps } from 'naive-ui' import { spinProps } from 'naive-ui'
@ -38,30 +55,10 @@ const GlobalSpin = defineComponent({
show={this.spinValue} show={this.spinValue}
themeOverrides={this.overrides} themeOverrides={this.overrides}
> >
{{ {{ ...this.$slots }}
default: () => this.$slots.default?.(),
description: () => 'loading...',
}}
</NSpin> </NSpin>
) )
}, },
}) })
export default GlobalSpin export default GlobalSpin
/**
*
*
*
* Naive UI Spin
*
* 使
* 1. import { useSpin } from '@/spin'
* 2. useSpin(true) | useSpin(false)
*
*
*
*
* 1. ,
* 2. 使
*/

View File

@ -167,59 +167,67 @@ export const useMenu = defineStore(
/** 取出所有 layout 下子路由 */ /** 取出所有 layout 下子路由 */
const layout = router.getRoutes().find((route) => route.name === 'layout') const layout = router.getRoutes().find((route) => route.name === 'layout')
const resolveOption = (option: IMenuOptions) => {
const { meta } = option
/** 设置 label, i18nKey 优先级最高 */
const label = computed(() =>
meta?.i18nKey
? t(`GlobalMenuOptions.${meta!.i18nKey}`)
: meta?.noLocalTitle,
)
/** 拼装菜单项 */
const route = {
...option,
key: option.path,
label: () =>
h(NEllipsis, null, {
default: () => label.value,
}),
breadcrumbLabel: label.value,
} as IMenuOptions
/** 是否有 icon */
const expandIcon = {
icon: () =>
h(
RayIcon,
{
name: meta!.icon as string,
size: 20,
},
{},
),
}
const attr: IMenuOptions = meta?.icon
? Object.assign({}, route, expandIcon)
: route
if (option.path === cacheMenuKey) {
/** 设置菜单标签 */
setMenuTagOptions(attr)
/** 设置浏览器标题 */
updateDocumentTitle(attr)
}
attr.show = validRole(option)
return attr
}
const resolveRoutes = (routes: IMenuOptions[], index: number) => { const resolveRoutes = (routes: IMenuOptions[], index: number) => {
return routes.map((curr) => { const catchArr: IMenuOptions[] = []
if (curr.children?.length) {
for (const curr of routes) {
if (curr.children?.length && validRole(curr)) {
curr.children = resolveRoutes(curr.children, index++) curr.children = resolveRoutes(curr.children, index++)
} else if (!validRole(curr)) {
continue
} }
const { meta } = curr catchArr.push(resolveOption(curr))
/** 设置 label, i18nKey 优先级最高 */ }
const label = computed(() =>
meta?.i18nKey
? t(`GlobalMenuOptions.${meta!.i18nKey}`)
: meta?.noLocalTitle,
)
/** 拼装菜单项 */ return catchArr
const route = {
...curr,
key: curr.path,
label: () =>
h(NEllipsis, null, {
default: () => label.value,
}),
breadcrumbLabel: label.value,
} as IMenuOptions
/** 是否有 icon */
const expandIcon = {
icon: () =>
h(
RayIcon,
{
name: meta!.icon as string,
size: 20,
},
{},
),
}
const attr: IMenuOptions = meta?.icon
? Object.assign({}, route, expandIcon)
: route
if (curr.path === cacheMenuKey) {
/** 设置菜单标签 */
setMenuTagOptions(attr)
/** 设置浏览器标题 */
updateDocumentTitle(attr)
}
attr.show = validRole(curr)
return attr
})
} }
/** 缓存菜单列表 */ /** 缓存菜单列表 */

View File

@ -1,6 +1,7 @@
import { getDefaultLocal } from '@/language/index' import { getDefaultLocal } from '@/language/index'
import { setCache } from '@use-utils/cache' import { setCache } from '@use-utils/cache'
import { set } from 'lodash-es' import { set } from 'lodash-es'
import { addClass, removeClass, colorToRgba } from '@/utils/element'
import type { ConditionalPick } from '@/types/type-utils' import type { ConditionalPick } from '@/types/type-utils'
import type { GlobalThemeOverrides } from 'naive-ui' import type { GlobalThemeOverrides } from 'naive-ui'
@ -14,12 +15,15 @@ interface SettingState {
spinSwitch: boolean spinSwitch: boolean
breadcrumbSwitch: boolean breadcrumbSwitch: boolean
localeLanguage: string localeLanguage: string
invertSwitch: boolean
} }
export const useSetting = defineStore( export const useSetting = defineStore(
'setting', 'setting',
() => { () => {
const { primaryColor } = __APP_CFG__ const {
appPrimaryColor: { primaryColor },
} = __APP_CFG__ // 默认主题色
const { locale } = useI18n() const { locale } = useI18n()
const settingState = reactive<SettingState>({ const settingState = reactive<SettingState>({
@ -34,6 +38,7 @@ export const useSetting = defineStore(
reloadRouteSwitch: true, // 刷新路由开关 reloadRouteSwitch: true, // 刷新路由开关
menuTagSwitch: true, // 多标签页开关 menuTagSwitch: true, // 多标签页开关
spinSwitch: false, // 全屏加载 spinSwitch: false, // 全屏加载
invertSwitch: false, // 反转色模式
breadcrumbSwitch: true, // 面包屑开关 breadcrumbSwitch: true, // 面包屑开关
localeLanguage: getDefaultLocal(), localeLanguage: getDefaultLocal(),
}) })
@ -58,6 +63,10 @@ export const useSetting = defineStore(
/** 设置主题色变量 */ /** 设置主题色变量 */
body.style.setProperty('--ray-theme-primary-color', value) body.style.setProperty('--ray-theme-primary-color', value)
body.style.setProperty(
'--ray-theme-primary-fade-color',
colorToRgba(value, 0.25),
)
} }
/** /**
@ -79,6 +88,17 @@ export const useSetting = defineStore(
} }
} }
/** 动态添加反转色 class name */
watch(
() => settingState.invertSwitch,
(newData) => {
const body = document.body
const className = 'ray-template--invert'
newData ? addClass(body, className) : removeClass(body, className)
},
)
return { return {
...toRefs(settingState), ...toRefs(settingState),
updateLocale, updateLocale,

View File

@ -49,3 +49,7 @@ body {
-webkit-font-smoothing: antialiased; -webkit-font-smoothing: antialiased;
-moz-osx-font-smoothing: grayscale; -moz-osx-font-smoothing: grayscale;
} }
body.ray-template--invert {
filter: invert(1);
}

View File

@ -6,6 +6,5 @@
$iconSpace: 5px; $iconSpace: 5px;
$width: 140px; $width: 140px;
$activedColor: #2d8cf0;
$hoverLightBackgroundColor: rgba(45, 140, 240, 0.1); $hoverLightBackgroundColor: rgba(45, 140, 240, 0.1);
$hoverDarkBackgroundColor: rgba(45, 140, 240, 0.15); $hoverDarkBackgroundColor: rgba(45, 140, 240, 0.15);

View File

@ -31,6 +31,11 @@ export interface PreloadingConfig {
titleColor?: string titleColor?: string
} }
export interface AppPrimaryColor {
primaryColor: string
primaryFadeColor: string
}
export interface Config { export interface Config {
server: ServerOptions server: ServerOptions
buildOptions: (mode: string) => BuildOptions buildOptions: (mode: string) => BuildOptions
@ -40,9 +45,9 @@ export interface Config {
sideBarLogo?: LayoutSideBarLogo sideBarLogo?: LayoutSideBarLogo
mixinCSS?: string mixinCSS?: string
rootRoute?: RootRoute rootRoute?: RootRoute
primaryColor?: string
preloadingConfig?: PreloadingConfig preloadingConfig?: PreloadingConfig
base?: string base?: string
appPrimaryColor?: AppPrimaryColor
} }
export type Recordable<T = unknown> = Record<string, T> export type Recordable<T = unknown> = Record<string, T>
@ -68,6 +73,7 @@ export interface AppConfig {
rootRoute: RootRoute rootRoute: RootRoute
primaryColor: string primaryColor: string
base?: string base?: string
appPrimaryColor: AppPrimaryColor
} }
export type AppConfigExport = Config & UserConfigExport export type AppConfigExport = Config & UserConfigExport

View File

@ -174,3 +174,43 @@ export const removeStyle = (el: HTMLElement, styles: string[]) => {
}) })
} }
} }
/**
*
* @param color
* @param alpha
* @returns rgba
*
* @remark rgba
*/
export const colorToRgba = (color: string, alpha = 1) => {
const hexPattern = /^#([0-9a-f]{3}|[0-9a-f]{6})$/i
const rgbPattern = /^rgb\((\d{1,3}),\s*(\d{1,3}),\s*(\d{1,3})\)$/i
const rgbaPattern =
/^rgba\((\d{1,3}),\s*(\d{1,3}),\s*(\d{1,3}),\s*(\d*(?:\.\d+)?)\)$/i
let result: string
if (hexPattern.test(color)) {
const hex = color.substring(1)
const rgb = [
parseInt(hex.substring(0, 2), 16),
parseInt(hex.substring(2, 4), 16),
parseInt(hex.substring(4, 6), 16),
]
result = 'rgb(' + rgb.join(', ') + ')'
} else if (rgbPattern.test(color)) {
result = color
} else if (rgbaPattern.test(color)) {
result = color
} else {
result = ''
}
if (result && !result.startsWith('rgba')) {
result = result.replace('rgb', 'rgba').replace(')', `, ${alpha})`)
}
return result
}

View File

@ -1,18 +1,6 @@
import './index.scss' import './index.scss'
import { import { NCard, NSwitch, NSpace, NP, NH6, NH2, NH3 } from 'naive-ui'
NCard,
NSwitch,
NLayout,
NDescriptions,
NDescriptionsItem,
NTag,
NSpace,
NP,
NH6,
NH2,
NH3,
} from 'naive-ui'
import RayChart from '@/components/RayChart/index' import RayChart from '@/components/RayChart/index'
const Echart = defineComponent({ const Echart = defineComponent({
@ -21,6 +9,9 @@ const Echart = defineComponent({
const baseChartRef = ref() const baseChartRef = ref()
const chartLoading = ref(false) const chartLoading = ref(false)
const chartAria = ref(false) const chartAria = ref(false)
const state = reactive({
loading: false,
})
const baseOptions = { const baseOptions = {
legend: {}, legend: {},
@ -177,11 +168,7 @@ const Echart = defineComponent({
} }
const handleLoadingShow = (bool: boolean) => { const handleLoadingShow = (bool: boolean) => {
if (baseChartRef.value) { state.loading = bool
const { echartInstance } = baseChartRef.value
bool ? echartInstance.showLoading() : echartInstance.hideLoading()
}
} }
const handleAriaShow = (bool: boolean) => { const handleAriaShow = (bool: boolean) => {
@ -208,6 +195,7 @@ const Echart = defineComponent({
handleChartRenderSuccess, handleChartRenderSuccess,
basePieOptions, basePieOptions,
baseLineOptions, baseLineOptions,
...toRefs(state),
} }
}, },
render() { render() {
@ -254,7 +242,7 @@ const Echart = defineComponent({
}} }}
</NSwitch> </NSwitch>
<div class="chart--container"> <div class="chart--container">
<RayChart ref="baseChartRef" options={this.baseOptions} /> <RayChart loading={this.loading} options={this.baseOptions} />
</div> </div>
<NH2></NH2> <NH2></NH2>
<NSwitch <NSwitch

View File

@ -33,7 +33,7 @@ const {
sideBarLogo, sideBarLogo,
mixinCSS, mixinCSS,
rootRoute, rootRoute,
primaryColor, appPrimaryColor,
preloadingConfig, preloadingConfig,
base, base,
} = config } = config
@ -55,7 +55,7 @@ const __APP_CFG__ = {
sideBarLogo, sideBarLogo,
}, },
rootRoute, rootRoute,
primaryColor, appPrimaryColor,
} }
// https://vitejs.dev/config/ // https://vitejs.dev/config/
@ -132,6 +132,7 @@ export default defineConfig(async ({ mode }) => {
}), }),
ViteEjsPlugin({ ViteEjsPlugin({
preloadingConfig, preloadingConfig,
appPrimaryColor,
}), }),
], ],
optimizeDeps: { optimizeDeps: {