231 lines
6.3 KiB
Vue
231 lines
6.3 KiB
Vue
<template>
|
||
<div class="layout-columns-aside">
|
||
<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 }"
|
||
:title="$t(v.meta.title)"
|
||
>
|
||
<div :class="setColumnsAsidelayout" v-if="!v.meta.isLink || (v.meta.isLink && v.meta.isIframe)">
|
||
<i :class="v.meta.icon"></i>
|
||
<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)
|
||
}}
|
||
</div>
|
||
</div>
|
||
<div :class="setColumnsAsidelayout" v-else>
|
||
<a :href="v.meta.isLink" target="_blank">
|
||
<i :class="v.meta.icon"></i>
|
||
<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)
|
||
}}
|
||
</div>
|
||
</a>
|
||
</div>
|
||
</li>
|
||
<div ref="columnsAsideActiveRef" :class="setColumnsAsideStyle"></div>
|
||
</ul>
|
||
</el-scrollbar>
|
||
</div>
|
||
</template>
|
||
|
||
<script lang="ts">
|
||
import { reactive, toRefs, ref, computed, onMounted, nextTick, getCurrentInstance, watch } from 'vue';
|
||
import { useRoute, useRouter, onBeforeRouteUpdate } from 'vue-router';
|
||
import { useStore } from '/@/store/index';
|
||
export default {
|
||
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: [],
|
||
});
|
||
// 设置分栏高亮风格
|
||
const setColumnsAsideStyle = computed(() => {
|
||
return store.state.themeConfig.themeConfig.columnsAsideStyle;
|
||
});
|
||
// 设置分栏布局风格
|
||
const setColumnsAsidelayout = computed(() => {
|
||
return store.state.themeConfig.themeConfig.columnsAsideLayout;
|
||
});
|
||
// 设置菜单高亮位置移动
|
||
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);
|
||
if (Object.keys(resData).length <= 0) return false;
|
||
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);
|
||
if (!currentSplitRoute) return false;
|
||
// 延迟拿值,防止取不到
|
||
setTimeout(() => {
|
||
onColumnsAsideDown(currentSplitRoute.k);
|
||
}, 0);
|
||
};
|
||
// 监听布局配置信息的变化,动态增加菜单高亮位置移动像素
|
||
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,
|
||
setColumnsAsidelayout,
|
||
onColumnsAsideMenuClick,
|
||
...toRefs(state),
|
||
};
|
||
},
|
||
};
|
||
</script>
|
||
|
||
<style scoped lang="scss">
|
||
.layout-columns-aside {
|
||
width: 70px;
|
||
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;
|
||
.columns-vertical {
|
||
margin: auto;
|
||
.columns-vertical-title {
|
||
padding-top: 1px;
|
||
}
|
||
}
|
||
.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;
|
||
}
|
||
}
|
||
}
|
||
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;
|
||
height: 44px;
|
||
width: 65px;
|
||
transform: translateX(-50%);
|
||
z-index: 0;
|
||
transition: 0.3s ease-in-out;
|
||
border-radius: 5px;
|
||
}
|
||
.columns-card {
|
||
@extend .columns-round;
|
||
top: 0;
|
||
height: 50px;
|
||
width: 100%;
|
||
border-radius: 0;
|
||
}
|
||
}
|
||
}
|
||
</style>
|