mirror of
https://github.com/WeBankFinTech/fes.js.git
synced 2025-04-06 03:59:53 +08:00
feat: layout支持multiTabs参数
This commit is contained in:
parent
1746696472
commit
6eb4bdd050
@ -15,7 +15,7 @@ export function rootContainer(childComponent, args) {
|
|||||||
userConfig.menus = fillMenuData(userConfig.menus, routeConfig);
|
userConfig.menus = fillMenuData(userConfig.menus, routeConfig);
|
||||||
return () => {
|
return () => {
|
||||||
const slots = {
|
const slots = {
|
||||||
default: () => <childComponent></childComponent>,
|
default: () => <childComponent keepAlive={userConfig.multiTabs}></childComponent>,
|
||||||
userCenter: () => {
|
userCenter: () => {
|
||||||
if(runtimeConfig.userCenter){
|
if(runtimeConfig.userCenter){
|
||||||
return (<runtimeConfig.userCenter></runtimeConfig.userCenter>)
|
return (<runtimeConfig.userCenter></runtimeConfig.userCenter>)
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
<template>
|
<template>
|
||||||
<a-layout class="main-layout">
|
<a-layout class="main-layout">
|
||||||
<a-layout-sider
|
<a-layout-sider
|
||||||
v-if="routeLayout"
|
v-if="routeHasLayout"
|
||||||
v-model:collapsed="collapsed"
|
v-model:collapsed="collapsed"
|
||||||
:width="sideWidth"
|
:width="sideWidth"
|
||||||
:class="{ collapsed: collapsed }"
|
:class="{ collapsed: collapsed }"
|
||||||
@ -16,16 +16,22 @@
|
|||||||
<Menu :menus="menus" :theme="theme" />
|
<Menu :menus="menus" :theme="theme" />
|
||||||
</a-layout-sider>
|
</a-layout-sider>
|
||||||
<a-layout>
|
<a-layout>
|
||||||
<a-layout-header v-if="routeLayout" class="layout-header">
|
<a-layout-header v-if="routeHasLayout" class="layout-header">
|
||||||
<div class="layout-header-user">
|
<div class="layout-header-user">
|
||||||
<slot name="userCenter"></slot>
|
<slot name="userCenter"></slot>
|
||||||
</div>
|
</div>
|
||||||
<slot name="locale"></slot>
|
<slot name="locale"></slot>
|
||||||
</a-layout-header>
|
</a-layout-header>
|
||||||
<a-layout-content class="layout-content">
|
<a-layout-content class="layout-content">
|
||||||
|
<template v-if="multiTabs">
|
||||||
|
<a-tabs :activeKey="route.path" @tabClick="switchTab" class="layout-content-tabs" hide-add type="editable-card">
|
||||||
|
<a-tab-pane v-for="page in openedPageList" :key="page.path" :tab="page.meta.title" closable>
|
||||||
|
</a-tab-pane>
|
||||||
|
</a-tabs>
|
||||||
|
</template>
|
||||||
<slot></slot>
|
<slot></slot>
|
||||||
</a-layout-content>
|
</a-layout-content>
|
||||||
<a-layout-footer v-if="routeLayout" class="layout-footer">
|
<a-layout-footer v-if="routeHasLayout" class="layout-footer">
|
||||||
Ant Design ©2020 Created by MumbleFe
|
Ant Design ©2020 Created by MumbleFe
|
||||||
</a-layout-footer>
|
</a-layout-footer>
|
||||||
</a-layout>
|
</a-layout>
|
||||||
@ -33,10 +39,14 @@
|
|||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
import { ref, computed } from 'vue';
|
import {
|
||||||
import { useRoute } from '@@/core/coreExports';
|
ref, computed, reactive, unref
|
||||||
|
} from 'vue';
|
||||||
|
import { useRoute, useRouter } from '@@/core/coreExports';
|
||||||
import Layout from 'ant-design-vue/lib/layout';
|
import Layout from 'ant-design-vue/lib/layout';
|
||||||
import 'ant-design-vue/lib/layout/style';
|
import 'ant-design-vue/lib/layout/style';
|
||||||
|
import Tabs from 'ant-design-vue/lib/tabs';
|
||||||
|
import 'ant-design-vue/lib/tabs/style';
|
||||||
import Menu from './Menu';
|
import Menu from './Menu';
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
@ -46,6 +56,8 @@ export default {
|
|||||||
[Layout.Content.name]: Layout.Content,
|
[Layout.Content.name]: Layout.Content,
|
||||||
[Layout.Header.name]: Layout.Header,
|
[Layout.Header.name]: Layout.Header,
|
||||||
[Layout.Footer.name]: Layout.Footer,
|
[Layout.Footer.name]: Layout.Footer,
|
||||||
|
[Tabs.name]: Tabs,
|
||||||
|
[Tabs.TabPane.name]: Tabs.TabPane,
|
||||||
Menu
|
Menu
|
||||||
},
|
},
|
||||||
props: {
|
props: {
|
||||||
@ -94,12 +106,27 @@ export default {
|
|||||||
},
|
},
|
||||||
setup() {
|
setup() {
|
||||||
const route = useRoute();
|
const route = useRoute();
|
||||||
const routeLayout = computed(() => {
|
const router = useRouter();
|
||||||
|
const openedPageList = reactive([]);
|
||||||
|
const routeHasLayout = computed(() => {
|
||||||
const _routeLayout = route.meta.layout;
|
const _routeLayout = route.meta.layout;
|
||||||
return _routeLayout === undefined ? true : _routeLayout;
|
return _routeLayout === undefined ? true : _routeLayout;
|
||||||
});
|
});
|
||||||
|
router.beforeEach((to) => {
|
||||||
|
if (!openedPageList.some(page => unref(page.path) === to.path)) {
|
||||||
|
openedPageList.push(to);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
});
|
||||||
|
// 还需要考虑参数
|
||||||
|
const switchTab = (path) => {
|
||||||
|
router.push(path);
|
||||||
|
};
|
||||||
return {
|
return {
|
||||||
routeLayout,
|
switchTab,
|
||||||
|
route,
|
||||||
|
openedPageList,
|
||||||
|
routeHasLayout,
|
||||||
collapsed: ref(false)
|
collapsed: ref(false)
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
@ -139,11 +166,14 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
.layout-header {
|
.layout-header {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
display: flex;
|
display: flex;
|
||||||
align-items: center;
|
align-items: center;
|
||||||
height: 48px;
|
height: 48px;
|
||||||
line-height: 48px;
|
line-height: 48px;
|
||||||
background: #fff;
|
background: #fff;
|
||||||
|
box-shadow: 0 1px 4px rgba(0,21,41,.08);
|
||||||
padding: 0 24px;
|
padding: 0 24px;
|
||||||
.layout-header-user {
|
.layout-header-user {
|
||||||
flex: 1
|
flex: 1
|
||||||
@ -151,6 +181,15 @@ export default {
|
|||||||
}
|
}
|
||||||
.layout-content {
|
.layout-content {
|
||||||
position: relative;
|
position: relative;
|
||||||
|
.layout-content-tabs {
|
||||||
|
background: rgb(255, 255, 255);
|
||||||
|
margin: 0px;
|
||||||
|
padding-top: 6px;
|
||||||
|
width: 100%;
|
||||||
|
.ant-tabs-nav-container {
|
||||||
|
padding-left: 16px;
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
.layout-footer {
|
.layout-footer {
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
@ -8,6 +8,8 @@ import { plugin } from './core/plugin';
|
|||||||
import './core/pluginRegister';
|
import './core/pluginRegister';
|
||||||
import { ApplyPluginsType } from '{{{ runtimePath }}}';
|
import { ApplyPluginsType } from '{{{ runtimePath }}}';
|
||||||
import { getRoutes } from './core/routes/routes';
|
import { getRoutes } from './core/routes/routes';
|
||||||
|
import BaseView from './core/routes/baseView';
|
||||||
|
|
||||||
{{{ imports }}}
|
{{{ imports }}}
|
||||||
|
|
||||||
{{{ entryCodeAhead }}}
|
{{{ entryCodeAhead }}}
|
||||||
@ -17,7 +19,9 @@ const renderClient = (opts = {}) => {
|
|||||||
const rootContainer = plugin.applyPlugins({
|
const rootContainer = plugin.applyPlugins({
|
||||||
type: ApplyPluginsType.modify,
|
type: ApplyPluginsType.modify,
|
||||||
key: 'rootContainer',
|
key: 'rootContainer',
|
||||||
initialValue: defineComponent(() => () => (<RouterView></RouterView>)),
|
initialValue: defineComponent((props) => {
|
||||||
|
return () => (<BaseView {...props}></BaseView>)
|
||||||
|
}),
|
||||||
args: {
|
args: {
|
||||||
routes: routes,
|
routes: routes,
|
||||||
plugin: plugin
|
plugin: plugin
|
||||||
|
@ -250,6 +250,8 @@ export default function (api) {
|
|||||||
|
|
||||||
const absRuntimeFilePath = join(namespace, 'runtime.js');
|
const absRuntimeFilePath = join(namespace, 'runtime.js');
|
||||||
|
|
||||||
|
const baseViewFilePath = join(namespace, 'baseView.vue');
|
||||||
|
|
||||||
api.onGenerateFiles(async () => {
|
api.onGenerateFiles(async () => {
|
||||||
const routesTpl = readFileSync(join(__dirname, 'template/routes.tpl'), 'utf-8');
|
const routesTpl = readFileSync(join(__dirname, 'template/routes.tpl'), 'utf-8');
|
||||||
const routes = await api.getRoutesJSON();
|
const routes = await api.getRoutesJSON();
|
||||||
@ -267,6 +269,11 @@ export default function (api) {
|
|||||||
path: absRuntimeFilePath,
|
path: absRuntimeFilePath,
|
||||||
content: readFileSync(join(__dirname, 'template/runtime.tpl'), 'utf-8')
|
content: readFileSync(join(__dirname, 'template/runtime.tpl'), 'utf-8')
|
||||||
});
|
});
|
||||||
|
|
||||||
|
api.writeTmpFile({
|
||||||
|
path: baseViewFilePath,
|
||||||
|
content: readFileSync(join(__dirname, 'template/baseView.vue'), 'utf-8')
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
api.addCoreExports(() => [
|
api.addCoreExports(() => [
|
||||||
|
@ -0,0 +1,22 @@
|
|||||||
|
<template>
|
||||||
|
<template v-if="keepAlive">
|
||||||
|
<router-view v-slot="{ Component }">
|
||||||
|
<keep-alive>
|
||||||
|
<component :is="Component" />
|
||||||
|
</keep-alive>
|
||||||
|
</router-view>
|
||||||
|
</template>
|
||||||
|
<template v-else>
|
||||||
|
<router-view></router-view>
|
||||||
|
</template>
|
||||||
|
</template>
|
||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
props: {
|
||||||
|
keepAlive: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
</script>
|
@ -15,6 +15,7 @@ export default {
|
|||||||
layout: {
|
layout: {
|
||||||
title: "Fes.js",
|
title: "Fes.js",
|
||||||
logo: 'https://gw.alipayobjects.com/zos/rmsportal/KDpgvguMpGfqaHPjicRK.svg',
|
logo: 'https://gw.alipayobjects.com/zos/rmsportal/KDpgvguMpGfqaHPjicRK.svg',
|
||||||
|
multiTabs: true,
|
||||||
menus: [{
|
menus: [{
|
||||||
name: 'index'
|
name: 'index'
|
||||||
}, {
|
}, {
|
||||||
|
@ -3,6 +3,7 @@
|
|||||||
<div>国际化: {{t("test")}}</div>
|
<div>国际化: {{t("test")}}</div>
|
||||||
fes & 拉夫德鲁 <br />
|
fes & 拉夫德鲁 <br />
|
||||||
accessOnepicess: {{accessOnepicess}}
|
accessOnepicess: {{accessOnepicess}}
|
||||||
|
<input />
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
<config>
|
<config>
|
||||||
|
Loading…
x
Reference in New Issue
Block a user