feat: layout支持side和top两种布局模式,两种模式支持fixedHeader和fixedSideBar

This commit is contained in:
万纯 2021-01-06 19:50:10 +08:00
parent 4eb8257f43
commit a5bfd009d3
11 changed files with 231 additions and 81 deletions

View File

@ -3,7 +3,6 @@
"version": "2.0.0-alpha.0", "version": "2.0.0-alpha.0",
"description": "@webank/fes-compiler", "description": "@webank/fes-compiler",
"main": "lib/index.js", "main": "lib/index.js",
"types": "lib/index.d.ts",
"files": [ "files": [
"lib" "lib"
], ],

View File

@ -6,7 +6,6 @@
"files": [ "files": [
"lib" "lib"
], ],
"module": "dist/index.esm.js",
"scripts": { "scripts": {
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },

View File

@ -9,7 +9,6 @@
"scripts": { "scripts": {
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },
"module": "dist/index.esm.js",
"repository": { "repository": {
"type": "git", "type": "git",
"url": "git+https://github.com/WeBankFinTech/fes.js.git", "url": "git+https://github.com/WeBankFinTech/fes.js.git",

View File

@ -6,7 +6,6 @@
"files": [ "files": [
"lib" "lib"
], ],
"module": "dist/index.esm.js",
"scripts": { "scripts": {
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },

View File

@ -33,7 +33,7 @@ const Layout = defineComponent({
return null; return null;
}, },
}; };
return <BaseLayout {...userConfig} v-slots={slots}></BaseLayout>; return <BaseLayout locale={ localeShared ? true : false } {...userConfig} v-slots={slots}></BaseLayout>;
}; };
} }
}) })

View File

@ -1,49 +1,75 @@
<template> <template>
<a-layout class="main-layout"> <a-layout
<a-layout-sider v-if="routeHasLayout"
v-if="routeHasLayout" :class="[
v-model:collapsed="collapsed" collapsed ? 'main-layout-collapsed' : '',
:width="sideWidth" `main-layout-navigation-${navigation}`
:class="{ collapsed: collapsed }" ]"
collapsible class="main-layout"
theme="dark" >
class="layout-sider" <template v-if="navigation !== 'top'">
> <div v-if="fixedSideBar" class="layout-sider-fixed-stuff"></div>
<div class="logo"> <a-layout-sider
<img :src="logo" class="logo-img" /> v-model:collapsed="collapsed"
<h1 class="logo-name">{{title}}</h1> :width="sideWidth"
</div> :class="[
<Menu :menus="menus" :theme="theme" /> 'layout-sider',
</a-layout-sider> fixedSideBar ? 'layout-sider-fixed' : ''
<a-layout> ]"
<a-layout-header v-if="routeHasLayout" class="layout-header"> collapsible
<div class="layout-header-user"> theme="dark"
>
<div class="layout-logo">
<img :src="logo" class="logo-img" />
<h1 class="logo-name">{{title}}</h1>
</div>
<Menu :menus="menus" :theme="theme" />
</a-layout-sider>
</template>
<a-layout class="child-layout">
<a-layout-header v-if="fixedHeader" class="layout-header">
</a-layout-header>
<a-layout-header
:class="[fixedHeader ? 'layout-header-fixed' : '']"
class="layout-header"
>
<template v-if="navigation === 'top'">
<div class="layout-logo">
<img :src="logo" class="logo-img" />
<h1 class="logo-name">{{title}}</h1>
</div>
<Menu :menus="menus" :theme="theme" class="layout-menu" mode="horizontal" />
</template>
<div class="layout-header-custom">
<slot name="userCenter"></slot> <slot name="userCenter"></slot>
</div> </div>
<slot name="locale"></slot> <template v-if="locale">
<slot name="locale"></slot>
</template>
</a-layout-header> </a-layout-header>
<a-layout-content class="layout-content"> <a-layout-content class="layout-content">
<MultiTabProvider v-if="multiTabs" /> <MultiTabProvider v-if="multiTabs" />
<router-view v-else></router-view> <router-view v-else></router-view>
</a-layout-content> </a-layout-content>
<a-layout-footer v-if="routeHasLayout" class="layout-footer"> <a-layout-footer 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>
</a-layout> </a-layout>
<div v-else class="layout-content">
<MultiTabProvider v-if="multiTabs" />
<router-view v-else></router-view>
</div>
</template> </template>
<script> <script>
import { import { ref, computed } from 'vue';
ref, computed
} from 'vue';
import { useRoute } from '@@/core/coreExports'; import { useRoute } 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 Menu from './Menu'; import Menu from './Menu';
import MultiTabProvider from './MultiTabProvider'; import MultiTabProvider from './MultiTabProvider';
export default { export default {
components: { components: {
[Layout.name]: Layout, [Layout.name]: Layout,
@ -75,7 +101,7 @@ export default {
}, },
theme: { theme: {
type: String, type: String,
default: 'dark' default: 'dark' // lightdark
}, },
navigation: { navigation: {
type: String, type: String,
@ -113,29 +139,86 @@ export default {
}; };
</script> </script>
<style lang="less"> <style lang="less" vars="{ sideWidth: sideWidth +'px' }">
.main-layout { .main-layout {
min-height: 100vh; min-height: 100vh;
.layout-sider{ &.main-layout-collapsed {
&.collapsed{ .layout-sider {
.logo{ .logo {
justify-content: center; justify-content: center;
.logo-name{ .logo-name {
display: none; display: none;
} }
} }
} }
.logo { .layout-sider-fixed-stuff {
overflow: hidden;
width: 80px;
}
}
&.main-layout-navigation-top {
.layout-header {
padding-left: 24px;
color: hsla(0,0%,100%,.65);
background: #001529;
.layout-menu {
line-height: 48px;
}
.layout-logo {
display: flex;
justify-content: flex-start;
align-items: center;
min-width: 165px;
height: 100%;
overflow: hidden;
transition: all .3s;
.logo-img {
height: 32px;
width: auto;
}
.logo-name {
overflow: hidden;
margin: 0 0 0 12px;
color: #fff;
font-weight: 600;
font-size: 18px;
line-height: 32px;
}
}
&.layout-header-fixed {
left: 0;
width: 100%;
}
}
}
.layout-sider-fixed-stuff {
overflow: hidden;
width: var(--sideWidth);
transition: width 0.2s;
flex-shrink: 0;
}
.child-layout {
position: relative;
}
.layout-sider {
&.layout-sider-fixed {
position: fixed;
left: 0;
top: 0;
bottom: 0;
width: 200px;
}
.layout-logo {
height: 32px; height: 32px;
margin: 16px; margin: 16px;
display: flex; display: flex;
justify-content: flex-start; justify-content: flex-start;
align-items: center; align-items: center;
.logo-img{ .logo-img {
height: 32px; height: 32px;
width: auto; width: auto;
} }
.logo-name{ .logo-name {
overflow: hidden; overflow: hidden;
margin: 0 0 0 12px; margin: 0 0 0 12px;
color: #fff; color: #fff;
@ -153,10 +236,18 @@ export default {
height: 48px; height: 48px;
line-height: 48px; line-height: 48px;
background: #fff; background: #fff;
box-shadow: 0 1px 4px rgba(0,21,41,.08); box-shadow: 0 1px 4px rgba(0, 21, 41, 0.08);
padding: 0 24px; padding: 0;
.layout-header-user { .layout-header-custom {
flex: 1 flex: 1;
}
&.layout-header-fixed {
position: fixed;
top: 0;
left: var(--sideWidth);
right: 0;
z-index: 10;
width: calc(100% - var(--sideWidth));
} }
} }
.layout-content { .layout-content {

View File

@ -1,11 +1,38 @@
<template> <template>
<a-tabs :activeKey="route.path" @tabClick="switchTab" class="layout-content-tabs" hide-add type="editable-card"> <a-tabs
<a-tab-pane v-for="page in openedPageList" :key="page.path" closable> :activeKey="route.path"
@tabClick="switchPage"
class="layout-content-tabs"
hide-add
type="editable-card"
>
<a-tab-pane v-for="page in pageList" :key="page.path" closable>
<template #tab> <template #tab>
{{page.name}} {{page.name}}
<ReloadOutlined v-show="route.path === page.path" @click="reloadTab(page.path)" class="layout-tabs-close-icon" /> <ReloadOutlined
v-show="route.path === page.path"
@click="reloadPage(page.path)"
class="layout-tabs-close-icon"
/>
</template> </template>
</a-tab-pane> </a-tab-pane>
<template #tabBarExtraContent>
<a-dropdown>
<div class="layout-tabs-more-icon">
<MoreOutlined />
</div>
<template #overlay>
<a-menu @click="handlerMore">
<a-menu-item key="closeOtherPage">
<a href="javascript:;">关闭其他</a>
</a-menu-item>
<a-menu-item key="reloadPage">
<a href="javascript:;">刷新当前页</a>
</a-menu-item>
</a-menu>
</template>
</a-dropdown>
</template>
</a-tabs> </a-tabs>
<router-view v-slot="{ Component, route }"> <router-view v-slot="{ Component, route }">
<keep-alive> <keep-alive>
@ -14,38 +41,46 @@
</router-view> </router-view>
</template> </template>
<script> <script>
import { import { reactive, unref } from 'vue';
reactive, unref
} from 'vue';
import Tabs from 'ant-design-vue/lib/tabs'; import Tabs from 'ant-design-vue/lib/tabs';
import Dropdown from 'ant-design-vue/lib/dropdown';
import Menu from 'ant-design-vue/lib/menu';
import 'ant-design-vue/lib/menu/style';
import 'ant-design-vue/lib/dropdown/style';
import 'ant-design-vue/lib/tabs/style'; import 'ant-design-vue/lib/tabs/style';
import { ReloadOutlined } from '@ant-design/icons-vue'; import { ReloadOutlined, MoreOutlined } from '@ant-design/icons-vue';
import { useRouter, useRoute } from '@@/core/coreExports'; import { useRouter, useRoute } from '@@/core/coreExports';
let i = 0; let i = 0;
const getKey = () => ++i; const getKey = () => ++i;
export default { export default {
components: { components: {
[Dropdown.name]: Dropdown,
[Menu.name]: Menu,
[Menu.Item.name]: Menu.Item,
[Tabs.name]: Tabs, [Tabs.name]: Tabs,
[Tabs.TabPane.name]: Tabs.TabPane, [Tabs.TabPane.name]: Tabs.TabPane,
ReloadOutlined ReloadOutlined,
MoreOutlined
}, },
setup() { setup() {
const route = useRoute(); const route = useRoute();
const router = useRouter(); const router = useRouter();
const openedPageList = reactive([{ const pageList = reactive([
path: unref(route.path), {
route: { path: unref(route.path),
query: unref(route.query), route: {
params: unref(route.params) query: unref(route.query),
}, params: unref(route.params)
name: unref(route.meta).name, },
key: getKey() name: unref(route.meta).name,
}]); key: getKey()
const findPage = path => openedPageList.find(item => unref(item.path) === path); }
]);
const findPage = path => pageList.find(item => unref(item.path) === unref(path));
router.beforeEach((to) => { router.beforeEach((to) => {
if (!findPage(to.path)) { if (!findPage(to.path)) {
openedPageList.push({ pageList.push({
path: to.path, path: to.path,
route: to, route: to,
name: to.meta.name, name: to.meta.name,
@ -55,7 +90,7 @@ export default {
return true; return true;
}); });
// //
const switchTab = (path) => { const switchPage = (path) => {
const selectedPage = findPage(path); const selectedPage = findPage(path);
if (selectedPage) { if (selectedPage) {
router.push({ router.push({
@ -65,12 +100,17 @@ export default {
}); });
} }
}; };
const reloadTab = (path) => { const reloadPage = (path) => {
const selectedPage = findPage(path); const selectedPage = findPage(path || unref(route.path));
if (selectedPage) { if (selectedPage) {
selectedPage.key = getKey(); selectedPage.key = getKey();
} }
}; };
const closeOtherPage = (path) => {
const selectedPage = findPage(path || unref(route.path));
pageList.length = 0;
pageList.push(selectedPage);
};
const getPageKey = (_route) => { const getPageKey = (_route) => {
const selectedPage = findPage(_route.path); const selectedPage = findPage(_route.path);
if (selectedPage) { if (selectedPage) {
@ -78,12 +118,25 @@ export default {
} }
return ''; return '';
}; };
const handlerMore = ({ key }) => {
console.log(key);
switch (key) {
case 'closeOtherPage':
closeOtherPage();
break;
case 'reloadPage':
reloadPage();
break;
default:
}
};
return { return {
route, route,
openedPageList, pageList,
getPageKey, getPageKey,
reloadTab, reloadPage,
switchTab switchPage,
handlerMore
}; };
} }
}; };
@ -96,15 +149,23 @@ export default {
width: 100%; width: 100%;
.ant-tabs-nav-container { .ant-tabs-nav-container {
padding-left: 16px; padding-left: 16px;
.layout-tabs-close-icon { }
vertical-align: middle; .layout-tabs-close-icon {
color: rgba(0, 0, 0, 0.45); vertical-align: middle;
font-size: 12px; color: rgba(0, 0, 0, 0.45);
margin-left: 6px; font-size: 12px;
margin-top: -2px; margin-left: 6px;
&:hover{ margin-top: -2px;
color: rgba(0, 0, 0, 0.8); &:hover {
} color: rgba(0, 0, 0, 0.8);
}
}
.layout-tabs-more-icon {
margin-right: 8px;
padding: 0 4px;
color: rgba(0, 0, 0, 0.45);
&:hover {
color: rgba(0, 0, 0, 0.8);
} }
} }
} }

View File

@ -6,7 +6,6 @@
"files": [ "files": [
"lib" "lib"
], ],
"module": "dist/index.esm.js",
"scripts": { "scripts": {
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },

View File

@ -1,6 +1,6 @@
<template> <template>
<a-dropdown> <a-dropdown>
<GlobalOutlined /> <div class="lang-icon"><GlobalOutlined /></div>
<template #overlay> <template #overlay>
<a-menu :selectedKeys="selectedKeys" @click="handleClick"> <a-menu :selectedKeys="selectedKeys" @click="handleClick">
<a-menu-item <a-menu-item
@ -53,6 +53,11 @@ export default {
</script> </script>
<style lang="less"> <style lang="less">
.lang-icon {
margin: 0 8px;
padding: 0 4px;
cursor: pointer;
}
.lang-item { .lang-item {
display: flex; display: flex;
align-items: center; align-items: center;

View File

@ -6,7 +6,6 @@
"files": [ "files": [
"lib" "lib"
], ],
"module": "dist/index.esm.js",
"scripts": { "scripts": {
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },

View File

@ -9,7 +9,6 @@
"scripts": { "scripts": {
"test": "echo \"Error: no test specified\" && exit 1" "test": "echo \"Error: no test specified\" && exit 1"
}, },
"module": "dist/index.esm.js",
"repository": { "repository": {
"type": "git", "type": "git",
"url": "git+https://github.com/WeBankFinTech/fes.js.git", "url": "git+https://github.com/WeBankFinTech/fes.js.git",