feat: layout支持multiTabs参数

This commit is contained in:
万纯 2021-01-05 16:18:39 +08:00
parent 1746696472
commit 6eb4bdd050
7 changed files with 83 additions and 9 deletions

View File

@ -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>)

View File

@ -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;

View File

@ -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

View File

@ -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(() => [

View File

@ -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>

View File

@ -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'
}, { }, {

View File

@ -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>