This commit is contained in:
XiaoDaiGua-Ray 2023-10-25 17:27:50 +08:00
parent b05edfeaeb
commit f6c343911e
27 changed files with 224 additions and 161 deletions

View File

@ -1,5 +1,19 @@
# CHANGE LOG
## 4.2.7
主要是做了一些统一命名的事情,以前由于写的比较放浪形骸现在正在慢慢更改这个大问题。
### Feats
- 优化顶部操作栏的渲染逻辑,现在将更加合理的管理数据渲染图标
- 二次封装 useFullscreen 方法,如果当前环境不支持全屏则会自动弹出提示
- 重命名 utils/hook 包名为 basic
- 移除 vuedraggable 插件
- RequestCanceler 方法部分属性与方法变为私有
- 重命名 layout 包下的一些组件 name
- 升级 echarts 至 5.4.3 版本
## 4.2.6
### Feats

View File

@ -1,7 +1,7 @@
{
"name": "ray-template",
"private": false,
"version": "4.2.6",
"version": "4.2.7",
"type": "module",
"engines": {
"node": ">=16.0.0",
@ -31,7 +31,7 @@
"crypto-js": "^4.1.1",
"currency.js": "^2.0.4",
"dayjs": "^1.11.7",
"echarts": "^5.4.0",
"echarts": "^5.4.3",
"lodash-es": "^4.17.21",
"mockjs": "1.1.0",
"naive-ui": "^2.35.0",
@ -42,7 +42,6 @@
"vue-hooks-plus": "1.8.5",
"vue-i18n": "^9.2.2",
"vue-router": "^4.2.4",
"vuedraggable": "^4.1.0",
"xlsx": "^0.18.5"
},
"devDependencies": {

View File

@ -19,14 +19,14 @@
import type { AppRawRequestConfig } from '@/axios/type'
export default class RequestCanceler {
pendingRequest: Map<string, AbortController>
private pendingRequest: Map<string, AbortController>
constructor() {
this.pendingRequest = new Map<string, AbortController>()
}
/** 是否需要加入取消请求表中 */
isApending(config: AppRawRequestConfig) {
private isApending(config: AppRawRequestConfig) {
return config.cancelConfig?.needCancel ?? true
}
@ -37,7 +37,7 @@ export default class RequestCanceler {
*
* @remark config request key
*/
generateRequestKey(config: AppRawRequestConfig) {
private generateRequestKey(config: AppRawRequestConfig) {
const { method, url } = config
return [

View File

@ -21,7 +21,7 @@
*/
import RequestCanceler from '@/axios/helper/RequestCanceler'
import { getAppEnvironment } from '@use-utils/hook'
import { getAppEnvironment } from '@/utils/basic'
import type {
RequestInterceptorConfig,

View File

@ -16,7 +16,7 @@ import RIcon from '@/components/RIcon/index'
import props from './props'
import { AwesomeQR } from 'awesome-qr'
import { isValueType, downloadAnyFile } from '@use-utils/hook'
import { isValueType, downloadAnyFile } from '@/utils/basic'
import { call } from '@/utils/vue/index'
import type { QRCodeRenderResponse, GIFBuffer } from './type'

View File

@ -19,7 +19,7 @@ import Print from './components/Print'
import props from './props'
import { call } from '@/utils/vue/index'
import { uuid } from '@use-utils/hook'
import { uuid } from '@/utils/basic'
import config from './config'
import type { DropdownOption, DataTableInst } from 'naive-ui'

View File

@ -13,7 +13,7 @@ import { NPopover } from 'naive-ui'
import RIcon from '@/components/RIcon/index'
import config from '../config'
import { useFullscreen } from 'vue-hooks-plus'
import { useFullscreen } from '@/hooks/web/index'
import type { TableProvider } from '../type'

View File

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

View File

@ -0,0 +1,40 @@
/**
*
* @author Ray <https://github.com/XiaoDaiGua-Ray>
*
* @date 2023-10-25
*
* @workspace ray-template
*
* @remark
*/
import { useFullscreen as hooksPlusUseFullscreen } from 'vue-hooks-plus'
import type { useFullscreen as UseFullscreen } from 'vue-hooks-plus'
type UseFullscreenParams = Parameters<typeof UseFullscreen>
export function useFullscreen(
target: UseFullscreenParams[0],
options?: UseFullscreenParams[1],
) {
const [
isFullscreen,
{ enterFullscreen, exitFullscreen, toggleFullscreen, isEnabled },
] = hooksPlusUseFullscreen(target, options)
if (!isEnabled) {
window.$message.warning('您当前环境不支持全屏模式')
}
return [
isFullscreen,
{
enterFullscreen,
exitFullscreen,
toggleFullscreen,
isEnabled,
},
] as const
}

View File

@ -23,8 +23,8 @@ import type { MenuInst } from 'naive-ui'
import type { NaiveMenuOptions } from '@/types/modules/component'
import type { AppMenuOption } from '@/types/modules/app'
const LayoutMenu = defineComponent({
name: 'LayoutMenu',
export default defineComponent({
name: 'AppMenu',
setup() {
const menuRef = ref<MenuInst | null>(null)
@ -114,5 +114,3 @@ const LayoutMenu = defineComponent({
)
},
})
export default LayoutMenu

View File

@ -30,7 +30,7 @@ import RIcon from '@/components/RIcon/index'
import RMoreDropdown from '@/components/RMoreDropdown/index'
import { useMenu, useSetting } from '@/store'
import { uuid } from '@/utils/hook'
import { uuid } from '@/utils/basic'
import { hasClass } from '@/utils/element'
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { ROOT_ROUTE } from '@/app-config/appConfig'
@ -40,7 +40,7 @@ import type { MenuOption, ScrollbarInst } from 'naive-ui'
import type { MenuTagOptions, AppMenuOption } from '@/types/modules/app'
export default defineComponent({
name: 'MenuTag',
name: 'AppMenuTag',
setup(_, { expose }) {
const scrollRef = ref<ScrollbarInst | null>(null)
@ -269,7 +269,7 @@ export default defineComponent({
return
}
const handleScrollX = (type: 'left' | 'right') => {
const scrollX = (type: 'left' | 'right') => {
const el = getScrollElement()
if (el) {
@ -438,7 +438,7 @@ export default defineComponent({
menuKey,
handleTagClick,
moreOptions,
handleScrollX,
scrollX,
scrollRef,
scrollBarUUID,
actionDropdownSelect,
@ -480,7 +480,7 @@ export default defineComponent({
width="20"
height="28"
customClassName="menu-tag__left-arrow"
onClick={this.handleScrollX.bind(this, 'left')}
onClick={this.scrollX.bind(this, 'left')}
/>
<NScrollbar
xScrollable
@ -524,7 +524,7 @@ export default defineComponent({
width="20"
height="28"
customClassName="menu-tag__right-arrow"
onClick={this.handleScrollX.bind(this, 'right')}
onClick={this.scrollX.bind(this, 'right')}
/>
<RMoreDropdown
options={this.moreOptions}

View File

@ -16,7 +16,7 @@ import RIcon from '@/components/RIcon/index'
import { tooltipProps } from 'naive-ui'
const TooltipIcon = defineComponent({
export default defineComponent({
name: 'TooltipIcon',
props: {
...tooltipProps,
@ -34,32 +34,38 @@ const TooltipIcon = defineComponent({
},
},
emits: ['click'],
setup(_, { emit }) {
const handleClick = (e?: MouseEvent) => {
setup(props, { emit }) {
const iconClick = (e?: MouseEvent) => {
emit('click', e)
}
const Icon = () => (
<RIcon
name={props.iconName}
size="18"
customClassName={`tooltip-text__icon ${props.customClassName}`}
cursor="pointer"
onClick={iconClick.bind(this)}
/>
)
return {
handleClick,
iconClick,
Icon,
}
},
render() {
return (
const { Icon } = this
return this.tooltipText ? (
<NTooltip {...this.$props}>
{{
trigger: () => (
<RIcon
name={this.iconName}
size="18"
customClassName={`tooltip-text__icon ${this.customClassName}`}
cursor="pointer"
onClick={this.handleClick.bind(this)}
/>
),
trigger: () => <Icon />,
default: () => this.tooltipText,
}}
</NTooltip>
) : (
<Icon />
)
},
})
export default TooltipIcon

View File

@ -1,6 +1,9 @@
import { useSetting, useSignin } from '@/store'
import { useI18n } from '@/hooks/web/index'
export const useAvatarOptions = () => [
import type { IconOptionsFC, IconOptions } from './type'
export const createAvatarOptions = () => [
{
key: 'person',
label: '个人信息',
@ -47,3 +50,71 @@ export const avatarDropdownClick = (key: string | number) => {
action ? action() : window.$message.info('这个人很懒, 没做这个功能~')
}
export const createLeftIconOptions = (opts: IconOptionsFC) => {
const { isTabletOrSmaller, reloadRouteSwitch } = opts
const { t } = useI18n()
const notTableOrSmallerOptions: IconOptions[] = [
{
name: 'reload',
size: 18,
tooltip: t('headerTooltip.Reload'),
iconClass: !reloadRouteSwitch.value ? 'ray-icon__reload--loading' : '',
eventKey: 'reload',
},
]
const tableOrSmallerOptions: IconOptions[] = [
{
name: 'menu',
size: 18,
eventKey: 'menu',
},
]
return isTabletOrSmaller!.value
? tableOrSmallerOptions
: notTableOrSmallerOptions
}
export const createRightIconOptions = (opts: IconOptionsFC) => {
const { isFullscreen, isTabletOrSmaller } = opts
const { t } = useI18n()
const basicOptions: IconOptions[] = [
{
name: 'fullscreen',
size: 18,
tooltip: 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 notTableOrSmallerOptions: IconOptions[] = [
{
name: 'search',
size: 18,
tooltip: t('headerTooltip.Search'),
eventKey: 'search',
},
...basicOptions,
]
const tableOrSmallerOptions: IconOptions[] = [...basicOptions]
return isTabletOrSmaller!.value
? tableOrSmallerOptions
: notTableOrSmallerOptions
}

View File

@ -19,7 +19,7 @@
import './index.scss'
import { NLayoutHeader, NSpace, NTooltip, NDropdown } from 'naive-ui'
import { NLayoutHeader, NSpace, NDropdown } from 'naive-ui'
import RIcon from '@/components/RIcon/index'
import TootipIcon from '@/layout/components/SiderBar/components/TooltipIcon/index'
import SettingDrawer from './components/SettingDrawer/index'
@ -29,20 +29,22 @@ import AppAvatar from '@/app-components/app/AppAvatar/index'
import { useSetting } from '@/store'
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 {
createAvatarOptions,
avatarDropdownClick,
createLeftIconOptions,
createRightIconOptions,
} from './hook'
import { useDevice, useFullscreen } from '@/hooks/web/index'
import { globalVariableToRefs, setVariable } from '@/hooks/variable/index'
import type { LeftIconOptions, IconEventMapOptions, IconEventMap } from './type'
import type { IconEventMapOptions, IconEventMap } from './type'
const SiderBar = defineComponent({
name: 'SiderBar',
export default defineComponent({
name: 'AppSiderBar',
setup() {
const settingStore = useSetting()
const { t } = useI18n()
const { updateLocale, changeSwitcher } = settingStore
const [isFullscreen, { toggleFullscreen }] = useFullscreen(
@ -62,69 +64,24 @@ const SiderBar = defineComponent({
*
*
*/
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 leftIconOptions = computed(() =>
createLeftIconOptions({
isFullscreen,
isTabletOrSmaller,
reloadRouteSwitch,
}),
)
/**
*
*
*/
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 rightTooltipIconOptions = computed(() =>
createRightIconOptions({
isFullscreen,
isTabletOrSmaller,
reloadRouteSwitch,
}),
)
const iconEventMap: IconEventMapOptions = {
// 刷新组件重新加载,手动设置 800ms loading 时长
reload: () => {
@ -156,24 +113,9 @@ const SiderBar = defineComponent({
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 {
leftIconOptions,
rightTooltipIconOptions,
t,
toolIconClick,
showSettings,
updateLocale,
@ -181,12 +123,9 @@ const SiderBar = defineComponent({
drawerPlacement,
breadcrumbSwitch,
globalSearchShown,
LeftToolIcon,
}
},
render() {
const { LeftToolIcon } = this
return (
<NLayoutHeader class="layout-header" bordered>
<GlobalSeach v-model:show={this.globalSearchShown} />
@ -200,18 +139,16 @@ const SiderBar = defineComponent({
wrapItem={false}
itemStyle={this.spaceItemStyle}
>
{this.leftIconOptions.map((curr) =>
curr.tooltip ? (
<NTooltip>
{{
trigger: () => <LeftToolIcon {...curr} />,
default: () => curr.tooltip,
}}
</NTooltip>
) : (
<LeftToolIcon {...curr} />
),
)}
{this.leftIconOptions.map((curr) => (
<TootipIcon
iconName={curr.name}
tooltipText={
isRef(curr.tooltip) ? curr.tooltip.value : curr.tooltip
}
customClassName={curr.iconClass}
onClick={this.toolIconClick.bind(this, curr.name)}
/>
))}
{this.breadcrumbSwitch ? <Breadcrumb /> : null}
</NSpace>
<NSpace
@ -225,6 +162,7 @@ const SiderBar = defineComponent({
tooltipText={
isRef(curr.tooltip) ? curr.tooltip.value : curr.tooltip
}
customClassName={curr.iconClass}
onClick={this.toolIconClick.bind(this, curr.name)}
/>
))}
@ -243,7 +181,7 @@ const SiderBar = defineComponent({
/>
</NDropdown>
<NDropdown
options={useAvatarOptions()}
options={createAvatarOptions()}
onSelect={avatarDropdownClick.bind(this)}
trigger="click"
>
@ -259,5 +197,3 @@ const SiderBar = defineComponent({
)
},
})
export default SiderBar

View File

@ -1,5 +1,5 @@
import type { DropdownOption } from 'naive-ui'
import type { ComputedRef } from 'vue'
import type { ComputedRef, Ref } from 'vue'
export interface IconEventMapOptions {
[propName: string]: (...args: unknown[]) => unknown
@ -17,14 +17,14 @@ export interface IconDropdownOptions extends UnknownObjectKey {
export interface IconOptions {
name: string
size?: number
tooltip?: string
tooltip?: ComputedRef<string> | string
eventKey?: string
dropdown?: IconDropdownOptions
iconClass?: string
}
export interface LeftIconOptions {
name: string
size: number
tooltip?: string
iconClass?: ComputedRef
export interface IconOptionsFC {
isTabletOrSmaller: Ref<boolean>
isFullscreen: Ref<boolean>
reloadRouteSwitch: Ref<boolean>
}

View File

@ -26,7 +26,7 @@ import { useSetting } from '@/store'
import type { GlobalThemeOverrides } from 'naive-ui'
const ContentWrapper = defineComponent({
name: 'ContentWrapper',
name: 'LayoutContentWrapper',
setup() {
const settingStore = useSetting()
const router = useRouter()

View File

@ -12,7 +12,7 @@
import MenuTag from '@/layout/components/MenuTag/index'
const FeatureWrapper = defineComponent({
name: 'FeatureWrapper',
name: 'LayoutFeatureWrapper',
setup() {
return {}
},

View File

@ -12,7 +12,7 @@
import './index.scss'
const FooterWrapper = defineComponent({
name: 'FooterWrapper',
name: 'LayoutFooterWrapper',
setup() {
const {
layout: { copyright },

View File

@ -13,7 +13,7 @@ import { NSpace } from 'naive-ui'
import SiderBar from '@/layout/components/SiderBar/index'
const HeaderWrapper = defineComponent({
name: 'HeaderWrapper',
name: 'LayoutHeaderWrapper',
setup() {
return {}
},

View File

@ -15,8 +15,8 @@ import type { Ref } from 'vue'
/**
*
* SiderBar MenuTag , LayoutConetent
* MenuTag , 使 LayoutConetent
* SiderBar MenuTag , LayoutContent
* MenuTag , 使 LayoutContent
*
* vueuse useElementSize
* ,

View File

@ -71,11 +71,11 @@ export const combineI18nMessages = (langs: I18nModules, prefix: string) => {
/** 获取所有语言 */
export const getAppLocalMessages = async (
LOCAL_OPTIONS: AppLocalesDropdownMixedOption[],
localOptions: AppLocalesDropdownMixedOption[],
) => {
const message = {} as AppCurrentAppMessages
for (const curr of LOCAL_OPTIONS) {
for (const curr of localOptions) {
const msg: AppLocalesModules = await import(`./lang/${curr.key}.ts`)
const key = curr.key

View File

@ -25,7 +25,7 @@ import { APP_CATCH_KEY, ROOT_ROUTE } from '@/app-config/appConfig'
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
import { WHITE_ROUTES } from '@/app-config/routerConfig'
import { validRole } from '@/router/helper/routerCopilot'
import { isValueType } from '@/utils/hook'
import { isValueType } from '@/utils/basic'
import type { Router, RouteLocationNormalized } from 'vue-router'
import type { AppRouteMeta } from '@/router/type'

View File

@ -15,7 +15,7 @@ import { useSignin } from '@/store'
import { useVueRouter } from '@/hooks/web/index'
import { ROOT_ROUTE } from '@/app-config/appConfig'
import { setStorage } from '@/utils/cache'
import { getAppEnvironment } from '@/utils/hook'
import { getAppEnvironment } from '@/utils/basic'
import type { Router } from 'vue-router'
import type { AppRouteMeta } from '@/router/type'

View File

@ -13,10 +13,9 @@
import { APP_MENU_CONFIG, ROOT_ROUTE } from '@/app-config/appConfig'
import RIcon from '@/components/RIcon/index'
import { isValueType } from '@/utils/hook'
import { getStorage, setStorage } from '@/utils/cache'
import { isValueType } from '@/utils/basic'
import { getStorage } from '@/utils/cache'
import type { VNode } from 'vue'
import type {
AppMenuOption,
MenuTagOptions,

View File

@ -1,4 +1,4 @@
import { isValueType } from '@use-utils/hook'
import { isValueType } from '@/utils/basic'
import { APP_REGEX } from '@/app-config/regexConfig'
import { unrefElement } from '@/utils/vue/index'

View File

@ -170,7 +170,6 @@ export default function (mode: string): PluginOption[] {
'vue-i18n',
'dayjs',
'echarts',
'vuedraggable',
'xlsx',
'axios',
'screenfull',