mirror of
https://github.com/XiaoDaiGua-Ray/ray-template.git
synced 2025-04-06 03:57:49 +08:00
v3.3.4
This commit is contained in:
parent
5d70662b87
commit
f5ce953b83
12
CHANGELOG.md
12
CHANGELOG.md
@ -1,5 +1,17 @@
|
|||||||
# CHANGE LOG
|
# CHANGE LOG
|
||||||
|
|
||||||
|
## 3.3.4
|
||||||
|
|
||||||
|
### Feats
|
||||||
|
|
||||||
|
- 新增 RayIframe 组件
|
||||||
|
- 同步更新 `naive-ui` 版本至最新版本(2.34.3 => 2.34.4)
|
||||||
|
- 支持更多 appConfig 配置
|
||||||
|
|
||||||
|
### TODO
|
||||||
|
|
||||||
|
- MenuTag: 切换页面时, 同步更新该标签的所在位置
|
||||||
|
|
||||||
## 3.3.3
|
## 3.3.3
|
||||||
|
|
||||||
### Feats
|
### Feats
|
||||||
|
@ -55,7 +55,7 @@
|
|||||||
|
|
||||||
## 前言
|
## 前言
|
||||||
|
|
||||||
> 该项目模板采用 `vue3.x` `vite4.0` `pinia` `tsx` 进行开发。
|
> 该项目模板采用 `vue3.x` `vite4.x` `pinia` `tsx` 进行开发。
|
||||||
> 使用 `naive ui` 作为组件库。
|
> 使用 `naive ui` 作为组件库。
|
||||||
> 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。
|
> 预设了最佳构建体验的配置与常用搬砖工具。意在提供一个简洁、快速上手的模板。
|
||||||
|
|
||||||
|
20
src/appConfig/regConfig.ts
Normal file
20
src/appConfig/regConfig.ts
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Ray <https://github.com/XiaoDaiGua-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)$/
|
3
src/components/RayIframe/index.ts
Normal file
3
src/components/RayIframe/index.ts
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
import RayIframe from './src/index'
|
||||||
|
|
||||||
|
export default RayIframe
|
13
src/components/RayIframe/src/index.scss
Normal file
13
src/components/RayIframe/src/index.scss
Normal file
@ -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;
|
||||||
|
}
|
||||||
|
}
|
170
src/components/RayIframe/src/index.tsx
Normal file
170
src/components/RayIframe/src/index.tsx
Normal file
@ -0,0 +1,170 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Ray <https://github.com/XiaoDaiGua-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<SpinProps>,
|
||||||
|
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<HTMLIFrameElement>()
|
||||||
|
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 (
|
||||||
|
<div
|
||||||
|
class={['ray-iframe', this.iframeWrapperClass]}
|
||||||
|
style={[this.cssVars]}
|
||||||
|
>
|
||||||
|
<NSpin {...this.customSpinProps} show={this.spinShow}>
|
||||||
|
{{
|
||||||
|
...this.$slots,
|
||||||
|
default: () => (
|
||||||
|
<iframe
|
||||||
|
class="ray-iframe__container"
|
||||||
|
ref="iframeRef"
|
||||||
|
src={this.src}
|
||||||
|
allow={this.allow}
|
||||||
|
name={this.name}
|
||||||
|
title={this.title}
|
||||||
|
></iframe>
|
||||||
|
),
|
||||||
|
}}
|
||||||
|
</NSpin>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
export default RayIframe
|
@ -36,7 +36,7 @@ import type { MenuOption, ScrollbarInst } from 'naive-ui'
|
|||||||
|
|
||||||
const MenuTag = defineComponent({
|
const MenuTag = defineComponent({
|
||||||
name: 'MenuTag',
|
name: 'MenuTag',
|
||||||
setup() {
|
setup(_, { expose }) {
|
||||||
const scrollRef = ref<ScrollbarInst | null>(null)
|
const scrollRef = ref<ScrollbarInst | null>(null)
|
||||||
|
|
||||||
const menuStore = useMenu()
|
const menuStore = useMenu()
|
||||||
@ -246,22 +246,33 @@ const MenuTag = defineComponent({
|
|||||||
menuModelValueChange(item.key as string, item)
|
menuModelValueChange(item.key as string, item)
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleScrollX = (type: 'left' | 'right') => {
|
const getScrollElement = () => {
|
||||||
const scroll = document.getElementById(scrollBarUUID) // 获取滚动条容器
|
const scroll = document.getElementById(scrollBarUUID) // 获取滚动条容器
|
||||||
|
|
||||||
if (scroll) {
|
if (scroll) {
|
||||||
/**
|
|
||||||
*
|
|
||||||
* 找到实际横向滚动元素(class: n-scrollbar-container)
|
|
||||||
* 获取 scrollLeft 属性后, 用于左右滚动边界值进行处理
|
|
||||||
*/
|
|
||||||
const scrollContentElement = Array.from(
|
const scrollContentElement = Array.from(
|
||||||
scroll.childNodes,
|
scroll.childNodes,
|
||||||
) as HTMLElement[]
|
) as HTMLElement[]
|
||||||
const findElement = scrollContentElement.find((el) =>
|
const findElement = scrollContentElement.find((el) =>
|
||||||
hasClass(el, 'n-scrollbar-container'),
|
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 =
|
const rolling =
|
||||||
type === 'left' ? Math.max(0, scrollX - 200) : scrollX + 200
|
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(
|
watch(
|
||||||
() => modelMenuTagOptions.value,
|
() => modelMenuTagOptions.value,
|
||||||
(newData) => {
|
(newData, oldData) => {
|
||||||
moreOptions.value.forEach((curr) => {
|
moreOptions.value.forEach((curr) => {
|
||||||
if (exclude.includes(curr.key)) {
|
if (exclude.includes(curr.key)) {
|
||||||
newData.length > 1
|
newData.length > 1
|
||||||
@ -355,10 +384,15 @@ const MenuTag = defineComponent({
|
|||||||
: (curr.disabled = true)
|
: (curr.disabled = true)
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
if (oldData?.length) {
|
||||||
|
if (newData.length > oldData?.length) {
|
||||||
|
updateScrollBarPosition()
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
immediate: true,
|
immediate: true,
|
||||||
deep: true,
|
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -370,6 +404,8 @@ const MenuTag = defineComponent({
|
|||||||
},
|
},
|
||||||
)
|
)
|
||||||
|
|
||||||
|
expose({})
|
||||||
|
|
||||||
return {
|
return {
|
||||||
modelMenuTagOptions,
|
modelMenuTagOptions,
|
||||||
menuModelValueChange,
|
menuModelValueChange,
|
||||||
@ -426,10 +462,6 @@ const MenuTag = defineComponent({
|
|||||||
{...{
|
{...{
|
||||||
id: this.scrollBarUUID,
|
id: this.scrollBarUUID,
|
||||||
}}
|
}}
|
||||||
themeOverrides={{
|
|
||||||
color: 'rgba(0, 0, 0, 0)',
|
|
||||||
colorHover: 'rgba(0, 0, 0, 0)',
|
|
||||||
}}
|
|
||||||
>
|
>
|
||||||
<NSpace
|
<NSpace
|
||||||
class="menu-tag-wrapper"
|
class="menu-tag-wrapper"
|
||||||
@ -450,6 +482,7 @@ const MenuTag = defineComponent({
|
|||||||
onContextmenu: this.handleContextMenu.bind(this, idx),
|
onContextmenu: this.handleContextMenu.bind(this, idx),
|
||||||
onMouseenter: this.menuTagMouseenter.bind(this, curr),
|
onMouseenter: this.menuTagMouseenter.bind(this, curr),
|
||||||
onMouseleave: this.menuTagMouseleave.bind(this, curr),
|
onMouseleave: this.menuTagMouseleave.bind(this, curr),
|
||||||
|
tag_data: curr.path,
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{typeof curr.label === 'function'
|
{typeof curr.label === 'function'
|
||||||
|
@ -23,6 +23,7 @@
|
|||||||
import { getCache, setCache } from '@/utils/cache'
|
import { getCache, setCache } from '@/utils/cache'
|
||||||
import { useSignin } from '@/store'
|
import { useSignin } from '@/store'
|
||||||
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/appConfig/appConfig'
|
import { APP_CATCH_KEY, ROOT_ROUTE } from '@/appConfig/appConfig'
|
||||||
|
import { redirectRouterToDashboard } from '@/router/helper/routerCopilot'
|
||||||
|
|
||||||
import type { Router, NavigationGuardNext } from 'vue-router'
|
import type { Router, NavigationGuardNext } from 'vue-router'
|
||||||
|
|
||||||
@ -31,13 +32,6 @@ export const permissionRouter = (router: Router) => {
|
|||||||
|
|
||||||
const { path } = ROOT_ROUTE
|
const { path } = ROOT_ROUTE
|
||||||
|
|
||||||
/** 如果没有权限, 则重定向至首页 */
|
|
||||||
const redirectToDashboard = (next: NavigationGuardNext) => {
|
|
||||||
next(path)
|
|
||||||
|
|
||||||
setCache('menuKey', path)
|
|
||||||
}
|
|
||||||
|
|
||||||
beforeEach((to, from, next) => {
|
beforeEach((to, from, next) => {
|
||||||
const token = getCache(APP_CATCH_KEY.token)
|
const token = getCache(APP_CATCH_KEY.token)
|
||||||
const route = getCache('menuKey')
|
const route = getCache('menuKey')
|
||||||
@ -70,13 +64,13 @@ export const permissionRouter = (router: Router) => {
|
|||||||
if (route !== 'no') {
|
if (route !== 'no') {
|
||||||
next(route)
|
next(route)
|
||||||
} else {
|
} else {
|
||||||
redirectToDashboard(next)
|
redirectRouterToDashboard(true)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
next()
|
next()
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
redirectToDashboard(next)
|
redirectRouterToDashboard(true)
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (to.path === '/' || from.path === '/login') {
|
if (to.path === '/' || from.path === '/login') {
|
||||||
|
@ -20,6 +20,7 @@ import {
|
|||||||
import { useSignin } from '@/store'
|
import { useSignin } from '@/store'
|
||||||
import { useVueRouter } from '@/router/helper/useVueRouter'
|
import { useVueRouter } from '@/router/helper/useVueRouter'
|
||||||
import { ROOT_ROUTE } from '@/appConfig/appConfig'
|
import { ROOT_ROUTE } from '@/appConfig/appConfig'
|
||||||
|
import { setCache } from '@/utils/cache'
|
||||||
|
|
||||||
import type { Router } from 'vue-router'
|
import type { Router } from 'vue-router'
|
||||||
|
|
||||||
@ -92,13 +93,15 @@ export const vueRouterRegister = (router: Router) => {
|
|||||||
*
|
*
|
||||||
* @param replace 是否使用
|
* @param replace 是否使用
|
||||||
*
|
*
|
||||||
* @remark 重定向路由至首页
|
* @remark 重定向路由至首页, 默认采用替换方法重定向
|
||||||
*/
|
*/
|
||||||
export const redirectRouterToDashboard = (isReplace?: boolean) => {
|
export const redirectRouterToDashboard = (isReplace = true) => {
|
||||||
const { router } = useVueRouter()
|
const { router } = useVueRouter()
|
||||||
|
|
||||||
const { push, replace } = router
|
const { push, replace } = router
|
||||||
const { path } = ROOT_ROUTE
|
const { path } = ROOT_ROUTE
|
||||||
|
|
||||||
isReplace ? push(path) : replace(path)
|
isReplace ? push(path) : replace(path)
|
||||||
|
|
||||||
|
setCache('menuKey', path)
|
||||||
}
|
}
|
||||||
|
14
src/router/modules/iframe.ts
Normal file
14
src/router/modules/iframe.ts
Normal file
@ -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
|
@ -1,5 +1,6 @@
|
|||||||
@import "@/styles/animate.scss";
|
@import "@/styles/animate.scss";
|
||||||
@import "@/styles/root.scss";
|
@import "@/styles/root.scss";
|
||||||
|
@import "@/styles/naive.scss";
|
||||||
|
|
||||||
body,
|
body,
|
||||||
h1,
|
h1,
|
||||||
|
5
src/styles/naive.scss
Normal file
5
src/styles/naive.scss
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
.n-spin-container,
|
||||||
|
.n-spin-container .n-spin-content {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
}
|
@ -1,4 +1,6 @@
|
|||||||
import { validteValueType } from '@use-utils/hook'
|
import { validteValueType } from '@use-utils/hook'
|
||||||
|
import { ELEMENT_UNIT } from '@/appConfig/regConfig'
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
* @param element Target element dom
|
* @param element Target element dom
|
||||||
@ -242,3 +244,19 @@ export const getElement = (element: string) => {
|
|||||||
return []
|
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'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
48
src/views/iframe/index.tsx
Normal file
48
src/views/iframe/index.tsx
Normal file
@ -0,0 +1,48 @@
|
|||||||
|
/**
|
||||||
|
*
|
||||||
|
* @author Ray <https://github.com/XiaoDaiGua-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 (
|
||||||
|
<NSpace vertical size={[20, 20]}>
|
||||||
|
<NSpace vertical size={[20, 20]}>
|
||||||
|
<h2>naive ui</h2>
|
||||||
|
<RayIframe
|
||||||
|
src="https://www.naiveui.com/zh-CN/dark"
|
||||||
|
height="500"
|
||||||
|
allow="fullscreen"
|
||||||
|
/>
|
||||||
|
</NSpace>
|
||||||
|
<NSpace vertical size={[20, 20]}>
|
||||||
|
<h2>vueuse</h2>
|
||||||
|
<RayIframe src="https://www.vueusejs.com/" height="500" />
|
||||||
|
</NSpace>
|
||||||
|
</NSpace>
|
||||||
|
)
|
||||||
|
},
|
||||||
|
})
|
||||||
|
|
||||||
|
export default IframeDemo
|
Loading…
x
Reference in New Issue
Block a user