2021-01-10 23:59:43 +08:00
|
|
|
|
<template>
|
2021-08-01 18:30:30 +08:00
|
|
|
|
<div class="layout-columns-aside">
|
2021-03-15 12:44:58 +08:00
|
|
|
|
<el-scrollbar>
|
|
|
|
|
<ul>
|
|
|
|
|
<li
|
|
|
|
|
v-for="(v, k) in columnsAsideList"
|
|
|
|
|
:key="k"
|
|
|
|
|
@click="onColumnsAsideMenuClick(v, k)"
|
|
|
|
|
:ref="
|
|
|
|
|
(el) => {
|
|
|
|
|
if (el) columnsAsideOffsetTopRefs[k] = el;
|
|
|
|
|
}
|
|
|
|
|
"
|
|
|
|
|
:class="{ 'layout-columns-active': liIndex === k }"
|
2021-04-01 11:13:38 +08:00
|
|
|
|
:title="$t(v.meta.title)"
|
2021-03-15 12:44:58 +08:00
|
|
|
|
>
|
2021-05-18 15:42:45 +08:00
|
|
|
|
<div :class="setColumnsAsidelayout" v-if="!v.meta.isLink || (v.meta.isLink && v.meta.isIframe)">
|
2021-03-15 12:44:58 +08:00
|
|
|
|
<i :class="v.meta.icon"></i>
|
2021-05-18 15:42:45 +08:00
|
|
|
|
<div class="columns-vertical-title font12">
|
|
|
|
|
{{
|
|
|
|
|
$t(v.meta.title) && $t(v.meta.title).length >= 4
|
|
|
|
|
? $t(v.meta.title).substr(0, setColumnsAsidelayout === 'columns-vertical' ? 4 : 3)
|
|
|
|
|
: $t(v.meta.title)
|
|
|
|
|
}}
|
2021-03-15 12:44:58 +08:00
|
|
|
|
</div>
|
|
|
|
|
</div>
|
2021-05-18 15:42:45 +08:00
|
|
|
|
<div :class="setColumnsAsidelayout" v-else>
|
2021-03-15 12:44:58 +08:00
|
|
|
|
<a :href="v.meta.isLink" target="_blank">
|
|
|
|
|
<i :class="v.meta.icon"></i>
|
2021-05-18 15:42:45 +08:00
|
|
|
|
<div class="columns-vertical-title font12">
|
|
|
|
|
{{
|
|
|
|
|
$t(v.meta.title) && $t(v.meta.title).length >= 4
|
|
|
|
|
? $t(v.meta.title).substr(0, setColumnsAsidelayout === 'columns-vertical' ? 4 : 3)
|
|
|
|
|
: $t(v.meta.title)
|
|
|
|
|
}}
|
2021-03-15 12:44:58 +08:00
|
|
|
|
</div>
|
|
|
|
|
</a>
|
|
|
|
|
</div>
|
|
|
|
|
</li>
|
|
|
|
|
<div ref="columnsAsideActiveRef" :class="setColumnsAsideStyle"></div>
|
|
|
|
|
</ul>
|
|
|
|
|
</el-scrollbar>
|
|
|
|
|
</div>
|
2021-01-10 23:59:43 +08:00
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script lang="ts">
|
2021-03-15 12:44:58 +08:00
|
|
|
|
import { reactive, toRefs, ref, computed, onMounted, nextTick, getCurrentInstance, watch } from 'vue';
|
|
|
|
|
import { useRoute, useRouter, onBeforeRouteUpdate } from 'vue-router';
|
2021-06-19 17:49:42 +08:00
|
|
|
|
import { useStore } from '/@/store/index';
|
2021-01-10 23:59:43 +08:00
|
|
|
|
export default {
|
2021-03-15 12:44:58 +08:00
|
|
|
|
name: 'layoutColumnsAside',
|
|
|
|
|
setup() {
|
|
|
|
|
const columnsAsideOffsetTopRefs: any = ref([]);
|
|
|
|
|
const columnsAsideActiveRef = ref();
|
|
|
|
|
const { proxy } = getCurrentInstance() as any;
|
|
|
|
|
const store = useStore();
|
|
|
|
|
const route = useRoute();
|
|
|
|
|
const router = useRouter();
|
|
|
|
|
const state: any = reactive({
|
|
|
|
|
columnsAsideList: [],
|
|
|
|
|
liIndex: 0,
|
|
|
|
|
difference: 0,
|
|
|
|
|
routeSplit: [],
|
|
|
|
|
});
|
2021-05-18 15:42:45 +08:00
|
|
|
|
// 设置分栏高亮风格
|
2021-03-15 12:44:58 +08:00
|
|
|
|
const setColumnsAsideStyle = computed(() => {
|
|
|
|
|
return store.state.themeConfig.themeConfig.columnsAsideStyle;
|
|
|
|
|
});
|
2021-05-18 15:42:45 +08:00
|
|
|
|
// 设置分栏布局风格
|
|
|
|
|
const setColumnsAsidelayout = computed(() => {
|
|
|
|
|
return store.state.themeConfig.themeConfig.columnsAsideLayout;
|
|
|
|
|
});
|
2021-03-15 12:44:58 +08:00
|
|
|
|
// 设置菜单高亮位置移动
|
|
|
|
|
const setColumnsAsideMove = (k: number) => {
|
|
|
|
|
state.liIndex = k;
|
|
|
|
|
columnsAsideActiveRef.value.style.top = `${columnsAsideOffsetTopRefs.value[k].offsetTop + state.difference}px`;
|
|
|
|
|
};
|
|
|
|
|
// 菜单高亮点击事件
|
|
|
|
|
const onColumnsAsideMenuClick = (v: Object, k: number) => {
|
|
|
|
|
setColumnsAsideMove(k);
|
|
|
|
|
let { path, redirect } = v as any;
|
|
|
|
|
if (redirect) router.push(redirect);
|
|
|
|
|
else router.push(path);
|
|
|
|
|
};
|
|
|
|
|
// 设置高亮动态位置
|
|
|
|
|
const onColumnsAsideDown = (k: number) => {
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
setColumnsAsideMove(k);
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
// 设置/过滤路由(非静态路由/是否显示在菜单中)
|
|
|
|
|
const setFilterRoutes = () => {
|
|
|
|
|
state.columnsAsideList = filterRoutesFun(store.state.routesList.routesList);
|
|
|
|
|
const resData: any = setSendChildren(route.path);
|
2021-05-31 16:06:53 +08:00
|
|
|
|
if (Object.keys(resData).length <= 0) return false;
|
2021-03-15 12:44:58 +08:00
|
|
|
|
onColumnsAsideDown(resData.item[0].k);
|
|
|
|
|
proxy.mittBus.emit('setSendColumnsChildren', resData);
|
|
|
|
|
};
|
|
|
|
|
// 传送当前子级数据到菜单中
|
|
|
|
|
const setSendChildren = (path: string) => {
|
|
|
|
|
const currentPathSplit = path.split('/');
|
|
|
|
|
let currentData: any = {};
|
|
|
|
|
state.columnsAsideList.map((v: any, k: number) => {
|
|
|
|
|
if (v.path === `/${currentPathSplit[1]}`) {
|
|
|
|
|
v['k'] = k;
|
|
|
|
|
currentData['item'] = [{ ...v }];
|
|
|
|
|
currentData['children'] = [{ ...v }];
|
|
|
|
|
if (v.children) currentData['children'] = v.children;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
return currentData;
|
|
|
|
|
};
|
|
|
|
|
// 路由过滤递归函数
|
|
|
|
|
const filterRoutesFun = (arr: Array<object>) => {
|
|
|
|
|
return arr
|
|
|
|
|
.filter((item: any) => !item.meta.isHide)
|
|
|
|
|
.map((item: any) => {
|
|
|
|
|
item = Object.assign({}, item);
|
|
|
|
|
if (item.children) item.children = filterRoutesFun(item.children);
|
|
|
|
|
return item;
|
|
|
|
|
});
|
|
|
|
|
};
|
|
|
|
|
// tagsView 点击时,根据路由查找下标 columnsAsideList,实现左侧菜单高亮
|
|
|
|
|
const setColumnsMenuHighlight = (path: string) => {
|
|
|
|
|
state.routeSplit = path.split('/');
|
|
|
|
|
state.routeSplit.shift();
|
|
|
|
|
const routeFirst = `/${state.routeSplit[0]}`;
|
|
|
|
|
const currentSplitRoute = state.columnsAsideList.find((v: any) => v.path === routeFirst);
|
2021-05-31 16:06:53 +08:00
|
|
|
|
if (!currentSplitRoute) return false;
|
2021-03-15 12:44:58 +08:00
|
|
|
|
// 延迟拿值,防止取不到
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
onColumnsAsideDown(currentSplitRoute.k);
|
|
|
|
|
}, 0);
|
|
|
|
|
};
|
2021-05-31 21:42:31 +08:00
|
|
|
|
// 监听布局配置信息的变化,动态增加菜单高亮位置移动像素
|
2021-03-15 12:44:58 +08:00
|
|
|
|
watch(store.state, (val) => {
|
|
|
|
|
val.themeConfig.themeConfig.columnsAsideStyle === 'columnsRound' ? (state.difference = 3) : (state.difference = 0);
|
|
|
|
|
});
|
|
|
|
|
// 页面加载时
|
|
|
|
|
onMounted(() => {
|
|
|
|
|
setFilterRoutes();
|
|
|
|
|
});
|
|
|
|
|
// 路由更新时
|
|
|
|
|
onBeforeRouteUpdate((to) => {
|
|
|
|
|
setColumnsMenuHighlight(to.path);
|
|
|
|
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(to.path));
|
|
|
|
|
});
|
|
|
|
|
return {
|
|
|
|
|
columnsAsideOffsetTopRefs,
|
|
|
|
|
columnsAsideActiveRef,
|
|
|
|
|
onColumnsAsideDown,
|
|
|
|
|
setColumnsAsideStyle,
|
2021-05-18 15:42:45 +08:00
|
|
|
|
setColumnsAsidelayout,
|
2021-03-15 12:44:58 +08:00
|
|
|
|
onColumnsAsideMenuClick,
|
|
|
|
|
...toRefs(state),
|
|
|
|
|
};
|
|
|
|
|
},
|
2021-01-10 23:59:43 +08:00
|
|
|
|
};
|
|
|
|
|
</script>
|
|
|
|
|
|
|
|
|
|
<style scoped lang="scss">
|
|
|
|
|
.layout-columns-aside {
|
2021-05-18 15:42:45 +08:00
|
|
|
|
width: 70px;
|
2021-03-15 12:44:58 +08:00
|
|
|
|
height: 100%;
|
|
|
|
|
background: var(--bg-columnsMenuBar);
|
|
|
|
|
ul {
|
|
|
|
|
position: relative;
|
|
|
|
|
li {
|
|
|
|
|
color: var(--bg-columnsMenuBarColor);
|
|
|
|
|
width: 100%;
|
|
|
|
|
height: 50px;
|
|
|
|
|
text-align: center;
|
|
|
|
|
display: flex;
|
|
|
|
|
cursor: pointer;
|
|
|
|
|
position: relative;
|
|
|
|
|
z-index: 1;
|
2021-05-18 15:42:45 +08:00
|
|
|
|
.columns-vertical {
|
2021-03-15 12:44:58 +08:00
|
|
|
|
margin: auto;
|
2021-05-18 15:42:45 +08:00
|
|
|
|
.columns-vertical-title {
|
2021-03-15 12:44:58 +08:00
|
|
|
|
padding-top: 1px;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-05-18 15:42:45 +08:00
|
|
|
|
.columns-horizontal {
|
|
|
|
|
display: flex;
|
|
|
|
|
height: 50px;
|
|
|
|
|
width: 100%;
|
|
|
|
|
align-items: center;
|
|
|
|
|
padding: 0 5px;
|
|
|
|
|
i {
|
|
|
|
|
margin-right: 3px;
|
|
|
|
|
}
|
|
|
|
|
a {
|
|
|
|
|
display: flex;
|
|
|
|
|
.columns-horizontal-title {
|
|
|
|
|
padding-top: 1px;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-03-15 12:44:58 +08:00
|
|
|
|
a {
|
|
|
|
|
text-decoration: none;
|
|
|
|
|
color: var(--bg-columnsMenuBarColor);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
.layout-columns-active {
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
transition: 0.3s ease-in-out;
|
|
|
|
|
}
|
|
|
|
|
.columns-round {
|
|
|
|
|
background: var(--color-primary);
|
|
|
|
|
color: #ffffff;
|
|
|
|
|
position: absolute;
|
|
|
|
|
left: 50%;
|
|
|
|
|
top: 2px;
|
2021-08-01 18:30:30 +08:00
|
|
|
|
height: 44px;
|
|
|
|
|
width: 65px;
|
2021-03-15 12:44:58 +08:00
|
|
|
|
transform: translateX(-50%);
|
|
|
|
|
z-index: 0;
|
|
|
|
|
transition: 0.3s ease-in-out;
|
|
|
|
|
border-radius: 5px;
|
|
|
|
|
}
|
|
|
|
|
.columns-card {
|
|
|
|
|
@extend .columns-round;
|
|
|
|
|
top: 0;
|
2021-08-01 18:30:30 +08:00
|
|
|
|
height: 50px;
|
2021-03-15 12:44:58 +08:00
|
|
|
|
width: 100%;
|
|
|
|
|
border-radius: 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-01-10 23:59:43 +08:00
|
|
|
|
}
|
2021-03-15 12:44:58 +08:00
|
|
|
|
</style>
|