159 lines
5.6 KiB
Vue
159 lines
5.6 KiB
Vue
<template>
|
||
<div class="h100" v-show="!isTagsViewCurrenFull">
|
||
<el-aside class="layout-aside" :class="setCollapseStyle">
|
||
<Logo v-if="setShowLogo" />
|
||
<el-scrollbar class="flex-auto" ref="layoutAsideScrollbarRef" @mouseenter="onAsideEnterLeave(true)" @mouseleave="onAsideEnterLeave(false)">
|
||
<Vertical :menuList="menuList" />
|
||
</el-scrollbar>
|
||
</el-aside>
|
||
</div>
|
||
</template>
|
||
|
||
<script lang="ts">
|
||
import { toRefs, reactive, computed, watch, getCurrentInstance, onBeforeMount } from 'vue';
|
||
import { useStore } from '/@/store/index';
|
||
import Logo from '/@/layout/logo/index.vue';
|
||
import Vertical from '/@/layout/navMenu/vertical.vue';
|
||
export default {
|
||
name: 'layoutAside',
|
||
components: { Logo, Vertical },
|
||
setup() {
|
||
const { proxy } = getCurrentInstance() as any;
|
||
const store = useStore();
|
||
const state: any = reactive({
|
||
menuList: [],
|
||
clientWidth: '',
|
||
});
|
||
// 获取布局配置信息
|
||
const getThemeConfig = computed(() => {
|
||
return store.state.themeConfig.themeConfig;
|
||
});
|
||
// 获取卡片全屏信息
|
||
const isTagsViewCurrenFull = computed(() => {
|
||
return store.state.tagsViewRoutes.isTagsViewCurrenFull;
|
||
});
|
||
// 设置菜单展开/收起时的宽度
|
||
const setCollapseStyle = computed(() => {
|
||
const { layout, isCollapse, menuBar } = store.state.themeConfig.themeConfig;
|
||
const asideBrColor =
|
||
menuBar === '#FFFFFF' || menuBar === '#FFF' || menuBar === '#fff' || menuBar === '#ffffff' ? 'layout-el-aside-br-color' : '';
|
||
// 判断是否是手机端
|
||
if (state.clientWidth <= 1000) {
|
||
if (isCollapse) {
|
||
document.body.setAttribute('class', 'el-popup-parent--hidden');
|
||
const asideEle = document.querySelector('.layout-container') as HTMLElement;
|
||
const modeDivs = document.createElement('div');
|
||
modeDivs.setAttribute('class', 'layout-aside-mobile-mode');
|
||
asideEle.appendChild(modeDivs);
|
||
modeDivs.addEventListener('click', closeLayoutAsideMobileMode);
|
||
return [asideBrColor, 'layout-aside-mobile', 'layout-aside-mobile-open'];
|
||
} else {
|
||
// 关闭弹窗
|
||
closeLayoutAsideMobileMode();
|
||
return [asideBrColor, 'layout-aside-mobile', 'layout-aside-mobile-close'];
|
||
}
|
||
} else {
|
||
if (layout === 'columns') {
|
||
// 分栏布局,菜单收起时宽度给 1px
|
||
if (isCollapse) {
|
||
return [asideBrColor, 'layout-aside-pc-1'];
|
||
} else {
|
||
return [asideBrColor, 'layout-aside-pc-220'];
|
||
}
|
||
} else {
|
||
// 其它布局给 64px
|
||
if (isCollapse) {
|
||
return [asideBrColor, 'layout-aside-pc-64'];
|
||
} else {
|
||
return [asideBrColor, 'layout-aside-pc-220'];
|
||
}
|
||
}
|
||
}
|
||
});
|
||
// 关闭移动端蒙版
|
||
const closeLayoutAsideMobileMode = () => {
|
||
const el = document.querySelector('.layout-aside-mobile-mode');
|
||
el && el.parentNode?.removeChild(el);
|
||
store.state.themeConfig.themeConfig.isCollapse = false;
|
||
document.body.setAttribute('class', '');
|
||
};
|
||
// 设置显示/隐藏 logo
|
||
const setShowLogo = computed(() => {
|
||
let { layout, isShowLogo } = store.state.themeConfig.themeConfig;
|
||
return (isShowLogo && layout === 'defaults') || (isShowLogo && layout === 'columns');
|
||
});
|
||
// 设置/过滤路由(非静态路由/是否显示在菜单中)
|
||
const setFilterRoutes = () => {
|
||
if (store.state.themeConfig.themeConfig.layout === 'columns') return false;
|
||
state.menuList = filterRoutesFun(store.state.routesList.routesList);
|
||
};
|
||
// 路由过滤递归函数
|
||
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;
|
||
});
|
||
};
|
||
// 设置菜单导航是否固定(移动端)
|
||
const initMenuFixed = (clientWidth: number) => {
|
||
state.clientWidth = clientWidth;
|
||
};
|
||
// 鼠标移入、移出
|
||
const onAsideEnterLeave = (bool: Boolean) => {
|
||
let { layout } = store.state.themeConfig.themeConfig;
|
||
if (layout !== 'columns') return false;
|
||
if (!bool) proxy.mittBus.emit('restoreDefault');
|
||
store.dispatch('routesList/setColumnsMenuHover', bool);
|
||
};
|
||
// 监听 themeConfig 配置文件的变化,更新菜单 el-scrollbar 的高度
|
||
watch(store.state.themeConfig.themeConfig, (val) => {
|
||
if (val.isShowLogoChange !== val.isShowLogo) {
|
||
if (!proxy.$refs.layoutAsideScrollbarRef) return false;
|
||
proxy.$refs.layoutAsideScrollbarRef.update();
|
||
}
|
||
});
|
||
// 监听vuex值的变化,动态赋值给菜单中
|
||
watch(store.state, (val) => {
|
||
let { layout, isClassicSplitMenu } = val.themeConfig.themeConfig;
|
||
if (layout === 'classic' && isClassicSplitMenu) return false;
|
||
setFilterRoutes();
|
||
});
|
||
// 页面加载前
|
||
onBeforeMount(() => {
|
||
initMenuFixed(document.body.clientWidth);
|
||
setFilterRoutes();
|
||
// 此界面不需要取消监听(proxy.mittBus.off('setSendColumnsChildren))
|
||
// 因为切换布局时有的监听需要使用,取消了监听,某些操作将不生效
|
||
proxy.mittBus.on('setSendColumnsChildren', (res: any) => {
|
||
state.menuList = res.children;
|
||
});
|
||
proxy.mittBus.on('setSendClassicChildren', (res: any) => {
|
||
let { layout, isClassicSplitMenu } = store.state.themeConfig.themeConfig;
|
||
if (layout === 'classic' && isClassicSplitMenu) {
|
||
state.menuList = [];
|
||
state.menuList = res.children;
|
||
}
|
||
});
|
||
proxy.mittBus.on('getBreadcrumbIndexSetFilterRoutes', () => {
|
||
setFilterRoutes();
|
||
});
|
||
proxy.mittBus.on('layoutMobileResize', (res: any) => {
|
||
initMenuFixed(res.clientWidth);
|
||
closeLayoutAsideMobileMode();
|
||
});
|
||
});
|
||
return {
|
||
setCollapseStyle,
|
||
setShowLogo,
|
||
getThemeConfig,
|
||
isTagsViewCurrenFull,
|
||
onAsideEnterLeave,
|
||
...toRefs(state),
|
||
};
|
||
},
|
||
};
|
||
</script>
|