| <script setup> | 
| import { nextTick, ref, inject, onMounted } from 'vue'; | 
| import { useRouter } from "vue-router"; | 
| import { get, post, postForm } from '@/utils/request.js' | 
| import { logout } from '@/config.js'; | 
| import * as Icons from "@ant-design/icons-vue"; | 
| import { message } from 'ant-design-vue'; | 
| import { | 
|   MenuUnfoldOutlined, | 
|   MenuFoldOutlined, | 
|   HomeOutlined, | 
|   CloseOutlined, | 
|   RedoOutlined, | 
|   UserOutlined, | 
|   TranslationOutlined, | 
|   ApartmentOutlined, | 
|   CaretLeftOutlined, | 
|   CaretRightOutlined, | 
| } from "@ant-design/icons-vue"; | 
| import { formatMessage, loadData } from '@/utils/localeUtils.js'; | 
| import AiView from '@/components/ai/index.vue' | 
|   | 
| const globalState = inject('globalState'); | 
| const selectedKeys = ref([]); | 
| let openKeys = ref([]); | 
| const collapsed = ref(false); | 
| const router = useRouter(); | 
| let routerCache = ref([]); | 
| let routerCacheList = ref([]); | 
| let currentCache = ref(null); | 
| let isRouterAlive = ref(true); | 
| const menuCache = ref([]); | 
| const hostList = ref([]); | 
| const tabsContent = ref(null); | 
|   | 
| const components = { | 
|   ...Icons, | 
| }; | 
|   | 
| onMounted(() => { | 
|   let name = router.currentRoute.value.name; | 
|   let path = router.currentRoute.value.path; | 
|   if (currentCache.value == null && path != '/') { | 
|     get('/api/menu/get/route', { | 
|       route: path | 
|     }).then((resp) => { | 
|       let result = resp.data; | 
|       let data = result.data; | 
|       if (result.code == 200) { | 
|         currentCache.value = name; | 
|         routerCache.value.push(name) | 
|         routerCacheList.value.push({ | 
|           key: path, | 
|           languageId: data.languageId, | 
|           name: name, | 
|         }) | 
|       } | 
|     }) | 
|   } | 
| }) | 
|   | 
| getMenu() | 
| function getMenu() { | 
|   get('/api/auth/menu', {}).then((result) => { | 
|     menuCache.value = result.data.data; | 
|   }) | 
| } | 
|   | 
| function menuSelect(item) { | 
|   router.push({ | 
|     path: item.key | 
|   }) | 
|   | 
|   let name = item.item.name; | 
|   currentCache.value = name; | 
|   | 
|   if (name != undefined && routerCache.value.indexOf(name) == -1) { | 
|     routerCache.value.push(item.item.name) | 
|     routerCacheList.value.push({ | 
|       key: item.key, | 
|       languageId: item.item.languageId, | 
|       name: item.item.name, | 
|     }) | 
|   } | 
| } | 
|   | 
| function closeTabs(param) { | 
|   let name = param.name; | 
|   let tmp = [] | 
|   let tmpList = []; | 
|   routerCache.value.forEach((item) => { | 
|     if (item != name) { | 
|       tmp.push(item); | 
|     } | 
|   }) | 
|   | 
|   routerCacheList.value.forEach((item) => { | 
|     if (item.name != name) { | 
|       tmpList.push(item); | 
|     } | 
|   }) | 
|   | 
|   if (tmp == 0) { | 
|     router.push({ | 
|       path: '/' | 
|     }) | 
|     routerCache.value.push('home') | 
|     routerCacheList.value.push({ | 
|       key: '/', | 
|       languageId: 'common.home', | 
|       name: '主页', | 
|     }) | 
|     selectedKeys.value = ['/'] | 
|   } else { | 
|     switchTabs(tmpList[0]); | 
|   } | 
|   routerCache.value = tmp; | 
|   routerCacheList.value = tmpList; | 
| } | 
|   | 
| function reloadTabs() { | 
|   const hide = message.loading(formatMessage('common.loading', '加载中')); | 
|   try { | 
|     isRouterAlive.value = false; | 
|     nextTick(() => { | 
|       isRouterAlive.value = true; | 
|       // message.success(formatMessage('common.success', '加载成功')); | 
|     }) | 
|   } catch (error) { | 
|     message.error(formatMessage('common.fail', '加载失败')); | 
|   } finally { | 
|     hide(); | 
|   } | 
| } | 
|   | 
| function closeAllTabs() { | 
|   routerCache.value = []; | 
|   routerCacheList.value = []; | 
|   router.push({ | 
|     path: '/' | 
|   }) | 
| } | 
|   | 
| function switchTabs(item) { | 
|   router.push({ | 
|     path: item.key | 
|   }) | 
|   | 
|   currentCache.value = item.name; | 
|   selectedKeys.value = [item.key] | 
|   | 
|   // open menu | 
|   let arr = item.key.split("/"); | 
|   let key = '/' + arr[1]; | 
|   openKeys.value = [key] | 
| } | 
|   | 
| const switchLocale = async (locale) => { | 
|   globalState.locale = locale; | 
|   localStorage.setItem('locale', locale) | 
|   loadData(locale); | 
|   reloadTabs() | 
| } | 
|   | 
| getHostList() | 
| function getHostList() { | 
|   post('/api/show/host.action', {}).then((resp) => { | 
|     let result = resp.data; | 
|     let data = result.data; | 
|     let hostId = data.hostId; | 
|     if (data.root) { | 
|       post('/api/host/list', {}).then((resp) => { | 
|         let result = resp.data; | 
|         let data = result.data; | 
|         hostList.value = data; | 
|         data.forEach((item) => { | 
|           if (item.id == hostId) { | 
|             globalState.currentHost = item; | 
|           } | 
|         }) | 
|       }) | 
|     } | 
|   }) | 
| } | 
|   | 
| const licenseDays = ref(365); | 
| getLicenseDays(); | 
| function getLicenseDays() { | 
|   post('/api/license/getLicenseDays', {}).then((resp) => { | 
|     let result = resp.data; | 
|     let data = result.data; | 
|     if (result.code == 200) { | 
|       licenseDays.value = data; | 
|     } | 
|   }) | 
| } | 
|   | 
| const switchHost = (item) => { | 
|   globalState.currentHost = item; | 
|   postForm('/api/root/change/host/auth', { | 
|     hostId: item.id | 
|   }).then((resp) => { | 
|     let result = resp.data; | 
|     if (result.code == 200) { | 
|       window.location.reload(); | 
|     } else { | 
|       message.error(formatMessage('common.fail', '加载失败')); | 
|     } | 
|   }) | 
| } | 
|   | 
| const windowReload = () => { | 
|   window.location.reload(); | 
| } | 
|   | 
| const handleScroll = (data) => { | 
|   let position = tabsContent.value.scrollLeft; | 
|   let offset = position * 0.1 + 30; | 
|   if (data == 'left') { | 
|     tabsContent.value.scrollLeft = position - offset; | 
|   } else { | 
|     tabsContent.value.scrollLeft = position + offset; | 
|   } | 
| } | 
|   | 
| </script> | 
|   | 
| <template> | 
|   <a-layout class="main"> | 
|     <a-layout-sider class="main-sider" v-model:collapsed="collapsed" :trigger="null" collapsible theme="dark"> | 
|       <div class="logo" /> | 
|       <a-menu v-model:openKeys="openKeys" v-model:selectedKeys="selectedKeys" @select="menuSelect" theme="dark" | 
|         mode="inline"> | 
|         <div> | 
|           <a-menu-item key="/" name="主页"> | 
|             <HomeOutlined /> {{ formatMessage('common.home', '主页') }} | 
|           </a-menu-item> | 
|         </div> | 
|   | 
|         <div v-for="(item, index) in menuCache" :key="index"> | 
|           <a-sub-menu :key="item.route" v-if="item.type == 0"> | 
|             <template #title> | 
|               <span> | 
|                 <component :is="components[ref(item.icon).value]" /> | 
|                 {{ formatMessage(item.languageId, item.name) }} | 
|               </span> | 
|             </template> | 
|             <div v-for="(child, idx) in item.children"> | 
|               <a-menu-item v-if="child.status == 1" :key="child.route" :name="child.name" | 
|                 :languageId="child.languageId"> | 
|                 {{ formatMessage(child.languageId, child.name) }} | 
|               </a-menu-item> | 
|             </div> | 
|           </a-sub-menu> | 
|         </div> | 
|       </a-menu> | 
|   | 
|     </a-layout-sider> | 
|     <a-layout> | 
|       <a-layout-header style="background: #fff; padding: 0;"> | 
|         <div class="header-top"> | 
|           <div class="header-top-left"> | 
|             <MenuUnfoldOutlined v-if="collapsed" class="trigger triggerLarge" @click="() => (collapsed = !collapsed)" /> | 
|             <MenuFoldOutlined v-else class="trigger" @click="() => (collapsed = !collapsed)" /> | 
|             <RedoOutlined class="trigger" @click="windowReload()" /> | 
|           </div> | 
|           <div class="header-top-right"> | 
|             <div class="trigger" style="color: red;" v-if="licenseDays <= 30"> | 
|               许可证有效期:{{ licenseDays }}天 | 
|             </div> | 
|             <div class="trigger" v-if="globalState.currentHost"> | 
|               <a-dropdown> | 
|                 <div> | 
|                   <ApartmentOutlined /> | 
|                   {{ globalState.currentHost?.name }} | 
|                 </div> | 
|                 <template #overlay> | 
|                   <a-menu> | 
|                     <a-menu-item v-for="(item, index) in hostList" :key="index" @click="switchHost(item)" | 
|                       :class="globalState.currentHost?.id == item.id ? 'active' : ''">{{ item.name }}</a-menu-item> | 
|                   </a-menu> | 
|                 </template> | 
|               </a-dropdown> | 
|             </div> | 
|             <div class="trigger"> | 
|               <a-dropdown> | 
|                 <div> | 
|                   <TranslationOutlined /> | 
|                   {{ globalState.localeList[globalState.locale]?.desc }} | 
|                 </div> | 
|                 <template #overlay> | 
|                   <a-menu> | 
|                     <div v-for="(item, key) in globalState.localeList" :key="key"> | 
|                       <a-menu-item @click="switchLocale(key)" :class="globalState.locale == key ? 'active' : ''">{{ | 
|                         item.desc }}</a-menu-item> | 
|                     </div> | 
|                   </a-menu> | 
|                 </template> | 
|               </a-dropdown> | 
|             </div> | 
|             <div> | 
|               <a-dropdown> | 
|                 <a class="header-user" @click.prevent> | 
|                   <UserOutlined /> | 
|                   <span>{{ globalState.user.username }}</span> | 
|                 </a> | 
|                 <template #overlay> | 
|                   <a-menu @click="logout"> | 
|                     <a-menu-item key="logout">{{ formatMessage('common.account.logout', '退出') }}</a-menu-item> | 
|                   </a-menu> | 
|                 </template> | 
|               </a-dropdown> | 
|             </div> | 
|           </div> | 
|         </div> | 
|       </a-layout-header> | 
|       <a-layout-content class="content-view"> | 
|         <div class="tabs-fixed"> | 
|           <div class="tabs-arrow-left" @click="handleScroll('left')"> | 
|             <CaretLeftOutlined /> | 
|           </div> | 
|   | 
|           <div class="tabs-content" ref="tabsContent"> | 
|             <div class="tabs-content-item"> | 
|               <div v-for="(item, index) in routerCacheList" :key="index" @click="switchTabs(item)" class="tabs-item" | 
|                 :class="currentCache == item.name ? 'tabs-item-active' : ''"> | 
|                 <div :class="currentCache == item.name ? '' : 'tabs-item-reload-none'" @click="reloadTabs" @click.stop> | 
|                   <RedoOutlined /> | 
|                 </div> | 
|                 <div>{{ formatMessage(item.languageId, item.name) }}</div> | 
|                 <div @click="closeTabs(item)" @click.stop> | 
|                   <CloseOutlined /> | 
|                 </div> | 
|               </div> | 
|             </div> | 
|           </div> | 
|   | 
|           <div class="tabs-arrow-right" @click="handleScroll('right')"> | 
|             <CaretRightOutlined /> | 
|           </div> | 
|         </div> | 
|   | 
|         <router-view v-slot="{ Component, route }" v-if="isRouterAlive"> | 
|           <keep-alive :include="routerCache"> | 
|             <component :is="Component" @pageReload="reloadTabs" /> | 
|           </keep-alive> | 
|         </router-view> | 
|       </a-layout-content> | 
|     </a-layout> | 
|   </a-layout> | 
|   | 
|   <AiView /> | 
| </template> | 
|   | 
| <style scoped></style> |