You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
298 lines
8.8 KiB
298 lines
8.8 KiB
3 years ago
|
<template>
|
||
|
<div class="layout-columns-aside">
|
||
|
<el-scrollbar>
|
||
|
<ul @mouseleave="onColumnsAsideMenuMouseleave()">
|
||
|
<li
|
||
|
v-for="(v, k) in columnsAsideList"
|
||
|
:key="k"
|
||
|
@click="onColumnsAsideMenuClick(v, k)"
|
||
|
@mouseenter="onColumnsAsideMenuMouseenter(v, k)"
|
||
|
:ref="
|
||
|
(el) => {
|
||
|
if (el) columnsAsideOffsetTopRefs[k] = el;
|
||
|
}
|
||
|
"
|
||
|
:class="{ 'layout-columns-active': liIndex === k, 'layout-columns-hover': liHoverIndex === k }"
|
||
|
:title="v.meta.title.indexOf('.')>0?$t(v.meta.title):v.meta.title"
|
||
|
>
|
||
|
<div :class="setColumnsAsidelayout" v-if="!v.meta.isLink || (v.meta.isLink && v.meta.isIframe)">
|
||
|
<SvgIcon :name="v.meta.icon" />
|
||
|
<div class="columns-vertical-title font12">
|
||
|
{{tMenuTitle(v.meta.title)}}
|
||
|
</div>
|
||
|
</div>
|
||
|
<div :class="setColumnsAsidelayout" v-else>
|
||
|
<a :href="v.meta.isLink" target="_blank">
|
||
|
<SvgIcon :name="v.meta.icon" />
|
||
|
<div class="columns-vertical-title font12">
|
||
|
{{tMenuTitle(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, onUnmounted, defineComponent } from 'vue';
|
||
|
import { useRoute, useRouter, onBeforeRouteUpdate, RouteRecordRaw } from 'vue-router';
|
||
|
import { useStore } from '/@/store/index';
|
||
|
import {useI18n} from "vue-i18n";
|
||
|
|
||
|
// 定义接口来定义对象的类型
|
||
|
interface ColumnsAsideState {
|
||
|
columnsAsideList: any[];
|
||
|
liIndex: number;
|
||
|
liOldIndex: null | number;
|
||
|
liHoverIndex: null | number;
|
||
|
liOldPath: null | string;
|
||
|
difference: number;
|
||
|
routeSplit: string[];
|
||
|
isNavHover: boolean;
|
||
|
}
|
||
|
|
||
|
export default defineComponent({
|
||
|
name: 'layoutColumnsAside',
|
||
|
setup() {
|
||
|
const columnsAsideOffsetTopRefs: any = ref([]);
|
||
|
const columnsAsideActiveRef = ref();
|
||
|
const { t } = useI18n();
|
||
|
const { proxy } = <any>getCurrentInstance();
|
||
|
const store = useStore();
|
||
|
const route = useRoute();
|
||
|
const router = useRouter();
|
||
|
const state = reactive<ColumnsAsideState>({
|
||
|
columnsAsideList: [],
|
||
|
liIndex: 0,
|
||
|
liOldIndex: null,
|
||
|
liHoverIndex: null,
|
||
|
liOldPath: null,
|
||
|
difference: 0,
|
||
|
routeSplit: [],
|
||
|
isNavHover: false,
|
||
|
});
|
||
|
// 设置菜单名称
|
||
|
const tMenuTitle = (title:string):string=>{
|
||
|
let rTitle = title.indexOf('.')>0?t(title):title;
|
||
|
rTitle && rTitle.length >= 4
|
||
|
? rTitle.substring(0, store.state.themeConfig.themeConfig.columnsAsideLayout === 'columns-vertical' ? 4 : 3)
|
||
|
: rTitle
|
||
|
return rTitle
|
||
|
};
|
||
|
// 设置分栏高亮风格
|
||
|
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 onColumnsAsideMenuMouseenter = (v: RouteRecordRaw, k: number) => {
|
||
|
let { path } = v;
|
||
|
state.liOldPath = path;
|
||
|
state.liOldIndex = k;
|
||
|
state.liHoverIndex = k;
|
||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(path));
|
||
|
store.dispatch('routesList/setColumnsMenuHover', false);
|
||
|
store.dispatch('routesList/setColumnsNavHover', true);
|
||
|
state.isNavHover = true;
|
||
|
};
|
||
|
// 鼠标移走时,显示原来的子级菜单
|
||
|
const onColumnsAsideMenuMouseleave = async () => {
|
||
|
await store.dispatch('routesList/setColumnsNavHover', false);
|
||
|
// 添加延时器,防止拿到的 store.state.routesList 值不是最新的
|
||
|
setTimeout(() => {
|
||
|
const { isColumnsMenuHover, isColumnsNavHover } = store.state.routesList;
|
||
|
if (!isColumnsMenuHover && !isColumnsNavHover) proxy.mittBus.emit('restoreDefault');
|
||
|
}, 100);
|
||
|
// state.isNavHover = false;
|
||
|
};
|
||
|
// 设置高亮动态位置
|
||
|
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((<any>currentSplitRoute).k);
|
||
|
}, 0);
|
||
|
};
|
||
|
// 监听布局配置信息的变化,动态增加菜单高亮位置移动像素
|
||
|
watch(store.state, (val) => {
|
||
|
val.themeConfig.themeConfig.columnsAsideStyle === 'columnsRound' ? (state.difference = 3) : (state.difference = 0);
|
||
|
if (!val.routesList.isColumnsMenuHover && !val.routesList.isColumnsNavHover) {
|
||
|
state.liHoverIndex = null;
|
||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(route.path));
|
||
|
} else {
|
||
|
state.liHoverIndex = state.liOldIndex;
|
||
|
if (!state.liOldPath) return false;
|
||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(state.liOldPath));
|
||
|
}
|
||
|
});
|
||
|
// 页面加载时
|
||
|
onMounted(() => {
|
||
|
setFilterRoutes();
|
||
|
// 销毁变量,防止鼠标再次移入时,保留了上次的记录
|
||
|
proxy.mittBus.on('restoreDefault', () => {
|
||
|
state.liOldIndex = null;
|
||
|
state.liOldPath = null;
|
||
|
});
|
||
|
});
|
||
|
// 页面卸载时
|
||
|
onUnmounted(() => {
|
||
|
proxy.mittBus.off('restoreDefault', () => {});
|
||
|
});
|
||
|
// 路由更新时
|
||
|
onBeforeRouteUpdate((to) => {
|
||
|
setColumnsMenuHighlight(to.path);
|
||
|
proxy.mittBus.emit('setSendColumnsChildren', setSendChildren(to.path));
|
||
|
});
|
||
|
return {
|
||
|
columnsAsideOffsetTopRefs,
|
||
|
columnsAsideActiveRef,
|
||
|
onColumnsAsideDown,
|
||
|
setColumnsAsideStyle,
|
||
|
setColumnsAsidelayout,
|
||
|
onColumnsAsideMenuClick,
|
||
|
onColumnsAsideMenuMouseenter,
|
||
|
onColumnsAsideMenuMouseleave,
|
||
|
tMenuTitle,
|
||
|
...toRefs(state),
|
||
|
};
|
||
|
},
|
||
|
});
|
||
|
</script>
|
||
|
|
||
|
<style scoped lang="scss">
|
||
|
.layout-columns-aside {
|
||
|
width: 70px;
|
||
|
height: 100%;
|
||
|
background: var(--next-bg-columnsMenuBar);
|
||
|
ul {
|
||
|
position: relative;
|
||
|
li {
|
||
|
color: var(--next-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(--next-bg-columnsMenuBarColor);
|
||
|
}
|
||
|
}
|
||
|
.layout-columns-active {
|
||
|
color: var(--el-color-white);
|
||
|
transition: 0.3s ease-in-out;
|
||
|
}
|
||
|
.layout-columns-hover {
|
||
|
color: var(--el-color-primary);
|
||
|
a {
|
||
|
color: var(--el-color-primary);
|
||
|
}
|
||
|
}
|
||
|
.columns-round {
|
||
|
background: var(--el-color-primary);
|
||
|
color: var(--el-color-white);
|
||
|
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>
|