mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-05 19:42:07 +08:00
update: 一些细节更新,新增路由切换内容区域加载动画
This commit is contained in:
parent
047c46f2a2
commit
76ec304ca7
@ -1,5 +1,10 @@
|
|||||||
node_modules/
|
node_modules/*
|
||||||
dist/
|
dist/*
|
||||||
yarn.lock
|
yarn.lock
|
||||||
yarn-error.log
|
yarn-error.log
|
||||||
visualizer.html
|
visualizer.html
|
||||||
|
pnpm-lock.yaml
|
||||||
|
.idea
|
||||||
|
auto-imports.d.ts
|
||||||
|
components.d.ts
|
||||||
|
visualizer.html
|
@ -5,6 +5,12 @@
|
|||||||
### Feats
|
### Feats
|
||||||
|
|
||||||
- 弃用 yarn 包管理器,使用 pnpm 作为模板包管理器
|
- 弃用 yarn 包管理器,使用 pnpm 作为模板包管理器
|
||||||
|
- 新增路由切换时的内容区域动画
|
||||||
|
|
||||||
|
### Fixes
|
||||||
|
|
||||||
|
- 修复 axios 模块的一些类型错误
|
||||||
|
- 剔除了一些无用方法,重写了一些方法
|
||||||
|
|
||||||
## 4.0.2
|
## 4.0.2
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"name": "ray-template",
|
"name": "ray-template",
|
||||||
"private": false,
|
"private": false,
|
||||||
"version": "4.0.3",
|
"version": "4.0.2",
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"dev": "vite",
|
"dev": "vite",
|
||||||
@ -102,9 +102,6 @@
|
|||||||
"admin template",
|
"admin template",
|
||||||
"中后台模板"
|
"中后台模板"
|
||||||
],
|
],
|
||||||
"files": [
|
|
||||||
"**/*"
|
|
||||||
],
|
|
||||||
"author": "Ray",
|
"author": "Ray",
|
||||||
"license": "ISC",
|
"license": "ISC",
|
||||||
"bugs": {
|
"bugs": {
|
||||||
|
@ -5,7 +5,6 @@ import type {
|
|||||||
HeadersDefaults,
|
HeadersDefaults,
|
||||||
AxiosDefaults,
|
AxiosDefaults,
|
||||||
Axios,
|
Axios,
|
||||||
InternalAxiosRequestConfig,
|
|
||||||
AxiosResponse,
|
AxiosResponse,
|
||||||
} from 'axios'
|
} from 'axios'
|
||||||
import type { AnyFunc } from '@/types/modules/utils'
|
import type { AnyFunc } from '@/types/modules/utils'
|
||||||
@ -80,7 +79,7 @@ export interface AxiosInstanceExpand extends Axios {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
export type RequestInterceptorConfig<T = any> = InternalAxiosRequestConfig<T>
|
export type RequestInterceptorConfig<T = any> = AxiosRequestConfig<T>
|
||||||
|
|
||||||
export type ResponseInterceptorConfig<T = any, K = any> = AxiosResponse<T, K>
|
export type ResponseInterceptorConfig<T = any, K = any> = AxiosResponse<T, K>
|
||||||
|
|
||||||
|
@ -19,7 +19,6 @@ import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVa
|
|||||||
import {
|
import {
|
||||||
rules,
|
rules,
|
||||||
useCondition,
|
useCondition,
|
||||||
autoFouceInput,
|
|
||||||
} from '@/components/AppComponents/AppLockScreen/hook'
|
} from '@/components/AppComponents/AppLockScreen/hook'
|
||||||
|
|
||||||
import type { FormInst, InputInst } from 'naive-ui'
|
import type { FormInst, InputInst } from 'naive-ui'
|
||||||
@ -49,7 +48,11 @@ const LockScreen = defineComponent({
|
|||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
autoFouceInput(inputInstRef)
|
onMounted(() => {
|
||||||
|
nextTick(() => {
|
||||||
|
inputInstRef.value?.focus()
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
return {
|
return {
|
||||||
...toRefs(state),
|
...toRefs(state),
|
||||||
|
@ -19,7 +19,6 @@ import { useSetting, useSignin } from '@/store'
|
|||||||
import {
|
import {
|
||||||
rules,
|
rules,
|
||||||
useCondition,
|
useCondition,
|
||||||
autoFouceInput,
|
|
||||||
} from '@/components/AppComponents/AppLockScreen/hook'
|
} from '@/components/AppComponents/AppLockScreen/hook'
|
||||||
import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVar'
|
import useAppLockScreen from '@/components/AppComponents/AppLockScreen/appLockVar'
|
||||||
|
|
||||||
|
@ -29,10 +29,3 @@ export const useCondition = () => {
|
|||||||
lockPassword: null,
|
lockPassword: null,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/** 自动获取焦点 */
|
|
||||||
export const autoFouceInput = (inputInstRef: Ref<InputInst | null>) => {
|
|
||||||
nextTick(() => {
|
|
||||||
inputInstRef.value?.focus()
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
@ -39,12 +39,13 @@ import { LabelLayout, UniversalTransition } from 'echarts/features' // 标签自
|
|||||||
import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
|
import { CanvasRenderer } from 'echarts/renderers' // `echarts` 渲染器
|
||||||
|
|
||||||
import { useSetting } from '@/store'
|
import { useSetting } from '@/store'
|
||||||
import { cloneDeep, debounce } from 'lodash-es'
|
import { cloneDeep, throttle } from 'lodash-es'
|
||||||
import { on, off, addStyle, completeSize } from '@/utils/element'
|
import { on, off, addStyle, completeSize } from '@/utils/element'
|
||||||
|
|
||||||
import type { PropType } from 'vue'
|
import type { PropType } from 'vue'
|
||||||
import type { EChartsInstance } from '@/types/modules/component'
|
import type { EChartsInstance } from '@/types/modules/component'
|
||||||
import type { AnyFunc } from '@/types/modules/utils'
|
import type { AnyFunc } from '@/types/modules/utils'
|
||||||
|
import type { DebouncedFunc } from 'lodash-es'
|
||||||
|
|
||||||
export type AutoResize =
|
export type AutoResize =
|
||||||
| boolean
|
| boolean
|
||||||
@ -125,6 +126,7 @@ const RayChart = defineComponent({
|
|||||||
* `chart` 是否跟随窗口尺寸变化自动变化
|
* `chart` 是否跟随窗口尺寸变化自动变化
|
||||||
*
|
*
|
||||||
* 如果为对象, 则可以指定其变化尺寸, 实现图表大小不等于容器大小的效果
|
* 如果为对象, 则可以指定其变化尺寸, 实现图表大小不等于容器大小的效果
|
||||||
|
* 默认每秒触发一次的频率
|
||||||
*/
|
*/
|
||||||
type: [Boolean, Object] as PropType<AutoResize>,
|
type: [Boolean, Object] as PropType<AutoResize>,
|
||||||
default: true,
|
default: true,
|
||||||
@ -195,10 +197,9 @@ const RayChart = defineComponent({
|
|||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* 拓展 `echarts` 图表
|
* 拓展 `echarts` 图表
|
||||||
*
|
* 用于自己手动拓展相关的包
|
||||||
* 由于官方并没有提供该类型, 手动去复刻成本过高, 故而采用 `any`
|
|
||||||
*/
|
*/
|
||||||
type: Array,
|
type: Array as PropType<(typeof CanvasRenderer)[]>,
|
||||||
default: () => [],
|
default: () => [],
|
||||||
},
|
},
|
||||||
watchOptions: {
|
watchOptions: {
|
||||||
@ -223,7 +224,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 resizeThrottle: DebouncedFunc<AnyFunc> // resize 防抖方法实例
|
||||||
|
|
||||||
const cssVarsRef = computed(() => {
|
const cssVarsRef = computed(() => {
|
||||||
const cssVars = {
|
const cssVars = {
|
||||||
@ -272,8 +273,7 @@ const RayChart = defineComponent({
|
|||||||
echarts.use([CanvasRenderer]) // 注册渲染器
|
echarts.use([CanvasRenderer]) // 注册渲染器
|
||||||
|
|
||||||
try {
|
try {
|
||||||
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
echarts.use(props.use)
|
||||||
echarts.use(props.use as any[])
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
console.error(
|
console.error(
|
||||||
'Error: wrong property and method passed in extend attribute',
|
'Error: wrong property and method passed in extend attribute',
|
||||||
@ -469,9 +469,9 @@ const RayChart = defineComponent({
|
|||||||
|
|
||||||
/** 注册事件 */
|
/** 注册事件 */
|
||||||
if (props.autoResize) {
|
if (props.autoResize) {
|
||||||
resizeDebounce = debounce(resizeChart, 500)
|
resizeThrottle = throttle(resizeChart, 1000)
|
||||||
|
|
||||||
on(window, 'resize', resizeDebounce)
|
on(window, 'resize', resizeThrottle)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
@ -480,7 +480,9 @@ const RayChart = defineComponent({
|
|||||||
/** 卸载 echarts */
|
/** 卸载 echarts */
|
||||||
destroyChart()
|
destroyChart()
|
||||||
/** 卸载事件柄 */
|
/** 卸载事件柄 */
|
||||||
off(window, 'resize', resizeDebounce)
|
off(window, 'resize', resizeThrottle)
|
||||||
|
/** 注销防抖 */
|
||||||
|
resizeThrottle.cancel()
|
||||||
})
|
})
|
||||||
|
|
||||||
expose({
|
expose({
|
||||||
|
@ -20,8 +20,9 @@ import { on, off } from '@use-utils/element'
|
|||||||
import type { Directive } from 'vue'
|
import type { Directive } from 'vue'
|
||||||
import type { DebounceBindingOptions } from './type'
|
import type { DebounceBindingOptions } from './type'
|
||||||
import type { AnyFunc } from '@/types/modules/utils'
|
import type { AnyFunc } from '@/types/modules/utils'
|
||||||
|
import type { DebouncedFunc } from 'lodash-es'
|
||||||
|
|
||||||
let debounceFunction: AnyFunc | null
|
let debounceFunction: DebouncedFunc<AnyFunc> | null
|
||||||
|
|
||||||
const debounceDirective: Directive<HTMLElement, DebounceBindingOptions> = {
|
const debounceDirective: Directive<HTMLElement, DebounceBindingOptions> = {
|
||||||
beforeMount: (el, binding) => {
|
beforeMount: (el, binding) => {
|
||||||
@ -39,6 +40,7 @@ const debounceDirective: Directive<HTMLElement, DebounceBindingOptions> = {
|
|||||||
const { trigger = 'click' } = binding.value
|
const { trigger = 'click' } = binding.value
|
||||||
|
|
||||||
if (debounceFunction) {
|
if (debounceFunction) {
|
||||||
|
debounceFunction.cancel()
|
||||||
off(el, trigger, debounceFunction)
|
off(el, trigger, debounceFunction)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -20,8 +20,9 @@ import { on, off } from '@use-utils/element'
|
|||||||
import type { Directive } from 'vue'
|
import type { Directive } from 'vue'
|
||||||
import type { ThrottleBindingOptions } from './type'
|
import type { ThrottleBindingOptions } from './type'
|
||||||
import type { AnyFunc } from '@/types/modules/utils'
|
import type { AnyFunc } from '@/types/modules/utils'
|
||||||
|
import type { DebouncedFunc } from 'lodash-es'
|
||||||
|
|
||||||
let throttleFunction: AnyFunc | null
|
let throttleFunction: DebouncedFunc<AnyFunc> | null
|
||||||
|
|
||||||
const throttleDirective: Directive<HTMLElement, ThrottleBindingOptions> = {
|
const throttleDirective: Directive<HTMLElement, ThrottleBindingOptions> = {
|
||||||
beforeMount: (el, binding) => {
|
beforeMount: (el, binding) => {
|
||||||
@ -39,6 +40,7 @@ const throttleDirective: Directive<HTMLElement, ThrottleBindingOptions> = {
|
|||||||
const { trigger = 'click' } = binding.value
|
const { trigger = 'click' } = binding.value
|
||||||
|
|
||||||
if (throttleFunction) {
|
if (throttleFunction) {
|
||||||
|
throttleFunction.cancel()
|
||||||
off(el, trigger, throttleFunction)
|
off(el, trigger, throttleFunction)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -12,6 +12,7 @@
|
|||||||
import './index.scss'
|
import './index.scss'
|
||||||
|
|
||||||
import RayTransitionComponent from '@/components/RayTransitionComponent/TransitionComponent.vue'
|
import RayTransitionComponent from '@/components/RayTransitionComponent/TransitionComponent.vue'
|
||||||
|
import { NSpin } from 'naive-ui'
|
||||||
|
|
||||||
import { useSetting } from '@/store'
|
import { useSetting } from '@/store'
|
||||||
|
|
||||||
@ -19,16 +20,35 @@ const ContentWrapper = defineComponent({
|
|||||||
name: 'ContentWrapper',
|
name: 'ContentWrapper',
|
||||||
setup() {
|
setup() {
|
||||||
const settingStore = useSetting()
|
const settingStore = useSetting()
|
||||||
|
const router = useRouter()
|
||||||
|
|
||||||
const { reloadRouteSwitch } = storeToRefs(settingStore)
|
const { reloadRouteSwitch } = storeToRefs(settingStore)
|
||||||
|
const spinning = ref(false)
|
||||||
|
|
||||||
|
const setupLayoutContentSpin = () => {
|
||||||
|
router.beforeEach(() => {
|
||||||
|
spinning.value = true
|
||||||
|
})
|
||||||
|
|
||||||
|
router.afterEach(() => {
|
||||||
|
setTimeout(() => {
|
||||||
|
spinning.value = false
|
||||||
|
}, 300)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
setupLayoutContentSpin()
|
||||||
|
|
||||||
return {
|
return {
|
||||||
reloadRouteSwitch,
|
reloadRouteSwitch,
|
||||||
|
spinning,
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
render() {
|
render() {
|
||||||
return this.reloadRouteSwitch ? (
|
return this.reloadRouteSwitch ? (
|
||||||
<RayTransitionComponent class="content-wrapper" />
|
<NSpin show={this.spinning} description="loading..." size="large">
|
||||||
|
<RayTransitionComponent class="content-wrapper" />
|
||||||
|
</NSpin>
|
||||||
) : (
|
) : (
|
||||||
<></>
|
<></>
|
||||||
)
|
)
|
||||||
|
@ -1,5 +1,45 @@
|
|||||||
## router 拓展
|
## router 拓展
|
||||||
|
|
||||||
|
## LAYOUT 说明
|
||||||
|
|
||||||
|
> router modules 包中的路由模块会与菜单一一映射,也就是说,路由模块的配置结构会影响菜单的展示。当你有子菜单需要配置时,你需要使用该组件。
|
||||||
|
|
||||||
|
```ts
|
||||||
|
import { t } from '@/locales/useI18n'
|
||||||
|
import { LAYOUT } from '@/router/constant/index'
|
||||||
|
|
||||||
|
import type { AppRouteRecordRaw } from '@/router/type'
|
||||||
|
|
||||||
|
const demo: AppRouteRecordRaw = {
|
||||||
|
/** 路由路径,如果为根菜单且无有菜单的时候可以配置为空字符串 */
|
||||||
|
path: '/demo',
|
||||||
|
/** views component name,必须与组件 name 属性匹配,否则缓存属性会失效 */
|
||||||
|
name: 'RDemo',
|
||||||
|
/** 这样配置后,就无须在 views 中重复写 <view-router /> 了 */
|
||||||
|
component: LAYOUT,
|
||||||
|
meta: {
|
||||||
|
/** t 方法是一个纯函数,为了配合 i18n ally 插件提示使用 */
|
||||||
|
i18nKey: t('menu.Demo'),
|
||||||
|
icon: 'demo',
|
||||||
|
order: 0,
|
||||||
|
},
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
path: '/demo-child',
|
||||||
|
name: 'RDemoChild',
|
||||||
|
component: () => import('@/views/demo-child/index'),
|
||||||
|
meta: {
|
||||||
|
i18nKey: t('menu.DemoChild'),
|
||||||
|
icon: 'demo',
|
||||||
|
order: 0,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
],
|
||||||
|
}
|
||||||
|
|
||||||
|
export default dashboard
|
||||||
|
```
|
||||||
|
|
||||||
## 类型
|
## 类型
|
||||||
|
|
||||||
```ts
|
```ts
|
||||||
|
@ -11,12 +11,12 @@
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* 该功能基于 <https://me.yka.moe/> 代码改进实现
|
* 该功能基于 <https://me.yka.moe/> 提供代码改进实现
|
||||||
|
*
|
||||||
* 自动展开所有路由
|
* 自动展开所有路由
|
||||||
|
* 把所有路由提升为顶层路由
|
||||||
*/
|
*/
|
||||||
|
|
||||||
import { cloneDeep } from 'lodash-es'
|
|
||||||
|
|
||||||
import type { AppRouteRecordRaw } from '@/router/type'
|
import type { AppRouteRecordRaw } from '@/router/type'
|
||||||
|
|
||||||
const isRootPath = (path: string) => path.startsWith('/')
|
const isRootPath = (path: string) => path.startsWith('/')
|
||||||
@ -35,37 +35,32 @@ const routePromotion = (
|
|||||||
result: AppRouteRecordRaw[] = [],
|
result: AppRouteRecordRaw[] = [],
|
||||||
path = '',
|
path = '',
|
||||||
) => {
|
) => {
|
||||||
// 如果没有小宝贝进来 则没有小宝贝出去
|
|
||||||
if (!Array.isArray(arr)) {
|
if (!Array.isArray(arr)) {
|
||||||
return []
|
return []
|
||||||
}
|
}
|
||||||
|
|
||||||
// 新来的小宝贝们先洗好澡澡哦
|
|
||||||
const sourceArr = arr
|
const sourceArr = arr
|
||||||
|
|
||||||
// 来开始我们的循环之旅吧
|
|
||||||
sourceArr.forEach((curr) => {
|
sourceArr.forEach((curr) => {
|
||||||
// 获取可爱的小宝贝哦
|
|
||||||
|
|
||||||
if (curr.children?.length) {
|
if (curr.children?.length) {
|
||||||
// 如果小宝贝有小小宝贝
|
|
||||||
|
|
||||||
// 小宝贝们有孩子了,/(ㄒoㄒ)/~~
|
|
||||||
routePromotion(
|
routePromotion(
|
||||||
curr.children,
|
curr.children,
|
||||||
result,
|
result,
|
||||||
path + (isRootPath(curr.path) ? curr.path : '/' + curr.path),
|
path + (isRootPath(curr.path) ? curr.path : '/' + curr.path),
|
||||||
)
|
)
|
||||||
} else {
|
} else {
|
||||||
// 小宝贝还是单身哦
|
const newPath =
|
||||||
// 乖乖的小宝贝快快进入口袋
|
path + (isRootPath(curr.path) ? curr.path : '/' + curr.path)
|
||||||
curr.path = path + (isRootPath(curr.path) ? curr.path : '/' + curr.path)
|
|
||||||
|
|
||||||
result.push(curr)
|
const newCurr: AppRouteRecordRaw = {
|
||||||
|
...curr,
|
||||||
|
path: newPath,
|
||||||
|
}
|
||||||
|
|
||||||
|
result.push(newCurr)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
// 返回都是根节点的小宝贝们
|
|
||||||
return result
|
return result
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -74,5 +69,7 @@ export const expandRoutes = (arr: AppRouteRecordRaw[]) => {
|
|||||||
return []
|
return []
|
||||||
}
|
}
|
||||||
|
|
||||||
return routePromotion(cloneDeep(arr))
|
const cloneArr = arr.slice()
|
||||||
|
|
||||||
|
return routePromotion(cloneArr)
|
||||||
}
|
}
|
||||||
|
@ -98,7 +98,7 @@ export const orderRoutes = (routes: AppRouteRecordRaw[]) => {
|
|||||||
export const scrollViewToTop = (route: RouteLocationNormalized) => {
|
export const scrollViewToTop = (route: RouteLocationNormalized) => {
|
||||||
const { meta } = route
|
const { meta } = route
|
||||||
|
|
||||||
/** 这个 id 是注入在 layout 中 */
|
/** LAYOUT_CONTENT_REF */
|
||||||
if (!meta?.ignoreAutoResetScroll) {
|
if (!meta?.ignoreAutoResetScroll) {
|
||||||
LAYOUT_CONTENT_REF.value?.scrollTo({
|
LAYOUT_CONTENT_REF.value?.scrollTo({
|
||||||
top: 0,
|
top: 0,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user