mirror of
https://github.com/WeBankFinTech/fes.js.git
synced 2025-04-06 03:59:53 +08:00
feat: layout支持side和top两种布局模式,两种模式支持fixedHeader和fixedSideBar
This commit is contained in:
parent
4eb8257f43
commit
a5bfd009d3
@ -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"
|
||||||
],
|
],
|
||||||
|
@ -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"
|
||||||
},
|
},
|
||||||
|
@ -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",
|
||||||
|
@ -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"
|
||||||
},
|
},
|
||||||
|
@ -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>;
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
@ -1,49 +1,75 @@
|
|||||||
<template>
|
<template>
|
||||||
<a-layout class="main-layout">
|
<a-layout
|
||||||
<a-layout-sider
|
|
||||||
v-if="routeHasLayout"
|
v-if="routeHasLayout"
|
||||||
|
:class="[
|
||||||
|
collapsed ? 'main-layout-collapsed' : '',
|
||||||
|
`main-layout-navigation-${navigation}`
|
||||||
|
]"
|
||||||
|
class="main-layout"
|
||||||
|
>
|
||||||
|
<template v-if="navigation !== 'top'">
|
||||||
|
<div v-if="fixedSideBar" class="layout-sider-fixed-stuff"></div>
|
||||||
|
<a-layout-sider
|
||||||
v-model:collapsed="collapsed"
|
v-model:collapsed="collapsed"
|
||||||
:width="sideWidth"
|
:width="sideWidth"
|
||||||
:class="{ collapsed: collapsed }"
|
:class="[
|
||||||
|
'layout-sider',
|
||||||
|
fixedSideBar ? 'layout-sider-fixed' : ''
|
||||||
|
]"
|
||||||
collapsible
|
collapsible
|
||||||
theme="dark"
|
theme="dark"
|
||||||
class="layout-sider"
|
|
||||||
>
|
>
|
||||||
<div class="logo">
|
<div class="layout-logo">
|
||||||
<img :src="logo" class="logo-img" />
|
<img :src="logo" class="logo-img" />
|
||||||
<h1 class="logo-name">{{title}}</h1>
|
<h1 class="logo-name">{{title}}</h1>
|
||||||
</div>
|
</div>
|
||||||
<Menu :menus="menus" :theme="theme" />
|
<Menu :menus="menus" :theme="theme" />
|
||||||
</a-layout-sider>
|
</a-layout-sider>
|
||||||
<a-layout>
|
</template>
|
||||||
<a-layout-header v-if="routeHasLayout" class="layout-header">
|
<a-layout class="child-layout">
|
||||||
<div class="layout-header-user">
|
<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>
|
||||||
|
<template v-if="locale">
|
||||||
<slot name="locale"></slot>
|
<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' // light、dark
|
||||||
},
|
},
|
||||||
navigation: {
|
navigation: {
|
||||||
type: String,
|
type: String,
|
||||||
@ -113,11 +139,11 @@ export default {
|
|||||||
};
|
};
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="less">
|
<style lang="less" vars="{ sideWidth: sideWidth +'px' }">
|
||||||
.main-layout {
|
.main-layout {
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
|
&.main-layout-collapsed {
|
||||||
.layout-sider {
|
.layout-sider {
|
||||||
&.collapsed{
|
|
||||||
.logo {
|
.logo {
|
||||||
justify-content: center;
|
justify-content: center;
|
||||||
.logo-name {
|
.logo-name {
|
||||||
@ -125,7 +151,64 @@ export default {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.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;
|
||||||
@ -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 {
|
||||||
|
@ -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,26 +41,33 @@
|
|||||||
</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),
|
path: unref(route.path),
|
||||||
route: {
|
route: {
|
||||||
query: unref(route.query),
|
query: unref(route.query),
|
||||||
@ -41,11 +75,12 @@ export default {
|
|||||||
},
|
},
|
||||||
name: unref(route.meta).name,
|
name: unref(route.meta).name,
|
||||||
key: getKey()
|
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,6 +149,7 @@ export default {
|
|||||||
width: 100%;
|
width: 100%;
|
||||||
.ant-tabs-nav-container {
|
.ant-tabs-nav-container {
|
||||||
padding-left: 16px;
|
padding-left: 16px;
|
||||||
|
}
|
||||||
.layout-tabs-close-icon {
|
.layout-tabs-close-icon {
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
color: rgba(0, 0, 0, 0.45);
|
color: rgba(0, 0, 0, 0.45);
|
||||||
@ -106,6 +160,13 @@ export default {
|
|||||||
color: rgba(0, 0, 0, 0.8);
|
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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
|
@ -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"
|
||||||
},
|
},
|
||||||
|
@ -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;
|
||||||
|
@ -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"
|
||||||
},
|
},
|
||||||
|
@ -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",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user