Files
front/src/store/modules/app/index.ts

115 lines
3.9 KiB
TypeScript
Raw Normal View History

2026-03-05 23:45:39 +08:00
import { defineStore } from 'pinia'
import { Notification } from '@arco-design/web-vue'
import type { NotificationReturn } from '@arco-design/web-vue/es/notification/interface'
import type { RouteRecordNormalized } from 'vue-router'
import defaultSettings from '@/config/settings.json'
2026-03-07 20:11:25 +08:00
import { userPmn } from '@/api/module/user'
2026-03-08 22:41:42 +08:00
import { localMenuData, transformMenuToRoutes, type ServerMenuItem } from '@/router/menu-data'
2026-03-07 20:11:25 +08:00
import { buildTree } from '@/utils/tree'
import SafeStorage, { AppStorageKey } from "@/utils/safeStorage";
2026-03-08 22:41:42 +08:00
import router from '@/router'
2026-03-05 23:45:39 +08:00
import { AppState } from './types'
2026-03-07 20:11:25 +08:00
2026-03-05 23:45:39 +08:00
const useAppStore = defineStore('app', {
state: (): AppState => ({ ...defaultSettings }),
getters: {
appCurrentSetting(state: AppState): AppState {
return { ...state }
},
appDevice(state: AppState) {
return state.device
},
appAsyncMenus(state: AppState): RouteRecordNormalized[] {
return state.serverMenu as unknown as RouteRecordNormalized[]
},
},
actions: {
// Update app settings
updateSettings(partial: Partial<AppState>) {
// @ts-ignore-next-line
this.$patch(partial)
},
// Change theme color
toggleTheme(dark: boolean) {
if (dark) {
this.theme = 'dark'
document.body.setAttribute('arco-theme', 'dark')
} else {
this.theme = 'light'
document.body.removeAttribute('arco-theme')
}
},
toggleDevice(device: string) {
this.device = device
},
toggleMenu(value: boolean) {
this.hideMenu = value
},
async fetchServerMenuConfig() {
2026-03-07 20:11:25 +08:00
const userInfo = SafeStorage.get(AppStorageKey.USER_INFO) as any
2026-03-05 23:45:39 +08:00
let notifyInstance: NotificationReturn | null = null
try {
2026-03-07 20:11:25 +08:00
// 使用本地菜单数据(接口未准备好)
// TODO: 接口准备好后,取消下面的注释,使用真实接口数据
const res = await userPmn({ id: userInfo.user_id, workspace: import.meta.env.VITE_APP_WORKSPACE })
console.log('res', res)
if (res.code === 0 && res?.details?.length) {
2026-03-08 22:41:42 +08:00
// 使用 buildTree 将扁平数据构建为树结构
const treeResult = buildTree(res.details[0].permissions as ServerMenuItem[], {
orderKey: 'order'
})
console.log('buildTree', treeResult)
// 使用 transformMenuToRoutes 将树结构转换为路由配置
const routes = transformMenuToRoutes(treeResult.rootItems as ServerMenuItem[])
console.log('transformMenuToRoutes', routes)
// 动态注册路由
routes.forEach((route) => {
// 打印路由结构以便调试
// console.log('Registering route:', JSON.stringify(route, (key, value) => {
// if (typeof value === 'function') return '[Function]'
// return value
// }, 2))
router.addRoute(route as any)
})
this.serverMenu = routes as unknown as RouteRecordNormalized[]
} else {
// 如果接口返回数据为空,使用本地数据
localMenuData.forEach((route) => {
router.addRoute(route as any)
})
this.serverMenu = localMenuData as unknown as RouteRecordNormalized[]
2026-03-07 20:11:25 +08:00
}
2026-03-05 23:45:39 +08:00
} catch (error) {
2026-03-08 22:41:42 +08:00
// 接口失败时使用本地数据
console.error('fetchServerMenuConfig error:', error)
localMenuData.forEach((route) => {
router.addRoute(route as any)
})
this.serverMenu = localMenuData as unknown as RouteRecordNormalized[]
2026-03-05 23:45:39 +08:00
}
},
clearServerMenu() {
2026-03-08 22:41:42 +08:00
// 清除动态注册的路由
if (this.serverMenu && this.serverMenu.length > 0) {
this.serverMenu.forEach((route) => {
const routeName = (route as any).name
if (routeName && typeof routeName === 'string') {
router.removeRoute(routeName)
}
})
}
2026-03-05 23:45:39 +08:00
this.serverMenu = []
},
},
})
export default useAppStore