组件修改
This commit is contained in:
@@ -58,29 +58,12 @@
|
|||||||
<!-- 主内容区 -->
|
<!-- 主内容区 -->
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
<!-- iframe 模式 -->
|
<!-- iframe 模式 -->
|
||||||
<div v-if="currentIframeUrl" class="iframe-container">
|
<IframeView
|
||||||
<div class="iframe-header">
|
v-if="currentIframeUrl"
|
||||||
<span class="iframe-title">{{ currentMenuItem?.label }}</span>
|
:url="currentIframeUrl"
|
||||||
<el-button
|
:title="currentMenuItem?.label"
|
||||||
text
|
:show-header="true"
|
||||||
@click="handleRefreshIframe"
|
/>
|
||||||
:icon="Refresh"
|
|
||||||
>
|
|
||||||
刷新
|
|
||||||
</el-button>
|
|
||||||
</div>
|
|
||||||
<iframe
|
|
||||||
ref="iframeRef"
|
|
||||||
:src="currentIframeUrl"
|
|
||||||
class="content-iframe"
|
|
||||||
frameborder="0"
|
|
||||||
@load="handleIframeLoad"
|
|
||||||
/>
|
|
||||||
<div v-if="iframeLoading" class="iframe-loading">
|
|
||||||
<Loader :size="20" class="is-loading" />
|
|
||||||
<span>加载中...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- 路由模式 -->
|
<!-- 路由模式 -->
|
||||||
<router-view v-else />
|
<router-view v-else />
|
||||||
@@ -101,13 +84,9 @@ import {
|
|||||||
ChevronRight,
|
ChevronRight,
|
||||||
User,
|
User,
|
||||||
Settings,
|
Settings,
|
||||||
LogOut,
|
LogOut
|
||||||
RefreshCw,
|
|
||||||
Loader
|
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
import { IframeView } from 'shared/components'
|
||||||
// el-button 图标需要传入组件
|
|
||||||
const Refresh = RefreshCw
|
|
||||||
import { ElMessage } from 'element-plus'
|
import { ElMessage } from 'element-plus'
|
||||||
import type { MenuItem } from 'shared/types'
|
import type { MenuItem } from 'shared/types'
|
||||||
|
|
||||||
@@ -117,8 +96,6 @@ const route = useRoute()
|
|||||||
// 状态管理
|
// 状态管理
|
||||||
const collapsed = ref(false)
|
const collapsed = ref(false)
|
||||||
const activeMenu = ref('home')
|
const activeMenu = ref('home')
|
||||||
const iframeLoading = ref(false)
|
|
||||||
const iframeRef = ref<HTMLIFrameElement>()
|
|
||||||
|
|
||||||
// 从 LocalStorage 获取用户名
|
// 从 LocalStorage 获取用户名
|
||||||
function getUserName(): string {
|
function getUserName(): string {
|
||||||
@@ -200,7 +177,7 @@ const menuItems = ref<MenuItem[]>(loadMenuFromStorage())
|
|||||||
|
|
||||||
// 当前菜单项
|
// 当前菜单项
|
||||||
const currentMenuItem = computed(() => {
|
const currentMenuItem = computed(() => {
|
||||||
return menuItems.value.find(item => item.key === activeMenu.value)
|
return menuItems.value.find((item: MenuItem) => item.key === activeMenu.value)
|
||||||
})
|
})
|
||||||
|
|
||||||
// 当前 iframe URL(从路由 meta 读取)
|
// 当前 iframe URL(从路由 meta 读取)
|
||||||
@@ -221,24 +198,9 @@ const handleMenuClick = (item: MenuItem) => {
|
|||||||
// 所有菜单都通过路由跳转
|
// 所有菜单都通过路由跳转
|
||||||
if (item.url) {
|
if (item.url) {
|
||||||
router.push(item.url)
|
router.push(item.url)
|
||||||
if (item.viewType === 'iframe') {
|
|
||||||
iframeLoading.value = true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// iframe 加载完成
|
|
||||||
const handleIframeLoad = () => {
|
|
||||||
iframeLoading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// 刷新 iframe
|
|
||||||
const handleRefreshIframe = () => {
|
|
||||||
if (iframeRef.value) {
|
|
||||||
iframeLoading.value = true
|
|
||||||
iframeRef.value.src = iframeRef.value.src
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户头像加载错误
|
// 用户头像加载错误
|
||||||
const handleAvatarError = () => {
|
const handleAvatarError = () => {
|
||||||
|
|||||||
@@ -51,19 +51,12 @@
|
|||||||
<!-- 主内容区 -->
|
<!-- 主内容区 -->
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
<!-- iframe 模式 -->
|
<!-- iframe 模式 -->
|
||||||
<div v-if="currentIframeUrl" class="iframe-container">
|
<IframeView
|
||||||
<iframe
|
v-if="currentIframeUrl"
|
||||||
ref="iframeRef"
|
:url="currentIframeUrl"
|
||||||
:src="currentIframeUrl"
|
:title="currentMenuItem?.label"
|
||||||
class="content-iframe"
|
:show-header="false"
|
||||||
frameborder="0"
|
/>
|
||||||
@load="handleIframeLoad"
|
|
||||||
/>
|
|
||||||
<div v-if="iframeLoading" class="iframe-loading">
|
|
||||||
<el-icon class="is-loading"><Loading /></el-icon>
|
|
||||||
<span>加载中...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- 路由模式 -->
|
<!-- 路由模式 -->
|
||||||
<router-view v-else />
|
<router-view v-else />
|
||||||
@@ -86,9 +79,9 @@ import {
|
|||||||
Setting,
|
Setting,
|
||||||
SwitchButton,
|
SwitchButton,
|
||||||
Refresh,
|
Refresh,
|
||||||
Loading,
|
|
||||||
Back
|
Back
|
||||||
} from '@element-plus/icons-vue'
|
} from '@element-plus/icons-vue'
|
||||||
|
import { IframeView } from 'shared/components'
|
||||||
import { PanelLeftClose, PanelLeftOpen } from 'lucide-vue-next'
|
import { PanelLeftClose, PanelLeftOpen } from 'lucide-vue-next'
|
||||||
import { ElMessage } from 'element-plus'
|
import { ElMessage } from 'element-plus'
|
||||||
import type { MenuItem } from 'shared/types'
|
import type { MenuItem } from 'shared/types'
|
||||||
@@ -99,8 +92,6 @@ const route = useRoute()
|
|||||||
// 状态管理
|
// 状态管理
|
||||||
const collapsed = ref(false)
|
const collapsed = ref(false)
|
||||||
const activeMenu = ref('home')
|
const activeMenu = ref('home')
|
||||||
const iframeLoading = ref(false)
|
|
||||||
const iframeRef = ref<HTMLIFrameElement>()
|
|
||||||
|
|
||||||
// 从 LocalStorage 获取用户名
|
// 从 LocalStorage 获取用户名
|
||||||
function getUserName(): string {
|
function getUserName(): string {
|
||||||
@@ -174,7 +165,7 @@ const menuItems = ref<MenuItem[]>(loadMenuFromStorage())
|
|||||||
|
|
||||||
// 当前菜单项
|
// 当前菜单项
|
||||||
const currentMenuItem = computed(() => {
|
const currentMenuItem = computed(() => {
|
||||||
return menuItems.value.find(item => item.key === activeMenu.value)
|
return menuItems.value.find((item: MenuItem) => item.key === activeMenu.value)
|
||||||
})
|
})
|
||||||
|
|
||||||
// 当前 iframe URL(从路由 meta 读取)
|
// 当前 iframe URL(从路由 meta 读取)
|
||||||
@@ -195,22 +186,6 @@ const handleMenuClick = (item: MenuItem) => {
|
|||||||
// 所有菜单都通过路由跳转
|
// 所有菜单都通过路由跳转
|
||||||
if (item.url) {
|
if (item.url) {
|
||||||
router.push(item.url)
|
router.push(item.url)
|
||||||
if (item.viewType === 'iframe') {
|
|
||||||
iframeLoading.value = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// iframe 加载完成
|
|
||||||
const handleIframeLoad = () => {
|
|
||||||
iframeLoading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// 刷新 iframe
|
|
||||||
const handleRefreshIframe = () => {
|
|
||||||
if (iframeRef.value) {
|
|
||||||
iframeLoading.value = true
|
|
||||||
iframeRef.value.src = iframeRef.value.src
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -69,19 +69,12 @@
|
|||||||
<!-- 主内容区 -->
|
<!-- 主内容区 -->
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
<!-- iframe 模式 -->
|
<!-- iframe 模式 -->
|
||||||
<div v-if="currentIframeUrl" class="iframe-container">
|
<IframeView
|
||||||
<iframe
|
v-if="currentIframeUrl"
|
||||||
ref="iframeRef"
|
:url="currentIframeUrl"
|
||||||
:src="currentIframeUrl"
|
:title="currentMenuItem?.label"
|
||||||
class="content-iframe"
|
:show-header="false"
|
||||||
frameborder="0"
|
/>
|
||||||
@load="handleIframeLoad"
|
|
||||||
/>
|
|
||||||
<div v-if="iframeLoading" class="iframe-loading">
|
|
||||||
<Loader :size="20" class="is-loading" />
|
|
||||||
<span>加载中...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- 路由模式 -->
|
<!-- 路由模式 -->
|
||||||
<router-view v-else />
|
<router-view v-else />
|
||||||
@@ -101,10 +94,9 @@ import {
|
|||||||
Headphones,
|
Headphones,
|
||||||
User,
|
User,
|
||||||
Settings,
|
Settings,
|
||||||
LogOut,
|
LogOut
|
||||||
RefreshCw,
|
|
||||||
Loader
|
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
import { IframeView } from 'shared/components'
|
||||||
import { ElMessage } from 'element-plus'
|
import { ElMessage } from 'element-plus'
|
||||||
import type { MenuItem } from 'shared/types'
|
import type { MenuItem } from 'shared/types'
|
||||||
|
|
||||||
@@ -114,8 +106,6 @@ const route = useRoute()
|
|||||||
// 状态管理
|
// 状态管理
|
||||||
const collapsed = ref(false)
|
const collapsed = ref(false)
|
||||||
const activeMenu = ref('home')
|
const activeMenu = ref('home')
|
||||||
const iframeLoading = ref(false)
|
|
||||||
const iframeRef = ref<HTMLIFrameElement>()
|
|
||||||
const hasAdmin = ref(false)
|
const hasAdmin = ref(false)
|
||||||
|
|
||||||
// 从 LocalStorage 获取用户名
|
// 从 LocalStorage 获取用户名
|
||||||
@@ -198,7 +188,7 @@ const menuItems = ref<MenuItem[]>(loadMenuFromStorage())
|
|||||||
|
|
||||||
// 当前菜单项
|
// 当前菜单项
|
||||||
const currentMenuItem = computed(() => {
|
const currentMenuItem = computed(() => {
|
||||||
return menuItems.value.find(item => item.key === activeMenu.value)
|
return menuItems.value.find((item: MenuItem) => item.key === activeMenu.value)
|
||||||
})
|
})
|
||||||
|
|
||||||
// 当前 iframe URL(从路由 meta 读取)
|
// 当前 iframe URL(从路由 meta 读取)
|
||||||
@@ -219,22 +209,6 @@ const handleMenuClick = (item: MenuItem) => {
|
|||||||
// 所有菜单都通过路由跳转
|
// 所有菜单都通过路由跳转
|
||||||
if (item.url) {
|
if (item.url) {
|
||||||
router.push(item.url)
|
router.push(item.url)
|
||||||
if (item.viewType === 'iframe') {
|
|
||||||
iframeLoading.value = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// iframe 加载完成
|
|
||||||
const handleIframeLoad = () => {
|
|
||||||
iframeLoading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// 刷新 iframe
|
|
||||||
const handleRefreshIframe = () => {
|
|
||||||
if (iframeRef.value) {
|
|
||||||
iframeLoading.value = true
|
|
||||||
iframeRef.value.src = iframeRef.value.src
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -127,7 +127,7 @@
|
|||||||
v-model="inputText"
|
v-model="inputText"
|
||||||
placeholder="请输入内容..."
|
placeholder="请输入内容..."
|
||||||
@keydown.enter.prevent="handleSend"
|
@keydown.enter.prevent="handleSend"
|
||||||
rows="1"
|
:rows="1"
|
||||||
ref="textareaRef"
|
ref="textareaRef"
|
||||||
></textarea>
|
></textarea>
|
||||||
<div class="input-actions">
|
<div class="input-actions">
|
||||||
|
|||||||
@@ -1,9 +1,23 @@
|
|||||||
<template>
|
<template>
|
||||||
<div class="iframe-view">
|
<div class="iframe-view">
|
||||||
|
<!-- 可选的头部区域 -->
|
||||||
|
<div v-if="showHeader && finalUrl" class="iframe-header">
|
||||||
|
<span class="iframe-title">{{ title }}</span>
|
||||||
|
<el-button
|
||||||
|
text
|
||||||
|
@click="refresh"
|
||||||
|
:icon="RefreshCw"
|
||||||
|
>
|
||||||
|
刷新
|
||||||
|
</el-button>
|
||||||
|
</div>
|
||||||
|
|
||||||
<iframe
|
<iframe
|
||||||
v-if="iframeUrl"
|
v-if="finalUrl"
|
||||||
:src="iframeUrl"
|
ref="iframeRef"
|
||||||
|
:src="finalUrl"
|
||||||
class="iframe-content"
|
class="iframe-content"
|
||||||
|
:class="{ 'with-header': showHeader }"
|
||||||
frameborder="0"
|
frameborder="0"
|
||||||
@load="handleLoad"
|
@load="handleLoad"
|
||||||
/>
|
/>
|
||||||
@@ -19,24 +33,62 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
import { ref, computed, onMounted } from 'vue'
|
import { ref, computed, onMounted, watch } from 'vue'
|
||||||
import { useRoute } from 'vue-router'
|
import { useRoute } from 'vue-router'
|
||||||
import { Loader, AlertTriangle } from 'lucide-vue-next'
|
import { Loader, AlertTriangle, RefreshCw } from 'lucide-vue-next'
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
url?: string // 直接传入的 URL(优先级高于 route.meta)
|
||||||
|
title?: string // 标题
|
||||||
|
showHeader?: boolean // 是否显示头部(带刷新按钮)
|
||||||
|
}
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<Props>(), {
|
||||||
|
url: '',
|
||||||
|
title: '',
|
||||||
|
showHeader: false
|
||||||
|
})
|
||||||
|
|
||||||
|
const emit = defineEmits<{
|
||||||
|
load: []
|
||||||
|
}>()
|
||||||
|
|
||||||
const route = useRoute()
|
const route = useRoute()
|
||||||
const loading = ref(true)
|
const loading = ref(true)
|
||||||
|
const iframeRef = ref<HTMLIFrameElement>()
|
||||||
|
|
||||||
// 从路由 meta 中获取 iframe URL
|
// 最终的 iframe URL(props.url 优先,否则从 route.meta 获取)
|
||||||
const iframeUrl = computed(() => {
|
const finalUrl = computed(() => {
|
||||||
return route.meta.iframeUrl as string || ''
|
return props.url || (route.meta.iframeUrl as string) || ''
|
||||||
})
|
})
|
||||||
|
|
||||||
function handleLoad() {
|
function handleLoad() {
|
||||||
loading.value = false
|
loading.value = false
|
||||||
|
emit('load')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// 刷新 iframe
|
||||||
|
function refresh() {
|
||||||
|
if (iframeRef.value) {
|
||||||
|
loading.value = true
|
||||||
|
iframeRef.value.src = iframeRef.value.src
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 监听 URL 变化,重新加载
|
||||||
|
watch(finalUrl, () => {
|
||||||
|
if (finalUrl.value) {
|
||||||
|
loading.value = true
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
// 暴露方法供外部调用
|
||||||
|
defineExpose({
|
||||||
|
refresh
|
||||||
|
})
|
||||||
|
|
||||||
onMounted(() => {
|
onMounted(() => {
|
||||||
console.log('[IframeView] 加载 iframe:', iframeUrl.value)
|
console.log('[IframeView] 加载 iframe:', finalUrl.value)
|
||||||
})
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -46,12 +98,35 @@ onMounted(() => {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.iframe-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 12px 16px;
|
||||||
|
background: var(--el-bg-color);
|
||||||
|
border-bottom: 1px solid var(--el-border-color-light);
|
||||||
|
flex-shrink: 0;
|
||||||
|
|
||||||
|
.iframe-title {
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 500;
|
||||||
|
color: var(--el-text-color-primary);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.iframe-content {
|
.iframe-content {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 100%;
|
height: 100%;
|
||||||
border: none;
|
border: none;
|
||||||
|
flex: 1;
|
||||||
|
|
||||||
|
&.with-header {
|
||||||
|
height: calc(100% - 49px);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.iframe-error {
|
.iframe-error {
|
||||||
@@ -82,9 +157,18 @@ onMounted(() => {
|
|||||||
background: var(--el-bg-color);
|
background: var(--el-bg-color);
|
||||||
gap: 12px;
|
gap: 12px;
|
||||||
|
|
||||||
.el-icon {
|
.is-loading {
|
||||||
font-size: 32px;
|
animation: rotating 1.5s linear infinite;
|
||||||
color: var(--el-color-primary);
|
color: var(--el-color-primary);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@keyframes rotating {
|
||||||
|
from {
|
||||||
|
transform: rotate(0deg);
|
||||||
|
}
|
||||||
|
to {
|
||||||
|
transform: rotate(360deg);
|
||||||
|
}
|
||||||
|
}
|
||||||
</style>
|
</style>
|
||||||
|
|||||||
@@ -65,29 +65,12 @@
|
|||||||
<!-- 主内容区 -->
|
<!-- 主内容区 -->
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
<!-- iframe 模式 -->
|
<!-- iframe 模式 -->
|
||||||
<div v-if="currentIframeUrl" class="iframe-container">
|
<IframeView
|
||||||
<div class="iframe-header">
|
v-if="currentIframeUrl"
|
||||||
<span class="iframe-title">{{ currentMenuItem?.label }}</span>
|
:url="currentIframeUrl"
|
||||||
<el-button
|
:title="currentMenuItem?.label"
|
||||||
text
|
:show-header="true"
|
||||||
@click="handleRefreshIframe"
|
/>
|
||||||
:icon="Refresh"
|
|
||||||
>
|
|
||||||
刷新
|
|
||||||
</el-button>
|
|
||||||
</div>
|
|
||||||
<iframe
|
|
||||||
ref="iframeRef"
|
|
||||||
:src="currentIframeUrl"
|
|
||||||
class="content-iframe"
|
|
||||||
frameborder="0"
|
|
||||||
@load="handleIframeLoad"
|
|
||||||
/>
|
|
||||||
<div v-if="iframeLoading" class="iframe-loading">
|
|
||||||
<Loader :size="20" class="is-loading" />
|
|
||||||
<span>加载中...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- 路由模式 -->
|
<!-- 路由模式 -->
|
||||||
<router-view v-else />
|
<router-view v-else />
|
||||||
@@ -112,13 +95,9 @@ import {
|
|||||||
ChevronDown,
|
ChevronDown,
|
||||||
User,
|
User,
|
||||||
Settings,
|
Settings,
|
||||||
LogOut,
|
LogOut
|
||||||
RefreshCw,
|
|
||||||
Loader
|
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
import { IframeView } from '@/components'
|
||||||
// el-button 图标需要传入组件
|
|
||||||
const Refresh = RefreshCw
|
|
||||||
|
|
||||||
import type { MenuItem } from '@/types/menu'
|
import type { MenuItem } from '@/types/menu'
|
||||||
|
|
||||||
@@ -217,8 +196,6 @@ const serviceTitle = computed(() => {
|
|||||||
// 状态管理
|
// 状态管理
|
||||||
const collapsed = ref(false)
|
const collapsed = ref(false)
|
||||||
const activeMenu = ref('home')
|
const activeMenu = ref('home')
|
||||||
const iframeLoading = ref(false)
|
|
||||||
const iframeRef = ref<HTMLIFrameElement>()
|
|
||||||
|
|
||||||
// 从 LocalStorage 获取用户名
|
// 从 LocalStorage 获取用户名
|
||||||
function getUserName(): string {
|
function getUserName(): string {
|
||||||
@@ -344,22 +321,6 @@ const handleMenuClick = (item: MenuItem) => {
|
|||||||
// 所有菜单都通过路由跳转
|
// 所有菜单都通过路由跳转
|
||||||
if (item.url) {
|
if (item.url) {
|
||||||
router.push(item.url)
|
router.push(item.url)
|
||||||
if (item.viewType === 'iframe') {
|
|
||||||
iframeLoading.value = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// iframe 加载完成
|
|
||||||
const handleIframeLoad = () => {
|
|
||||||
iframeLoading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// 刷新 iframe
|
|
||||||
const handleRefreshIframe = () => {
|
|
||||||
if (iframeRef.value) {
|
|
||||||
iframeLoading.value = true
|
|
||||||
iframeRef.value.src = iframeRef.value.src
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -137,29 +137,12 @@
|
|||||||
<!-- 主内容区 -->
|
<!-- 主内容区 -->
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
<!-- iframe 模式 -->
|
<!-- iframe 模式 -->
|
||||||
<div v-if="currentIframeUrl" class="iframe-container">
|
<IframeView
|
||||||
<div class="iframe-header">
|
v-if="currentIframeUrl"
|
||||||
<span class="iframe-title">{{ currentMenuItem?.label }}</span>
|
:url="currentIframeUrl"
|
||||||
<el-button
|
:title="currentMenuItem?.label"
|
||||||
text
|
:show-header="true"
|
||||||
@click="handleRefreshIframe"
|
/>
|
||||||
:icon="Refresh"
|
|
||||||
>
|
|
||||||
刷新
|
|
||||||
</el-button>
|
|
||||||
</div>
|
|
||||||
<iframe
|
|
||||||
ref="iframeRef"
|
|
||||||
:src="currentIframeUrl"
|
|
||||||
class="content-iframe"
|
|
||||||
frameborder="0"
|
|
||||||
@load="handleIframeLoad"
|
|
||||||
/>
|
|
||||||
<div v-if="iframeLoading" class="iframe-loading">
|
|
||||||
<Loader :size="20" class="is-loading" />
|
|
||||||
<span>加载中...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- 路由模式 -->
|
<!-- 路由模式 -->
|
||||||
<router-view v-else />
|
<router-view v-else />
|
||||||
@@ -177,13 +160,9 @@ import {
|
|||||||
Server,
|
Server,
|
||||||
Monitor,
|
Monitor,
|
||||||
ChevronDown,
|
ChevronDown,
|
||||||
ChevronRight,
|
ChevronRight
|
||||||
RefreshCw,
|
|
||||||
Loader
|
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
import { IframeView } from '@/components'
|
||||||
// el-button 刷新图标需要传入组件
|
|
||||||
const Refresh = RefreshCw
|
|
||||||
|
|
||||||
// ... (rest of the code remains the same)
|
// ... (rest of the code remains the same)
|
||||||
import type { MenuItem } from '@/types/menu'
|
import type { MenuItem } from '@/types/menu'
|
||||||
@@ -287,8 +266,6 @@ const serviceTitle = computed(() => {
|
|||||||
// 状态管理
|
// 状态管理
|
||||||
const collapsed = ref(false)
|
const collapsed = ref(false)
|
||||||
const activeMenu = ref('home')
|
const activeMenu = ref('home')
|
||||||
const iframeLoading = ref(false)
|
|
||||||
const iframeRef = ref<HTMLIFrameElement>()
|
|
||||||
|
|
||||||
// 从 LocalStorage 获取用户名
|
// 从 LocalStorage 获取用户名
|
||||||
function getUserName(): string {
|
function getUserName(): string {
|
||||||
@@ -488,22 +465,6 @@ const handleMenuClick = (item: MenuItem) => {
|
|||||||
// 所有菜单都通过路由跳转
|
// 所有菜单都通过路由跳转
|
||||||
if (item.url) {
|
if (item.url) {
|
||||||
router.push(item.url)
|
router.push(item.url)
|
||||||
if (item.viewType === 'iframe') {
|
|
||||||
iframeLoading.value = true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// iframe 加载完成
|
|
||||||
const handleIframeLoad = () => {
|
|
||||||
iframeLoading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// 刷新 iframe
|
|
||||||
const handleRefreshIframe = () => {
|
|
||||||
if (iframeRef.value) {
|
|
||||||
iframeLoading.value = true
|
|
||||||
iframeRef.value.src = iframeRef.value.src
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -33,29 +33,12 @@
|
|||||||
<!-- 主内容区 -->
|
<!-- 主内容区 -->
|
||||||
<main class="main-content">
|
<main class="main-content">
|
||||||
<!-- iframe 模式 -->
|
<!-- iframe 模式 -->
|
||||||
<div v-if="currentIframeUrl" class="iframe-container">
|
<IframeView
|
||||||
<div class="iframe-header">
|
v-if="currentIframeUrl"
|
||||||
<span class="iframe-title">{{ currentMenuItem?.label }}</span>
|
:url="currentIframeUrl"
|
||||||
<el-button
|
:title="currentMenuItem?.label"
|
||||||
text
|
:show-header="true"
|
||||||
@click="handleRefreshIframe"
|
/>
|
||||||
:icon="Refresh"
|
|
||||||
>
|
|
||||||
刷新
|
|
||||||
</el-button>
|
|
||||||
</div>
|
|
||||||
<iframe
|
|
||||||
ref="iframeRef"
|
|
||||||
:src="currentIframeUrl"
|
|
||||||
class="content-iframe"
|
|
||||||
frameborder="0"
|
|
||||||
@load="handleIframeLoad"
|
|
||||||
/>
|
|
||||||
<div v-if="iframeLoading" class="iframe-loading">
|
|
||||||
<Loader :size="20" class="is-loading" />
|
|
||||||
<span>加载中...</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<!-- 路由模式 -->
|
<!-- 路由模式 -->
|
||||||
<router-view v-else />
|
<router-view v-else />
|
||||||
@@ -76,13 +59,9 @@ import {
|
|||||||
ChevronRight,
|
ChevronRight,
|
||||||
User,
|
User,
|
||||||
Settings,
|
Settings,
|
||||||
LogOut,
|
LogOut
|
||||||
RefreshCw,
|
|
||||||
Loader
|
|
||||||
} from 'lucide-vue-next'
|
} from 'lucide-vue-next'
|
||||||
|
import { IframeView } from 'shared/components'
|
||||||
// el-button 图标需要传入组件
|
|
||||||
const Refresh = RefreshCw
|
|
||||||
import { ElMessage } from 'element-plus'
|
import { ElMessage } from 'element-plus'
|
||||||
import type { MenuItem } from 'shared/types'
|
import type { MenuItem } from 'shared/types'
|
||||||
|
|
||||||
@@ -92,8 +71,6 @@ const route = useRoute()
|
|||||||
// 状态管理
|
// 状态管理
|
||||||
const collapsed = ref(false)
|
const collapsed = ref(false)
|
||||||
const activeMenu = ref('home')
|
const activeMenu = ref('home')
|
||||||
const iframeLoading = ref(false)
|
|
||||||
const iframeRef = ref<HTMLIFrameElement>()
|
|
||||||
|
|
||||||
// 从 LocalStorage 获取用户名
|
// 从 LocalStorage 获取用户名
|
||||||
function getUserName(): string {
|
function getUserName(): string {
|
||||||
@@ -175,7 +152,7 @@ const menuItems = ref<MenuItem[]>(loadMenuFromStorage())
|
|||||||
|
|
||||||
// 当前菜单项
|
// 当前菜单项
|
||||||
const currentMenuItem = computed(() => {
|
const currentMenuItem = computed(() => {
|
||||||
return menuItems.value.find(item => item.key === activeMenu.value)
|
return menuItems.value.find((item: MenuItem) => item.key === activeMenu.value)
|
||||||
})
|
})
|
||||||
|
|
||||||
// 当前 iframe URL(从路由 meta 读取)
|
// 当前 iframe URL(从路由 meta 读取)
|
||||||
@@ -196,30 +173,9 @@ const handleMenuClick = (item: MenuItem) => {
|
|||||||
// 所有菜单都通过路由跳转
|
// 所有菜单都通过路由跳转
|
||||||
if (item.url) {
|
if (item.url) {
|
||||||
router.push(item.url)
|
router.push(item.url)
|
||||||
if (item.viewType === 'iframe') {
|
|
||||||
iframeLoading.value = true
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// iframe 加载完成
|
|
||||||
const handleIframeLoad = () => {
|
|
||||||
iframeLoading.value = false
|
|
||||||
}
|
|
||||||
|
|
||||||
// 刷新 iframe
|
|
||||||
const handleRefreshIframe = () => {
|
|
||||||
if (iframeRef.value) {
|
|
||||||
iframeLoading.value = true
|
|
||||||
iframeRef.value.src = iframeRef.value.src
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户头像加载错误
|
|
||||||
const handleAvatarError = () => {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
// 用户操作
|
// 用户操作
|
||||||
const handleUserCommand = (command: string) => {
|
const handleUserCommand = (command: string) => {
|
||||||
switch (command) {
|
switch (command) {
|
||||||
|
|||||||
@@ -33,7 +33,7 @@
|
|||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="系统提示词">
|
<el-form-item label="系统提示词">
|
||||||
<el-input v-model="agentConfig.systemPrompt" type="textarea" rows="4" placeholder="输入系统提示词,用于指导AI的行为" />
|
<el-input v-model="agentConfig.systemPrompt" type="textarea" :rows="4" placeholder="输入系统提示词,用于指导AI的行为" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
|
|
||||||
<el-form-item label="启用知识库">
|
<el-form-item label="启用知识库">
|
||||||
@@ -73,7 +73,7 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<div class="test-section">
|
<div class="test-section">
|
||||||
<el-input v-model="testMessage" type="textarea" rows="3" placeholder="输入测试消息..." style="margin-bottom: 12px;" />
|
<el-input v-model="testMessage" type="textarea" :rows="3" placeholder="输入测试消息..." style="margin-bottom: 12px;" />
|
||||||
<el-button type="primary" @click="testAgent">发送测试</el-button>
|
<el-button type="primary" @click="testAgent">发送测试</el-button>
|
||||||
|
|
||||||
<div v-if="testResponse" class="test-response">
|
<div v-if="testResponse" class="test-response">
|
||||||
|
|||||||
@@ -140,7 +140,7 @@
|
|||||||
</el-select>
|
</el-select>
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
<el-form-item label="故障描述">
|
<el-form-item label="故障描述">
|
||||||
<el-input v-model="formData.remark" type="textarea" rows="4" placeholder="请输入故障描述" />
|
<el-input v-model="formData.remark" type="textarea" :rows="4" placeholder="请输入故障描述" />
|
||||||
</el-form-item>
|
</el-form-item>
|
||||||
</el-form>
|
</el-form>
|
||||||
<template #footer>
|
<template #footer>
|
||||||
|
|||||||
@@ -156,7 +156,7 @@
|
|||||||
@input="adjustHeight"
|
@input="adjustHeight"
|
||||||
@keydown="handleKeyDown"
|
@keydown="handleKeyDown"
|
||||||
placeholder="请输入您的问题,例如:电源模块过热报警怎么处理..."
|
placeholder="请输入您的问题,例如:电源模块过热报警怎么处理..."
|
||||||
rows="1"
|
:rows="1"
|
||||||
class="chat-textarea"
|
class="chat-textarea"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
Reference in New Issue
Block a user