<template>
|
<div class="vue-project-layout">
|
<div class="layout">
|
<div class="layout-header">
|
<div class="logo">
|
<img :src="logo" />
|
</div>
|
<div class="user-dropdown">
|
<Dropdown trigger="click" @on-click="onClick">
|
<a href="javascript:void(0)">
|
<Avatar icon="ios-person" :src="avatar" size="large" />
|
<span class="username">{{ userName }}</span>
|
<Icon type="md-arrow-dropdown" size="18"></Icon>
|
</a>
|
<template #list>
|
<DropdownMenu>
|
<DropdownItem name="profile">个人中心</DropdownItem>
|
<DropdownItem name="logout">退出登录</DropdownItem>
|
</DropdownMenu>
|
</template>
|
</Dropdown>
|
</div>
|
<Skin></Skin>
|
<Locale></Locale>
|
</div>
|
<div class="layout-body">
|
<div class="menu-list">
|
<Menu
|
:active-name="activeName"
|
:open-names="openNames"
|
accordion
|
@on-select="onMenuClick"
|
ref="refMenu"
|
>
|
<template v-for="list in menuList" :key="list.id">
|
<Submenu v-if="list.isFolder" :name="list.id">
|
<template #title>
|
<Icon :custom="list.icon" />
|
{{ list.title }}
|
</template>
|
<MenuItem
|
v-for="app in list.children"
|
:key="app.id"
|
:name="app.id"
|
>
|
<Icon :custom="app.icon" />
|
{{ app.title }}</MenuItem
|
>
|
</Submenu>
|
<MenuItem v-else :name="list.id">
|
<Icon :custom="list.icon" />
|
{{ list.title }}
|
</MenuItem>
|
</template>
|
</Menu>
|
</div>
|
<div class="view">
|
<RouterView v-slot="{ Component }">
|
<template v-if="Component">
|
<KeepAlive>
|
<component :is="Component" :key="$route.fullPath" />
|
</KeepAlive>
|
</template>
|
</RouterView>
|
</div>
|
</div>
|
</div>
|
</div>
|
</template>
|
|
<script>
|
import { ref } from "vue";
|
import Skin from "./skin.vue";
|
import Locale from "./locale.vue";
|
import menuList from "./menu.json";
|
import { showError } from "@/libs/util";
|
export default {
|
name: "Layout",
|
components: {
|
Skin,
|
Locale,
|
},
|
data() {
|
return {
|
menuList,
|
activeName: "",
|
openNames: [],
|
logo: "",
|
avatar: "http://115.29.185.26:5101/userphoto?login=sa",
|
userName: "用户名",
|
};
|
},
|
computed: {
|
viewList() {
|
let routes = [];
|
this.menuList.forEach((menu) => {
|
if (menu.isFolder) routes.push(...menu.children);
|
else routes.push(menu);
|
});
|
return routes;
|
},
|
},
|
methods: {
|
onClick(name) {
|
if (name == "logout") {
|
this.$router.push({ name: "logout" });
|
}
|
},
|
onMenuClick(id) {
|
let route = this.viewList.find((r) => r.id == id);
|
if (!route)
|
this.showError(
|
`未找到id=${id}所对应的路由,请检查菜单、路由的配置是否正确`
|
);
|
|
this.activeName = route.id;
|
this.menuList.forEach((list) => {
|
if (list.isFolder && list.children.find((l) => l.id == route.id))
|
this.openNames = [list.id];
|
else if (list.id == route.id) this.openNames = [];
|
});
|
this.$nextTick(() => {
|
this.refMenu.updateOpened();
|
});
|
|
this.$router.push({
|
name: route.name,
|
query: {
|
menuId: id,
|
},
|
});
|
},
|
showError(ex) {
|
showError(this, ex);
|
},
|
},
|
mounted() {
|
let firstRoute = this.viewList[0];
|
this.onMenuClick(firstRoute.id);
|
},
|
setup() {
|
let refMenu = ref(null);
|
return { refMenu };
|
},
|
};
|
</script>
|
|
<style lang="less">
|
.vue-project-layout {
|
width: 100%;
|
height: 100%;
|
overflow: hidden;
|
.layout {
|
width: 100%;
|
height: 100%;
|
display: flex;
|
overflow: hidden;
|
-webkit-box-orient: vertical;
|
-webkit-box-direction: normal;
|
flex-direction: column;
|
-webkit-box-flex: 1;
|
background: #f5f7f9;
|
.layout-header {
|
width: 100%;
|
padding: 0;
|
height: 64px;
|
line-height: 64px;
|
flex: 0 0 auto;
|
.logo {
|
height: 64px;
|
display: inline-block;
|
width: 240px;
|
padding: 5px 20px;
|
text-align: center;
|
transition: width 0.2sease-out;
|
img {
|
max-width: 200px;
|
max-height: 54px;
|
vertical-align: sub;
|
}
|
}
|
.user-dropdown {
|
float: right;
|
height: 64px;
|
padding-right: 40px;
|
line-height: 64px;
|
.username {
|
display: inline-block;
|
margin-left: 5px;
|
color: #fff;
|
}
|
.ivu-icon {
|
color: #fff;
|
}
|
}
|
}
|
.layout-body {
|
width: 100%;
|
height: calc(100% - 64px);
|
display: flex;
|
flex: auto;
|
.menu-list {
|
flex-shrink: 0;
|
flex-grow: 0;
|
flex-basis: 240px;
|
height: 100%;
|
border-right: 1px solid #dcdee2;
|
overflow-x: hidden;
|
overflow-y: auto;
|
.ivu-menu {
|
height: 100%;
|
}
|
|
.ivu-menu-vertical.ivu-menu-light:after {
|
display: none;
|
}
|
}
|
}
|
}
|
}
|
</style>
|