@gx-design-vue/create-gx-cli 0.1.10 → 0.1.12
Sign up to get free protection for your applications and to get access to all the features.
- package/package.json +2 -1
- package/template-gx-design-thin/.editorconfig +19 -0
- package/template-gx-design-thin/.env +4 -0
- package/template-gx-design-thin/.env.development +15 -0
- package/template-gx-design-thin/.env.production +28 -0
- package/template-gx-design-thin/.eslintignore +16 -0
- package/template-gx-design-thin/.prettierignore +9 -0
- package/template-gx-design-thin/LICENSE +21 -0
- package/template-gx-design-thin/README.md +112 -0
- package/template-gx-design-thin/build/constant.ts +1 -0
- package/template-gx-design-thin/build/generate/generateModifyVars.ts +14 -0
- package/template-gx-design-thin/build/rollupOptions/index.ts +22 -0
- package/template-gx-design-thin/build/util/hash.ts +17 -0
- package/template-gx-design-thin/build/util/index.ts +131 -0
- package/template-gx-design-thin/build/vite/cdn.ts +63 -0
- package/template-gx-design-thin/build/vite/optimizer.ts +26 -0
- package/template-gx-design-thin/build/vite/plugin/appConfig.ts +91 -0
- package/template-gx-design-thin/build/vite/plugin/autoImport.ts +26 -0
- package/template-gx-design-thin/build/vite/plugin/compress.ts +31 -0
- package/template-gx-design-thin/build/vite/plugin/html.ts +39 -0
- package/template-gx-design-thin/build/vite/plugin/index.ts +75 -0
- package/template-gx-design-thin/build/vite/plugin/mock.ts +14 -0
- package/template-gx-design-thin/build/vite/plugin/pwa.ts +31 -0
- package/template-gx-design-thin/build/vite/plugin/visualizer.ts +14 -0
- package/template-gx-design-thin/build/vite/plugin/viteMock/client.ts +88 -0
- package/template-gx-design-thin/build/vite/plugin/viteMock/createMockServer.ts +271 -0
- package/template-gx-design-thin/build/vite/plugin/viteMock/index.ts +69 -0
- package/template-gx-design-thin/build/vite/plugin/viteMock/types.ts +48 -0
- package/template-gx-design-thin/build/vite/plugin/viteMock/utils.ts +48 -0
- package/template-gx-design-thin/build/vite/plugin/viteNotice.ts +40 -0
- package/template-gx-design-thin/commitlint.config.cjs +32 -0
- package/template-gx-design-thin/config/default/defaultSettings.ts +78 -0
- package/template-gx-design-thin/config/default/network.ts +10 -0
- package/template-gx-design-thin/config/default/proxy.ts +47 -0
- package/template-gx-design-thin/config/default/theme.ts +3 -0
- package/template-gx-design-thin/config/index.ts +11 -0
- package/template-gx-design-thin/eslint.config.js +51 -0
- package/template-gx-design-thin/index.html +42 -0
- package/template-gx-design-thin/mock/_createProductionServer.ts +19 -0
- package/template-gx-design-thin/mock/_util.ts +33 -0
- package/template-gx-design-thin/mock/config/menu.ts +21 -0
- package/template-gx-design-thin/mock/config/user.ts +123 -0
- package/template-gx-design-thin/mock/datasSource/system/menu.ts +10 -0
- package/template-gx-design-thin/mock/datasSource/user/account.ts +30 -0
- package/template-gx-design-thin/mock/datasSource/user/index.ts +47 -0
- package/template-gx-design-thin/mock/util/crypto.ts +23 -0
- package/template-gx-design-thin/mock/util/table.ts +92 -0
- package/template-gx-design-thin/mock/util/utils.ts +73 -0
- package/template-gx-design-thin/package.json +129 -0
- package/template-gx-design-thin/pnpm-lock.yaml +12575 -0
- package/template-gx-design-thin/prettier.config.cjs +18 -0
- package/template-gx-design-thin/public/resource/css/index.css +119 -0
- package/template-gx-design-thin/public/resource/css/normalize.css +396 -0
- package/template-gx-design-thin/public/resource/img/favicon.ico +0 -0
- package/template-gx-design-thin/public/resource/img/logo.png +0 -0
- package/template-gx-design-thin/public/resource/img/pro_icon.svg +1 -0
- package/template-gx-design-thin/public/resource/img/pwa-192x192.png +0 -0
- package/template-gx-design-thin/public/resource/img/pwa-512x512.png +0 -0
- package/template-gx-design-thin/src/App.vue +42 -0
- package/template-gx-design-thin/src/assets/error_images/403.png +0 -0
- package/template-gx-design-thin/src/assets/error_images/404.png +0 -0
- package/template-gx-design-thin/src/assets/error_images/cloud.png +0 -0
- package/template-gx-design-thin/src/assets/login_images/login_background.svg +1 -0
- package/template-gx-design-thin/src/assets/logo.png +0 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.css +94 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.eot +0 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.js +1 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.json +142 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.svg +1 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.ttf +0 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.woff +0 -0
- package/template-gx-design-thin/src/assets/menu_font/iconfont.woff2 +0 -0
- package/template-gx-design-thin/src/assets/menu_font/index.less +94 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.css +42 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.eot +0 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.js +1 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.json +51 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.svg +1 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.ttf +0 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.woff +0 -0
- package/template-gx-design-thin/src/assets/public_icon/iconfont.woff2 +0 -0
- package/template-gx-design-thin/src/assets/public_icon/index.less +42 -0
- package/template-gx-design-thin/src/assets/public_images/nodata.svg +1 -0
- package/template-gx-design-thin/src/common/global.ts +4 -0
- package/template-gx-design-thin/src/components/GDesign/Result/index.tsx +144 -0
- package/template-gx-design-thin/src/components/GDesign/Result/style.less +140 -0
- package/template-gx-design-thin/src/components/GDesign/utils/index.ts +7 -0
- package/template-gx-design-thin/src/components/GlobalLayout/Confirm/index.ts +21 -0
- package/template-gx-design-thin/src/components/GlobalLayout/Empty/index.vue +18 -0
- package/template-gx-design-thin/src/components/GlobalLayout/RightContent/index.tsx +126 -0
- package/template-gx-design-thin/src/components/GlobalLayout/RightContent/style.ts +77 -0
- package/template-gx-design-thin/src/components/GlobalLayout/Spin/index.tsx +30 -0
- package/template-gx-design-thin/src/components/PageLoading/index.tsx +51 -0
- package/template-gx-design-thin/src/components/index.ts +6 -0
- package/template-gx-design-thin/src/core/ant-design/index.ts +10 -0
- package/template-gx-design-thin/src/core/gx-admin-design/index.ts +6 -0
- package/template-gx-design-thin/src/core/gx-design/index.ts +6 -0
- package/template-gx-design-thin/src/core/gx-pro-design/index.ts +8 -0
- package/template-gx-design-thin/src/core/index.ts +84 -0
- package/template-gx-design-thin/src/design/ant-design/index.less +4 -0
- package/template-gx-design-thin/src/design/ant-design/layout.less +22 -0
- package/template-gx-design-thin/src/design/ant-design/menu.less +48 -0
- package/template-gx-design-thin/src/design/ant-design/spin.less +23 -0
- package/template-gx-design-thin/src/design/ant-design/tooltip.less +7 -0
- package/template-gx-design-thin/src/design/color.less +1 -0
- package/template-gx-design-thin/src/design/config.less +5 -0
- package/template-gx-design-thin/src/design/imageEditor.less +180 -0
- package/template-gx-design-thin/src/design/index.less +95 -0
- package/template-gx-design-thin/src/design/mixin.less +65 -0
- package/template-gx-design-thin/src/design/normalize.less +391 -0
- package/template-gx-design-thin/src/design/root.less +3 -0
- package/template-gx-design-thin/src/hooks/core/index.ts +3 -0
- package/template-gx-design-thin/src/hooks/core/useRequest/index.ts +118 -0
- package/template-gx-design-thin/src/hooks/event/index.ts +3 -0
- package/template-gx-design-thin/src/hooks/event/useClipboard.ts +15 -0
- package/template-gx-design-thin/src/hooks/web/index.ts +5 -0
- package/template-gx-design-thin/src/hooks/web/useThemeStyle.ts +16 -0
- package/template-gx-design-thin/src/layout/BasicLayout.vue +123 -0
- package/template-gx-design-thin/src/layout/BlankLayout.vue +5 -0
- package/template-gx-design-thin/src/layout/ContentView.vue +50 -0
- package/template-gx-design-thin/src/layout/IframeView.vue +1 -0
- package/template-gx-design-thin/src/layout/UserLayout.vue +7 -0
- package/template-gx-design-thin/src/main.ts +34 -0
- package/template-gx-design-thin/src/plugins/index.ts +2 -0
- package/template-gx-design-thin/src/router/guard/index.ts +83 -0
- package/template-gx-design-thin/src/router/guard/permissions.ts +70 -0
- package/template-gx-design-thin/src/router/guard/stateGuard.ts +10 -0
- package/template-gx-design-thin/src/router/helper/routeHelper.ts +223 -0
- package/template-gx-design-thin/src/router/helper/utils.ts +19 -0
- package/template-gx-design-thin/src/router/index.ts +31 -0
- package/template-gx-design-thin/src/router/routes/index.ts +86 -0
- package/template-gx-design-thin/src/router/routes/modules/dataSource.ts +12 -0
- package/template-gx-design-thin/src/services/common/index.ts +11 -0
- package/template-gx-design-thin/src/services/systemCenter/index.ts +1 -0
- package/template-gx-design-thin/src/services/systemCenter/menu.ts +9 -0
- package/template-gx-design-thin/src/services/userCenter/account.ts +42 -0
- package/template-gx-design-thin/src/services/userCenter/index.ts +28 -0
- package/template-gx-design-thin/src/store/index.ts +32 -0
- package/template-gx-design-thin/src/store/modules/dict.ts +28 -0
- package/template-gx-design-thin/src/store/modules/global.ts +42 -0
- package/template-gx-design-thin/src/store/modules/permission.ts +19 -0
- package/template-gx-design-thin/src/store/modules/routes.ts +113 -0
- package/template-gx-design-thin/src/store/modules/tabsRouter.ts +76 -0
- package/template-gx-design-thin/src/store/modules/user.ts +136 -0
- package/template-gx-design-thin/src/utils/accessToken.ts +80 -0
- package/template-gx-design-thin/src/utils/crypto/base64.ts +101 -0
- package/template-gx-design-thin/src/utils/crypto/index.ts +57 -0
- package/template-gx-design-thin/src/utils/env.ts +50 -0
- package/template-gx-design-thin/src/utils/fetchFile.ts +81 -0
- package/template-gx-design-thin/src/utils/index.ts +123 -0
- package/template-gx-design-thin/src/utils/pageTitle.ts +20 -0
- package/template-gx-design-thin/src/utils/request/XHR.ts +139 -0
- package/template-gx-design-thin/src/utils/request/axiosCancel.ts +69 -0
- package/template-gx-design-thin/src/utils/request/checkStatus.ts +25 -0
- package/template-gx-design-thin/src/utils/request/index.ts +144 -0
- package/template-gx-design-thin/src/utils/request/typings.ts +171 -0
- package/template-gx-design-thin/src/utils/storage.ts +199 -0
- package/template-gx-design-thin/src/utils/uploadFile.ts +27 -0
- package/template-gx-design-thin/src/utils/util.ts +198 -0
- package/template-gx-design-thin/src/utils/validate.ts +216 -0
- package/template-gx-design-thin/src/views/Iframe/index.vue +76 -0
- package/template-gx-design-thin/src/views/Page/one.vue +13 -0
- package/template-gx-design-thin/src/views/Page/two.vue +13 -0
- package/template-gx-design-thin/src/views/exception/403/index.vue +7 -0
- package/template-gx-design-thin/src/views/exception/404/index.vue +9 -0
- package/template-gx-design-thin/src/views/user/login/index.vue +109 -0
- package/template-gx-design-thin/src/views/user/login/style.less +38 -0
- package/template-gx-design-thin/tsconfig.json +46 -0
- package/template-gx-design-thin/types/ant-design-import.d.ts +99 -0
- package/template-gx-design-thin/types/auto-imports.d.ts +81 -0
- package/template-gx-design-thin/types/components.d.ts +24 -0
- package/template-gx-design-thin/types/config.d.ts +44 -0
- package/template-gx-design-thin/types/global.d.ts +97 -0
- package/template-gx-design-thin/types/mock.d.ts +16 -0
- package/template-gx-design-thin/types/module.d.ts +20 -0
- package/template-gx-design-thin/types/response.d.ts +15 -0
- package/template-gx-design-thin/unocss.config.ts +101 -0
- package/template-gx-design-thin/vite.config.ts +120 -0
- package/template-gx-design-thin/yarn.lock +9492 -0
@@ -0,0 +1,50 @@
|
|
1
|
+
<script setup lang="ts">
|
2
|
+
import { computed, ref } from 'vue'
|
3
|
+
import type { BasicLayoutProps, Meta } from '@gx-design-vue/pro-layout'
|
4
|
+
import { PageTranstion } from '@gx-design-vue/pro-layout'
|
5
|
+
import IframeView from '../views/Iframe/index.vue'
|
6
|
+
|
7
|
+
defineProps({
|
8
|
+
reloadStatus: {
|
9
|
+
type: Boolean as VuePropType<boolean>,
|
10
|
+
default: true
|
11
|
+
},
|
12
|
+
animate: {
|
13
|
+
type: Object as VuePropType<BasicLayoutProps['animate']>,
|
14
|
+
default: () => {
|
15
|
+
return {}
|
16
|
+
}
|
17
|
+
}
|
18
|
+
})
|
19
|
+
|
20
|
+
const { global } = useStore()
|
21
|
+
const router = useRouter()
|
22
|
+
|
23
|
+
const keepliveRouterNames = ref([])
|
24
|
+
|
25
|
+
const iframeSrc = computed(() => {
|
26
|
+
const meta = router.currentRoute.value?.meta as Meta
|
27
|
+
return meta?.target && Number(meta?.targetStatus) === 0 ? meta?.target || '' : ''
|
28
|
+
})
|
29
|
+
|
30
|
+
watch(() => global.keepAlive, () => {
|
31
|
+
keepliveRouterNames.value = router.getRoutes()
|
32
|
+
.filter(item => global.keepAlive || (item.meta as Meta)?.keepAlive)
|
33
|
+
.map(item => item.name)
|
34
|
+
}, { immediate: true })
|
35
|
+
</script>
|
36
|
+
|
37
|
+
<template>
|
38
|
+
<router-view>
|
39
|
+
<template #default="{ Component }">
|
40
|
+
<PageTranstion v-bind="animate">
|
41
|
+
<template v-if="reloadStatus">
|
42
|
+
<keep-alive :include="keepliveRouterNames">
|
43
|
+
<component :is="Component" />
|
44
|
+
</keep-alive>
|
45
|
+
</template>
|
46
|
+
</PageTranstion>
|
47
|
+
</template>
|
48
|
+
</router-view>
|
49
|
+
<IframeView v-if="iframeSrc" :frameSrc="iframeSrc" />
|
50
|
+
</template>
|
@@ -0,0 +1 @@
|
|
1
|
+
<template></template>
|
@@ -0,0 +1,34 @@
|
|
1
|
+
import { createApp } from 'vue'
|
2
|
+
|
3
|
+
import App from './App.vue'
|
4
|
+
import { setupStore } from './store'
|
5
|
+
import { setupGlobCommon } from './core'
|
6
|
+
import { router, setupRouter } from './router'
|
7
|
+
import { setupRouterGuard } from './router/guard'
|
8
|
+
|
9
|
+
import 'uno.css'
|
10
|
+
import 'nprogress/nprogress.css'
|
11
|
+
import './design/index.less'
|
12
|
+
|
13
|
+
// plugins
|
14
|
+
import './plugins'
|
15
|
+
|
16
|
+
function startApp() {
|
17
|
+
const app = createApp(App)
|
18
|
+
|
19
|
+
// store
|
20
|
+
setupStore(app)
|
21
|
+
|
22
|
+
// global
|
23
|
+
setupGlobCommon(app)
|
24
|
+
|
25
|
+
// route
|
26
|
+
setupRouter(app)
|
27
|
+
|
28
|
+
// guardRoute
|
29
|
+
setupRouterGuard(router)
|
30
|
+
|
31
|
+
app.mount('#app')
|
32
|
+
}
|
33
|
+
|
34
|
+
startApp()
|
@@ -0,0 +1,83 @@
|
|
1
|
+
import type { Router } from 'vue-router'
|
2
|
+
import type { MenuDataItem } from '@gx-design-vue/pro-layout'
|
3
|
+
import NProgress from 'nprogress'
|
4
|
+
import { defaultSettings } from '@gx-config'
|
5
|
+
import getPageTitle from '@/utils/pageTitle'
|
6
|
+
import { scrollToContainer } from '@/utils/util'
|
7
|
+
import { createStateGuard } from './stateGuard'
|
8
|
+
import { createPermissionGuard } from './permissions'
|
9
|
+
|
10
|
+
const { routesWhiteList } = defaultSettings
|
11
|
+
|
12
|
+
export function setupRouterGuard(router: Router) {
|
13
|
+
createPageGuard(router)
|
14
|
+
createPageLoadingGuard(router)
|
15
|
+
createScrollGuard(router)
|
16
|
+
createProgressGuard(router)
|
17
|
+
createPermissionGuard(router)
|
18
|
+
createStateGuard(router)
|
19
|
+
}
|
20
|
+
|
21
|
+
export function createPageGuard(router: Router) {
|
22
|
+
const routes = useStoreRoutes()
|
23
|
+
const global = useStoreGlobal()
|
24
|
+
|
25
|
+
router.afterEach((to) => {
|
26
|
+
const { meta } = to as MenuDataItem
|
27
|
+
document.title = getPageTitle(meta.title || '')
|
28
|
+
if (
|
29
|
+
global.globalLayout.layout !== 'wide' && routes.routerLoadList.every(item => item !== to.path) && routesWhiteList.includes(
|
30
|
+
to.path)
|
31
|
+
) {
|
32
|
+
routes.addRouterLoadList(to.path)
|
33
|
+
}
|
34
|
+
})
|
35
|
+
}
|
36
|
+
|
37
|
+
export function createPageLoadingGuard(router: Router) {
|
38
|
+
const routes = useStoreRoutes()
|
39
|
+
const global = useStoreGlobal()
|
40
|
+
|
41
|
+
router.beforeEach(async (to) => {
|
42
|
+
if (
|
43
|
+
global.globalLayout.layout !== 'wide' &&
|
44
|
+
routes.routerLoadList.every(item => item !== to.path) &&
|
45
|
+
routesWhiteList.includes(to.path)
|
46
|
+
) {
|
47
|
+
routes.setRouteState({
|
48
|
+
routerLoading: true
|
49
|
+
})
|
50
|
+
}
|
51
|
+
|
52
|
+
return true
|
53
|
+
})
|
54
|
+
|
55
|
+
router.afterEach((_) => {
|
56
|
+
setTimeout(() => {
|
57
|
+
routes.setRouteState({
|
58
|
+
routerLoading: false
|
59
|
+
})
|
60
|
+
}, global.globalLayout.layout === 'wide' ? 0 : 200)
|
61
|
+
})
|
62
|
+
}
|
63
|
+
|
64
|
+
export function createScrollGuard(router: Router) {
|
65
|
+
const global = useStoreGlobal()
|
66
|
+
|
67
|
+
router.afterEach((_) => {
|
68
|
+
!global.disabledScrollTop && scrollToContainer({ count: 0 })
|
69
|
+
})
|
70
|
+
}
|
71
|
+
|
72
|
+
export function createProgressGuard(router: Router) {
|
73
|
+
const global = useStoreGlobal()
|
74
|
+
router.beforeEach(() => {
|
75
|
+
if (global.showProgressBar)
|
76
|
+
NProgress.start()
|
77
|
+
return true
|
78
|
+
})
|
79
|
+
|
80
|
+
router.afterEach(() => {
|
81
|
+
NProgress.done()
|
82
|
+
})
|
83
|
+
}
|
@@ -0,0 +1,70 @@
|
|
1
|
+
import type { Router } from 'vue-router'
|
2
|
+
import { defaultSettings } from '@gx-config'
|
3
|
+
|
4
|
+
const { authentication, loginInterception, recordRoute, routesWhiteList } = defaultSettings
|
5
|
+
|
6
|
+
export function createPermissionGuard(router: Router) {
|
7
|
+
const userStore = useStoreUser()
|
8
|
+
const routeStore = useStoreRoutes()
|
9
|
+
const permissionStore = useStorePermission()
|
10
|
+
|
11
|
+
router.beforeEach(async (to, _, next) => {
|
12
|
+
if (loginInterception) {
|
13
|
+
const token = userStore.accessToken
|
14
|
+
if (routesWhiteList.includes(to.path)) {
|
15
|
+
if (token) {
|
16
|
+
next({ path: '/', replace: true })
|
17
|
+
}
|
18
|
+
next()
|
19
|
+
return
|
20
|
+
}
|
21
|
+
if (!token) {
|
22
|
+
userStore.resetPermissions()
|
23
|
+
if (recordRoute) {
|
24
|
+
next({ path: '/user/login', query: { redirect: to.path }, replace: true })
|
25
|
+
} else {
|
26
|
+
next({ path: '/user/login', replace: true })
|
27
|
+
}
|
28
|
+
return
|
29
|
+
}
|
30
|
+
|
31
|
+
const hasRoles = permissionStore.role.length > 0
|
32
|
+
if (hasRoles) {
|
33
|
+
next()
|
34
|
+
return
|
35
|
+
}
|
36
|
+
|
37
|
+
const checkUserPremission = await userStore.checkUserPremission()
|
38
|
+
if (!checkUserPremission) {
|
39
|
+
userStore.resetPermissions()
|
40
|
+
if (recordRoute) {
|
41
|
+
next({ path: '/user/login', query: { redirect: to.path }, replace: true })
|
42
|
+
} else {
|
43
|
+
next({ path: '/user/login', replace: true })
|
44
|
+
}
|
45
|
+
return
|
46
|
+
}
|
47
|
+
|
48
|
+
const routes = authentication === 'all'
|
49
|
+
? await routeStore.setAllRoutes()
|
50
|
+
: await routeStore.setRoutes()
|
51
|
+
routeStore.setRouteState({
|
52
|
+
meunLoading: false
|
53
|
+
})
|
54
|
+
if (routes?.length) {
|
55
|
+
routes.forEach((route) => {
|
56
|
+
router.addRoute(route as any)
|
57
|
+
})
|
58
|
+
|
59
|
+
console.log(routes)
|
60
|
+
|
61
|
+
next({ path: to.fullPath, replace: true })
|
62
|
+
return
|
63
|
+
}
|
64
|
+
|
65
|
+
next({ path: '/exception/403', replace: true })
|
66
|
+
}
|
67
|
+
|
68
|
+
next()
|
69
|
+
})
|
70
|
+
}
|
@@ -0,0 +1,223 @@
|
|
1
|
+
import { cloneDeep } from 'lodash-es'
|
2
|
+
import type { AppRouteModule, MenuDataItem } from '@gx-design-vue/pro-layout'
|
3
|
+
import { getMaxFloor } from '@gx-design-vue/pro-utils'
|
4
|
+
import { warning } from '@gx-design/utils'
|
5
|
+
|
6
|
+
/**
|
7
|
+
* @description: default layout
|
8
|
+
*/
|
9
|
+
export const BasicLayout = () => import('@/layout/BasicLayout.vue')
|
10
|
+
export const IframeView = () => import('@/layout/IframeView.vue')
|
11
|
+
export const EXCEPTION_COMPONENT = () => import('@/views/exception/404/index.vue')
|
12
|
+
|
13
|
+
const LayoutMap = new Map<string, any>()
|
14
|
+
|
15
|
+
LayoutMap.set('BasicLayout', BasicLayout)
|
16
|
+
LayoutMap.set('IframeView', IframeView)
|
17
|
+
|
18
|
+
let dynamicViewsModules: Record<string, () => Promise<Recordable>>
|
19
|
+
|
20
|
+
/**
|
21
|
+
* @Author gx12358
|
22
|
+
* @DateTime 2021/5/14
|
23
|
+
* @lastTime 2021/5/14
|
24
|
+
* @description 根路由
|
25
|
+
*/
|
26
|
+
const rootRouter: MenuDataItem[] = buildMenu([
|
27
|
+
{
|
28
|
+
path: '/',
|
29
|
+
name: 'index',
|
30
|
+
meta: { title: '首页' },
|
31
|
+
component: 'BasicLayout',
|
32
|
+
redirect: '',
|
33
|
+
children: []
|
34
|
+
}
|
35
|
+
])
|
36
|
+
|
37
|
+
// Dynamic introduction
|
38
|
+
function asyncImportRoute(component: string) {
|
39
|
+
dynamicViewsModules = dynamicViewsModules || import.meta.glob('../../views/**/*.{vue,tsx}')
|
40
|
+
return dynamicImport(dynamicViewsModules, component)
|
41
|
+
}
|
42
|
+
|
43
|
+
function dynamicImport(
|
44
|
+
dynamicViewsModules: Record<string, () => Promise<Recordable>>,
|
45
|
+
component: string
|
46
|
+
) {
|
47
|
+
const keys = Object.keys(dynamicViewsModules)
|
48
|
+
const matchKeys = keys.filter((key) => {
|
49
|
+
const k = key.replace('../../views', '')
|
50
|
+
if (!component)
|
51
|
+
return false
|
52
|
+
const startFlag = component.startsWith('/')
|
53
|
+
const endFlag = component.endsWith('.vue') || component.endsWith('.tsx')
|
54
|
+
const startIndex = startFlag ? 0 : 1
|
55
|
+
const lastIndex = endFlag ? k.length : k.lastIndexOf('.')
|
56
|
+
return k.substring(startIndex, lastIndex) === component
|
57
|
+
})
|
58
|
+
if (matchKeys?.length === 1) {
|
59
|
+
const matchKey = matchKeys[0]
|
60
|
+
return dynamicViewsModules[matchKey]
|
61
|
+
} else if (matchKeys?.length > 1) {
|
62
|
+
warning(
|
63
|
+
true,
|
64
|
+
'请不要在views文件夹下的同一层次目录中创建具有相同文件名的“.vue”和“.TSX”文件。这将导致动态导入失败'
|
65
|
+
)
|
66
|
+
} else {
|
67
|
+
warning(
|
68
|
+
true,
|
69
|
+
'在src/views/下找不到`' + component + '.vue` 或 `' + component + '.tsx`, 请自行创建!'
|
70
|
+
)
|
71
|
+
return EXCEPTION_COMPONENT
|
72
|
+
}
|
73
|
+
|
74
|
+
return false
|
75
|
+
}
|
76
|
+
|
77
|
+
/**
|
78
|
+
* @Author gx12358
|
79
|
+
* @DateTime 2021/5/14
|
80
|
+
* @lastTime 2021/5/14
|
81
|
+
* @description 将后台树形数据菜单和本地菜单结合
|
82
|
+
*/
|
83
|
+
export function getRootMenu(rows: MenuDataItem[]): MenuDataItem[] {
|
84
|
+
let menus: MenuDataItem[] = []
|
85
|
+
if (getMaxFloor(cloneDeep(rows)) > 1) {
|
86
|
+
menus = buildMenu(rows)
|
87
|
+
} else {
|
88
|
+
buildtree(rows, menus, 0)
|
89
|
+
}
|
90
|
+
rootRouter[0].children = menus
|
91
|
+
return cloneDeep(rootRouter)
|
92
|
+
}
|
93
|
+
|
94
|
+
/**
|
95
|
+
* @Author gx12358
|
96
|
+
* @DateTime 2021/5/14
|
97
|
+
* @lastTime 2021/5/14
|
98
|
+
* @description 格式化 后端 结构信息并递归生成层级路由表
|
99
|
+
*/
|
100
|
+
export const generator = (routerMap: MenuDataItem[], parent?: AppRouteModule) => {
|
101
|
+
return routerMap.map((item: MenuDataItem) => {
|
102
|
+
const parentPath = parent?.path || ''
|
103
|
+
const currentRouter: AppRouteModule = {
|
104
|
+
// 路由地址 动态拼接生成如 /dashboard/workplace
|
105
|
+
path: parentPath
|
106
|
+
? `${parent.path === '/' ? '' : parentPath}/${item.path}`
|
107
|
+
: `/${item.path}`,
|
108
|
+
// 路由名称,建议唯一
|
109
|
+
name: item.name || '',
|
110
|
+
// 该路由对应页面的 组件 优先根据组件名或者key从constantRouterComponents获取,没有则通过组件名地址查询
|
111
|
+
component: item.component
|
112
|
+
? (LayoutMap.get(item.component || item.key) || asyncImportRoute(item.component))
|
113
|
+
: undefined,
|
114
|
+
// meta: 页面标题, 菜单图标, 页面权限(供指令权限用,可去掉)
|
115
|
+
redirect: item.redirect,
|
116
|
+
meta: {
|
117
|
+
title: item.title || '',
|
118
|
+
tagFixed: item.tagFixed as boolean,
|
119
|
+
tagHidden: item.tagHidden as boolean,
|
120
|
+
icon: item.icon || undefined,
|
121
|
+
homePage: item.homePage || 0,
|
122
|
+
iconType: item.iconType || undefined,
|
123
|
+
hideInMenu: item.hideInMenu || false,
|
124
|
+
hideChildrenInMenu: item.hideChildrenInMenu || false,
|
125
|
+
target: item.target,
|
126
|
+
keepAlive: item.keepAlive,
|
127
|
+
targetStatus: item.targetStatus,
|
128
|
+
animateDisabled: item.animateDisabled,
|
129
|
+
currenFulltPath: item.currenFulltPath
|
130
|
+
}
|
131
|
+
}
|
132
|
+
// 为了防止出现后端返回结果不规范,处理有可能出现拼接出两个 反斜杠
|
133
|
+
if (!currentRouter.path.startsWith('http')) {
|
134
|
+
currentRouter.path = currentRouter.path.replace('//', '/')
|
135
|
+
}
|
136
|
+
// 重定向
|
137
|
+
item.redirect && (currentRouter.redirect = item.redirect)
|
138
|
+
// 是否有子菜单,并递归处理
|
139
|
+
if (item.children && item.children.length > 0) {
|
140
|
+
// Recursion
|
141
|
+
currentRouter.children = generator(item.children, currentRouter)
|
142
|
+
}
|
143
|
+
return currentRouter
|
144
|
+
})
|
145
|
+
}
|
146
|
+
|
147
|
+
function handleMenuParams(menuItem: MenuDataItem): MenuDataItem {
|
148
|
+
const {
|
149
|
+
title = '',
|
150
|
+
menuType,
|
151
|
+
icon = '',
|
152
|
+
iconType = 1, // 菜单图标类型 0:本地 1:自定义 2:图片
|
153
|
+
tagFixed = '1', // 标签栏固定状态(标签栏路由地址是否固定(只有标签栏为显示转态才生效))0:是 1:否
|
154
|
+
tagHidden = '0', // 标签栏显示状态(隐藏的路由是否显示在标签栏中(只有标签栏为显示转态才生效))0:显示 1:隐藏
|
155
|
+
homePageFlag = 0, // 是否为主页(选择后为登录后跳转改地址,不选择默认跳转 /)0:否 1:是
|
156
|
+
isFrame = '1', // 是否外链 0:是 1:否
|
157
|
+
keepAlive = false,
|
158
|
+
animateDisabled = false,
|
159
|
+
redirect,
|
160
|
+
currenFulltPath,
|
161
|
+
outLinkType = 0 // 外链类型(选择是系统内则以iframe形式在系统内部展示,否则跳转新页面打开) 0:系统内 1:系统外
|
162
|
+
} = menuItem.meta ?? menuItem
|
163
|
+
return {
|
164
|
+
title,
|
165
|
+
name: menuItem.name || title,
|
166
|
+
key: menuItem.name || title,
|
167
|
+
icon,
|
168
|
+
keepAlive,
|
169
|
+
menuType,
|
170
|
+
iconType: iconType || 1,
|
171
|
+
hideInMenu: !!menuItem.hidden,
|
172
|
+
hideChildrenInMenu: !!menuItem.hideChildrenInMenu,
|
173
|
+
homePage: homePageFlag,
|
174
|
+
path: menuItem.path && menuItem.path.length > 0
|
175
|
+
? menuItem.path
|
176
|
+
: undefined,
|
177
|
+
component: menuItem.component,
|
178
|
+
redirect: redirect === 'noRedirect'
|
179
|
+
? ''
|
180
|
+
: redirect,
|
181
|
+
animateDisabled,
|
182
|
+
currenFulltPath,
|
183
|
+
tagFixed: tagFixed === '0',
|
184
|
+
tagHidden: tagHidden === '1',
|
185
|
+
target: isFrame === '0' ? menuItem.target : '',
|
186
|
+
targetStatus: outLinkType || 0
|
187
|
+
}
|
188
|
+
}
|
189
|
+
|
190
|
+
/**
|
191
|
+
* @Author gx12358
|
192
|
+
* @DateTime 2021/5/14
|
193
|
+
* @lastTime 2021/5/14
|
194
|
+
* @description 将后台树形结构菜单数据添加后修改属性(具体修改看后台返回值)
|
195
|
+
*/
|
196
|
+
export function buildMenu(list: MenuDataItem[]) {
|
197
|
+
return list.map((muenuItem) => {
|
198
|
+
return { ...handleMenuParams(muenuItem), children: buildMenu(muenuItem.children || []) }
|
199
|
+
})
|
200
|
+
}
|
201
|
+
|
202
|
+
/**
|
203
|
+
* @Author gx12358
|
204
|
+
* @DateTime 2021/5/14
|
205
|
+
* @lastTime 2021/5/14
|
206
|
+
* @description 将后台菜单数据变成树形结构(具体修改看后台返回值)
|
207
|
+
*/
|
208
|
+
export function buildtree(
|
209
|
+
rootMenu: MenuDataItem[],
|
210
|
+
menuList: MenuDataItem[],
|
211
|
+
parentId: string | number
|
212
|
+
) {
|
213
|
+
rootMenu.forEach((muenuItem) => {
|
214
|
+
if (muenuItem.parentId === parentId) {
|
215
|
+
const child: MenuDataItem = { ...handleMenuParams(muenuItem), children: [] }
|
216
|
+
buildtree(rootMenu, child.children, muenuItem.menuId)
|
217
|
+
if (child.children.length === 0) {
|
218
|
+
delete child.children
|
219
|
+
}
|
220
|
+
menuList.push(child)
|
221
|
+
}
|
222
|
+
})
|
223
|
+
}
|
@@ -0,0 +1,19 @@
|
|
1
|
+
export function getFirstLastChild(data: any[]) {
|
2
|
+
let newPath
|
3
|
+
// 获取第一个children的path
|
4
|
+
const getRoutePath = function (newdata) {
|
5
|
+
let firstPath = ''
|
6
|
+
if (newdata.children && newdata.children.length > 0) {
|
7
|
+
firstPath = getRoutePath(newdata.children[0])
|
8
|
+
} else {
|
9
|
+
firstPath = `${newdata.path}`
|
10
|
+
}
|
11
|
+
return firstPath
|
12
|
+
}
|
13
|
+
if (data[0].children && data[0].children.length > 0) {
|
14
|
+
newPath = getRoutePath(data[0].children[0])
|
15
|
+
} else {
|
16
|
+
newPath = data[0].path
|
17
|
+
}
|
18
|
+
return newPath
|
19
|
+
}
|
@@ -0,0 +1,31 @@
|
|
1
|
+
import type { App } from 'vue'
|
2
|
+
import type { RouteRecordRaw } from 'vue-router'
|
3
|
+
import { createRouter, createWebHashHistory, createWebHistory } from 'vue-router'
|
4
|
+
import { defaultSettings } from '@gx-config'
|
5
|
+
import { constantRoutes } from './routes'
|
6
|
+
|
7
|
+
const { routerMode } = defaultSettings
|
8
|
+
|
9
|
+
function handleRouterMode() {
|
10
|
+
switch (routerMode) {
|
11
|
+
case 'hash':
|
12
|
+
return createWebHashHistory()
|
13
|
+
break
|
14
|
+
case 'browser':
|
15
|
+
return createWebHistory()
|
16
|
+
break
|
17
|
+
default:
|
18
|
+
return createWebHashHistory()
|
19
|
+
break
|
20
|
+
}
|
21
|
+
}
|
22
|
+
|
23
|
+
export const router = createRouter({
|
24
|
+
history: handleRouterMode(),
|
25
|
+
routes: constantRoutes as unknown as RouteRecordRaw[]
|
26
|
+
})
|
27
|
+
|
28
|
+
// 配置路由器
|
29
|
+
export function setupRouter(app: App<Element>) {
|
30
|
+
app.use(router)
|
31
|
+
}
|
@@ -0,0 +1,86 @@
|
|
1
|
+
import type { AppRouteModule } from '@gx-design-vue/pro-layout'
|
2
|
+
import UserLayout from '@/layout/UserLayout.vue'
|
3
|
+
import BlankLayout from '@/layout/BlankLayout.vue'
|
4
|
+
import BasicLayout from '@/layout/BasicLayout.vue'
|
5
|
+
|
6
|
+
// import.meta.globEager() 直接引入所有的模块 Vite 独有的功能
|
7
|
+
const modules = import.meta.glob('./modules/**/*.ts', { eager: true })
|
8
|
+
const routeModuleList: AppRouteModule[] = []
|
9
|
+
|
10
|
+
// 加入到路由集合中
|
11
|
+
Object.keys(modules).forEach((key) => {
|
12
|
+
const mod = (modules[key] as any).default || {}
|
13
|
+
const modList = Array.isArray(mod) ? [ ...mod ] : [ mod ]
|
14
|
+
routeModuleList.push(...modList)
|
15
|
+
})
|
16
|
+
|
17
|
+
// 基本路由
|
18
|
+
export const constantRoutes: AppRouteModule[] = [
|
19
|
+
{
|
20
|
+
path: '/user',
|
21
|
+
component: UserLayout,
|
22
|
+
redirect: '/user/login',
|
23
|
+
children: [
|
24
|
+
{
|
25
|
+
path: '/user/login',
|
26
|
+
name: 'Login',
|
27
|
+
meta: {
|
28
|
+
hideInMenu: true,
|
29
|
+
title: '登录'
|
30
|
+
},
|
31
|
+
component: () => import('@/views/user/login/index.vue')
|
32
|
+
}
|
33
|
+
]
|
34
|
+
},
|
35
|
+
{
|
36
|
+
path: '/exception',
|
37
|
+
name: 'Exception',
|
38
|
+
component: BlankLayout,
|
39
|
+
redirect: '/exception/403',
|
40
|
+
meta: {
|
41
|
+
hideInMenu: true,
|
42
|
+
title: '错误页',
|
43
|
+
icon: 'error-warning-line'
|
44
|
+
},
|
45
|
+
children: [
|
46
|
+
{
|
47
|
+
path: '/exception/403',
|
48
|
+
name: 'Error403',
|
49
|
+
component: () => import('@/views/exception/403/index.vue'),
|
50
|
+
meta: {
|
51
|
+
hideInMenu: true,
|
52
|
+
title: '403',
|
53
|
+
icon: 'error-warning-line'
|
54
|
+
}
|
55
|
+
},
|
56
|
+
{
|
57
|
+
path: '/exception/404',
|
58
|
+
name: 'Error404',
|
59
|
+
component: () => import('@/views/exception/404/index.vue'),
|
60
|
+
meta: {
|
61
|
+
hideInMenu: true,
|
62
|
+
title: '404',
|
63
|
+
icon: 'error-warning-line'
|
64
|
+
}
|
65
|
+
}
|
66
|
+
]
|
67
|
+
}
|
68
|
+
]
|
69
|
+
|
70
|
+
// authentication为all(后端生成的路由)本地路由
|
71
|
+
export const asyncRoutes: AppRouteModule[] = routeModuleList
|
72
|
+
|
73
|
+
// 自定义路由
|
74
|
+
export const localRoutes: AppRouteModule[] = [
|
75
|
+
{
|
76
|
+
path: '/',
|
77
|
+
name: '首页',
|
78
|
+
component: BasicLayout,
|
79
|
+
redirect: '/workplace',
|
80
|
+
children: routeModuleList
|
81
|
+
},
|
82
|
+
{
|
83
|
+
path: '/:path(.*)*',
|
84
|
+
redirect: '/exception/404'
|
85
|
+
}
|
86
|
+
]
|
@@ -0,0 +1,11 @@
|
|
1
|
+
import type { DownLoadRequestConfig } from '@/utils/fetchFile'
|
2
|
+
import fetchFile from '@/utils/fetchFile'
|
3
|
+
|
4
|
+
export function download(params: DownLoadRequestConfig) {
|
5
|
+
const { showTip = true, method = 'get' } = params
|
6
|
+
return fetchFile({
|
7
|
+
...params,
|
8
|
+
showTip,
|
9
|
+
method
|
10
|
+
})
|
11
|
+
}
|
@@ -0,0 +1 @@
|
|
1
|
+
export * from './menu'
|