diff --git a/CHANGELOG.md b/CHANGELOG.md index 26418873..b9431c43 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,17 @@ # CHANGE LOG +## 3.3.4 + +### Feats + +- 新增 RayIframe 组件 +- 同步更新 `naive-ui` 版本至最新版本(2.34.3 => 2.34.4) +- 支持更多 appConfig 配置 + +### TODO + +- MenuTag: 切换页面时, 同步更新该标签的所在位置 + ## 3.3.3 ### Feats diff --git a/README.md b/README.md index 38698c3d..5e643bab 100644 --- a/README.md +++ b/README.md @@ -55,7 +55,7 @@ ## 前言 -> 该项目模板采用 `vue3.x` `vite4.0` `pinia` `tsx` 进行开发。 +> 该项目模板采用 `vue3.x` `vite4.x` `pinia` `tsx` 进行开发。 > 使用 `naive ui` 作为组件库。 > 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。 diff --git a/src/appConfig/regConfig.ts b/src/appConfig/regConfig.ts new file mode 100644 index 00000000..6fe34539 --- /dev/null +++ b/src/appConfig/regConfig.ts @@ -0,0 +1,20 @@ +/** + * + * @author Ray + * + * @date 2023-06-12 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** + * + * 正则入口 + * 系统公共正则, 配置在该文件中 + */ + +/** css 尺寸单位匹配 */ +export const ELEMENT_UNIT = + /^\d+(\.\d+)?(px|em|rem|%|vw|vh|vmin|vmax|cm|mm|in|pt|pc|ch|ex|q|s|ms|deg|rad|turn|grad|hz|khz|dpi|dpcm|dppx|fr|auto)$/ diff --git a/src/components/RayIframe/index.ts b/src/components/RayIframe/index.ts new file mode 100644 index 00000000..ecea3ef2 --- /dev/null +++ b/src/components/RayIframe/index.ts @@ -0,0 +1,3 @@ +import RayIframe from './src/index' + +export default RayIframe diff --git a/src/components/RayIframe/src/index.scss b/src/components/RayIframe/src/index.scss new file mode 100644 index 00000000..1a29ca87 --- /dev/null +++ b/src/components/RayIframe/src/index.scss @@ -0,0 +1,13 @@ +.ray-iframe { + width: var(--ray-iframe-width); + height: var(--ray-iframe-height); + box-sizing: border-box; + border: var(--ray-iframe-frameborder); + + & .ray-iframe__container { + width: 100%; + height: 100%; + border: 0; + outline: 0; + } +} diff --git a/src/components/RayIframe/src/index.tsx b/src/components/RayIframe/src/index.tsx new file mode 100644 index 00000000..173665f9 --- /dev/null +++ b/src/components/RayIframe/src/index.tsx @@ -0,0 +1,170 @@ +/** + * + * @author Ray + * + * @date 2023-06-09 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +import './index.scss' + +import { NSpin } from 'naive-ui' + +import { completeSize, on, off } from '@use-utils/element' + +import type { PropType } from 'vue' +import type { SpinProps } from 'naive-ui' + +const RayIframe = defineComponent({ + name: 'RayIframe', + props: { + src: { + /** iframe url */ + type: String, + required: true, + }, + iframeWrapperClass: { + /** 自定义类名 */ + type: String, + default: '', + }, + frameborder: { + /** 边框尺寸, 0 则不显示 */ + type: Number, + default: 0, + }, + width: { + /** iframe 宽度 */ + type: [String, Number], + default: '100%', + }, + height: { + /** iframe 高度 */ + type: [String, Number], + default: '100%', + }, + allow: { + /** + * + * iframe 特征策略 + * + * ``` + * 全屏激活: allow = 'fullscreen' + * 允许跨域: allow = 'payment' + * ``` + * + * 但是该配置属性受到浏览器安全策略影响, 使用前请仔细阅读文档 + */ + type: String, + default: null, + }, + name: { + /** iframe 定位嵌入的浏览上下文的名称 */ + type: String, + default: null, + }, + title: { + /** 标识 iframe 的主要内容 */ + type: String, + default: null, + }, + success: { + /** + * + * iframe 加载成功回调 + * 返回值: iframe 对象, Event + */ + type: Function, + default: null, + }, + error: { + /** + * + * iframe 加载失败回调 + * 返回值: iframe 对象, Event + */ + type: Function, + default: null, + }, + customSpinProps: { + type: Object as PropType, + default: () => ({}), + }, + }, + setup(props) { + const cssVars = computed(() => { + const cssVar = { + '--ray-iframe-frameborder': completeSize(props.frameborder), + '--ray-iframe-width': completeSize(props.width), + '--ray-iframe-height': completeSize(props.height), + } + + return cssVar + }) + const iframeRef = ref() + const spinShow = ref(true) + + const iframeLoadSuccess = (e: Event) => { + spinShow.value = false + + props.success?.(iframeRef.value, e) + } + + const iframeLoadError = (e: Event) => { + spinShow.value = false + + props.error?.(iframeRef.value, e) + } + + const getIframeRef = () => { + const iframeEl = iframeRef.value as HTMLElement + + return iframeEl + } + + onMounted(() => { + on(getIframeRef(), 'load', iframeLoadSuccess.bind(this)) + on(getIframeRef(), 'error', iframeLoadError) + }) + + onBeforeUnmount(() => { + off(getIframeRef(), 'load', iframeLoadSuccess) + off(getIframeRef(), 'error', iframeLoadError) + }) + + return { + cssVars, + iframeRef, + spinShow, + } + }, + render() { + return ( +
+ + {{ + ...this.$slots, + default: () => ( + + ), + }} + +
+ ) + }, +}) + +export default RayIframe diff --git a/src/layout/components/MenuTag/index.tsx b/src/layout/components/MenuTag/index.tsx index 69aa8cdb..592fa2f7 100644 --- a/src/layout/components/MenuTag/index.tsx +++ b/src/layout/components/MenuTag/index.tsx @@ -36,7 +36,7 @@ import type { MenuOption, ScrollbarInst } from 'naive-ui' const MenuTag = defineComponent({ name: 'MenuTag', - setup() { + setup(_, { expose }) { const scrollRef = ref(null) const menuStore = useMenu() @@ -246,22 +246,33 @@ const MenuTag = defineComponent({ menuModelValueChange(item.key as string, item) } - const handleScrollX = (type: 'left' | 'right') => { + const getScrollElement = () => { const scroll = document.getElementById(scrollBarUUID) // 获取滚动条容器 if (scroll) { - /** - * - * 找到实际横向滚动元素(class: n-scrollbar-container) - * 获取 scrollLeft 属性后, 用于左右滚动边界值进行处理 - */ const scrollContentElement = Array.from( scroll.childNodes, ) as HTMLElement[] const findElement = scrollContentElement.find((el) => hasClass(el, 'n-scrollbar-container'), ) - const scrollX = findElement!.scrollLeft || 0 + + return findElement + } + + return undefined + } + + const handleScrollX = (type: 'left' | 'right') => { + const el = getScrollElement() + + if (el) { + /** + * + * 找到实际横向滚动元素(class: n-scrollbar-container) + * 获取 scrollLeft 属性后, 用于左右滚动边界值进行处理 + */ + const scrollX = el!.scrollLeft || 0 const rolling = type === 'left' ? Math.max(0, scrollX - 200) : scrollX + 200 @@ -344,10 +355,28 @@ const MenuTag = defineComponent({ } } + /** + * + * 每当新的页面打开后, 将滚动条横向滚到至底部 + * 使用 nextTick 避免元素未渲染挂载至页面 + */ + const updateScrollBarPosition = () => { + const el = getScrollElement() + + if (el) { + nextTick().then(() => { + scrollRef.value?.scrollTo({ + left: 99999, + behavior: 'smooth', + }) + }) + } + } + /** 如果有且只有一个标签页时, 禁止全部关闭操作 */ watch( () => modelMenuTagOptions.value, - (newData) => { + (newData, oldData) => { moreOptions.value.forEach((curr) => { if (exclude.includes(curr.key)) { newData.length > 1 @@ -355,10 +384,15 @@ const MenuTag = defineComponent({ : (curr.disabled = true) } }) + + if (oldData?.length) { + if (newData.length > oldData?.length) { + updateScrollBarPosition() + } + } }, { immediate: true, - deep: true, }, ) @@ -370,6 +404,8 @@ const MenuTag = defineComponent({ }, ) + expose({}) + return { modelMenuTagOptions, menuModelValueChange, @@ -426,10 +462,6 @@ const MenuTag = defineComponent({ {...{ id: this.scrollBarUUID, }} - themeOverrides={{ - color: 'rgba(0, 0, 0, 0)', - colorHover: 'rgba(0, 0, 0, 0)', - }} > {typeof curr.label === 'function' diff --git a/src/router/helper/permission.ts b/src/router/helper/permission.ts index 21b4accf..d47e23f3 100644 --- a/src/router/helper/permission.ts +++ b/src/router/helper/permission.ts @@ -23,6 +23,7 @@ import { getCache, setCache } from '@/utils/cache' import { useSignin } from '@/store' import { APP_CATCH_KEY, ROOT_ROUTE } from '@/appConfig/appConfig' +import { redirectRouterToDashboard } from '@/router/helper/routerCopilot' import type { Router, NavigationGuardNext } from 'vue-router' @@ -31,13 +32,6 @@ export const permissionRouter = (router: Router) => { const { path } = ROOT_ROUTE - /** 如果没有权限, 则重定向至首页 */ - const redirectToDashboard = (next: NavigationGuardNext) => { - next(path) - - setCache('menuKey', path) - } - beforeEach((to, from, next) => { const token = getCache(APP_CATCH_KEY.token) const route = getCache('menuKey') @@ -70,13 +64,13 @@ export const permissionRouter = (router: Router) => { if (route !== 'no') { next(route) } else { - redirectToDashboard(next) + redirectRouterToDashboard(true) } } else { next() } } else { - redirectToDashboard(next) + redirectRouterToDashboard(true) } } else { if (to.path === '/' || from.path === '/login') { diff --git a/src/router/helper/routerCopilot.ts b/src/router/helper/routerCopilot.ts index 0d2e5695..f0456441 100644 --- a/src/router/helper/routerCopilot.ts +++ b/src/router/helper/routerCopilot.ts @@ -20,6 +20,7 @@ import { import { useSignin } from '@/store' import { useVueRouter } from '@/router/helper/useVueRouter' import { ROOT_ROUTE } from '@/appConfig/appConfig' +import { setCache } from '@/utils/cache' import type { Router } from 'vue-router' @@ -92,13 +93,15 @@ export const vueRouterRegister = (router: Router) => { * * @param replace 是否使用 * - * @remark 重定向路由至首页 + * @remark 重定向路由至首页, 默认采用替换方法重定向 */ -export const redirectRouterToDashboard = (isReplace?: boolean) => { +export const redirectRouterToDashboard = (isReplace = true) => { const { router } = useVueRouter() const { push, replace } = router const { path } = ROOT_ROUTE isReplace ? push(path) : replace(path) + + setCache('menuKey', path) } diff --git a/src/router/modules/iframe.ts b/src/router/modules/iframe.ts new file mode 100644 index 00000000..7752391d --- /dev/null +++ b/src/router/modules/iframe.ts @@ -0,0 +1,14 @@ +import type { AppRouteRecordRaw } from '@/router/type' + +const iframe: AppRouteRecordRaw = { + path: '/iframe', + name: 'IframeDemo', + component: () => import('@/views/iframe/index'), + meta: { + icon: 'rely', + order: 2, + noLocalTitle: 'iframe', + }, +} + +export default iframe diff --git a/src/styles/base.scss b/src/styles/base.scss index 26d35043..46c1b05c 100644 --- a/src/styles/base.scss +++ b/src/styles/base.scss @@ -1,5 +1,6 @@ @import "@/styles/animate.scss"; @import "@/styles/root.scss"; +@import "@/styles/naive.scss"; body, h1, diff --git a/src/styles/naive.scss b/src/styles/naive.scss new file mode 100644 index 00000000..e3fcfd72 --- /dev/null +++ b/src/styles/naive.scss @@ -0,0 +1,5 @@ +.n-spin-container, +.n-spin-container .n-spin-content { + width: 100%; + height: 100%; +} diff --git a/src/utils/element.ts b/src/utils/element.ts index 3ae2461d..72fc1ae6 100644 --- a/src/utils/element.ts +++ b/src/utils/element.ts @@ -1,4 +1,6 @@ import { validteValueType } from '@use-utils/hook' +import { ELEMENT_UNIT } from '@/appConfig/regConfig' + /** * * @param element Target element dom @@ -242,3 +244,19 @@ export const getElement = (element: string) => { return [] } } + +/** + * + * @param size css size + * + * @remark 自动补全尺寸 + */ +export const completeSize = (size: number | string) => { + if (typeof size === 'number') { + return size.toString() + 'px' + } else if (ELEMENT_UNIT.test(size)) { + return size + } else { + return size + 'px' + } +} diff --git a/src/views/iframe/index.tsx b/src/views/iframe/index.tsx new file mode 100644 index 00000000..160a1679 --- /dev/null +++ b/src/views/iframe/index.tsx @@ -0,0 +1,48 @@ +/** + * + * @author Ray + * + * @date 2023-06-09 + * + * @workspace ray-template + * + * @remark 今天也是元气满满撸代码的一天 + */ + +/** + * + * RayIframe 组件使用示例 + * + * 具体使用参考 props 代码注释 + * 做了简单的一个组件封装, 希望有用 + */ + +import { NSpace } from 'naive-ui' +import RayIframe from '@/components/RayIframe/index' + +const IframeDemo = defineComponent({ + name: 'IframeDemo', + setup() { + return {} + }, + render() { + return ( + + +

naive ui

+ +
+ +

vueuse

+ +
+
+ ) + }, +}) + +export default IframeDemo