aside.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. <template>
  2. <div class="h100" >
  3. <!-- <div class="h100" v-show="!isTagsViewCurrenFull">-->
  4. <el-aside class="layout-aside" :class="setCollapseStyle">
  5. <Logo />
  6. <!-- <Logo v-if="setShowLogo" />-->
  7. <el-scrollbar class="flex-auto" ref="layoutAsideScrollbarRef" @mouseenter="onAsideEnterLeave(true)" @mouseleave="onAsideEnterLeave(false)">
  8. <Vertical :menuList="state.menuList" />
  9. </el-scrollbar>
  10. </el-aside>
  11. </div>
  12. </template>
  13. <script setup lang="ts" name="layoutAside">
  14. import { defineAsyncComponent, reactive, computed, watch, onBeforeMount, ref } from 'vue';
  15. import { storeToRefs } from 'pinia';
  16. import pinia from '/@/stores';
  17. import { useRoutesList } from '/@/stores/routesList';
  18. import { useThemeConfig } from '/@/stores/themeConfig';
  19. import { useTagsViewRoutes } from '/@/stores/tagsViewRoutes';
  20. import mittBus from '/@/utils/mitt';
  21. // 引入组件
  22. const Logo = defineAsyncComponent(() => import('/@/layout/logo/index.vue'));
  23. const Vertical = defineAsyncComponent(() => import('/@/layout/navMenu/vertical.vue'));
  24. // 定义变量内容
  25. const layoutAsideScrollbarRef = ref();
  26. const stores = useRoutesList();
  27. const storesThemeConfig = useThemeConfig();
  28. const storesTagsViewRoutes = useTagsViewRoutes();
  29. const { routesList } = storeToRefs(stores);
  30. const { themeConfig } = storeToRefs(storesThemeConfig);
  31. const { isTagsViewCurrenFull } = storeToRefs(storesTagsViewRoutes);
  32. const state = reactive<AsideState>({
  33. menuList: [],
  34. clientWidth: 0,
  35. });
  36. // 设置菜单展开/收起时的宽度
  37. const setCollapseStyle = computed(() => {
  38. const { layout, isCollapse, menuBar } = themeConfig.value;
  39. const asideBrTheme = ['#FFFFFF', '#FFF', '#fff', '#ffffff'];
  40. const asideBrColor = asideBrTheme.includes(menuBar) ? 'layout-el-aside-br-color' : '';
  41. // 判断是否是手机端
  42. if (state.clientWidth <= 1000) {
  43. if (isCollapse) {
  44. document.body.setAttribute('class', 'el-popup-parent--hidden');
  45. const asideEle = document.querySelector('.layout-container') as HTMLElement;
  46. const modeDivs = document.createElement('div');
  47. modeDivs.setAttribute('class', 'layout-aside-mobile-mode');
  48. asideEle.appendChild(modeDivs);
  49. modeDivs.addEventListener('click', closeLayoutAsideMobileMode);
  50. return [asideBrColor, 'layout-aside-mobile', 'layout-aside-mobile-open'];
  51. } else {
  52. // 关闭弹窗
  53. closeLayoutAsideMobileMode();
  54. return [asideBrColor, 'layout-aside-mobile', 'layout-aside-mobile-close'];
  55. }
  56. } else {
  57. if (layout === 'columns') {
  58. // 分栏布局,菜单收起时宽度给 1px
  59. if (isCollapse) return [asideBrColor, 'layout-aside-pc-1'];
  60. else return [asideBrColor, 'layout-aside-pc-220'];
  61. } else {
  62. // 其它布局给 64px
  63. if (isCollapse) return [asideBrColor, 'layout-aside-pc-64'];
  64. else return [asideBrColor, 'layout-aside-pc-220'];
  65. }
  66. }
  67. });
  68. // 设置显示/隐藏 logo
  69. const setShowLogo = computed(() => {
  70. let { layout, isShowLogo } = themeConfig.value;
  71. return (isShowLogo && layout === 'defaults') || (isShowLogo && layout === 'columns');
  72. });
  73. // 关闭移动端蒙版
  74. const closeLayoutAsideMobileMode = () => {
  75. const el = document.querySelector('.layout-aside-mobile-mode');
  76. el?.setAttribute('style', 'animation: error-img-two 0.3s');
  77. setTimeout(() => {
  78. el?.parentNode?.removeChild(el);
  79. }, 300);
  80. const clientWidth = document.body.clientWidth;
  81. if (clientWidth < 1000) themeConfig.value.isCollapse = false;
  82. document.body.setAttribute('class', '');
  83. };
  84. // 设置/过滤路由(非静态路由/是否显示在菜单中)
  85. const setFilterRoutes = () => {
  86. if (themeConfig.value.layout === 'columns') return false;
  87. state.menuList = filterRoutesFun(routesList.value);
  88. };
  89. // 路由过滤递归函数
  90. const filterRoutesFun = <T extends RouteItem>(arr: T[]): T[] => {
  91. return arr
  92. .filter((item: T) => !item.meta?.isHide)
  93. .map((item: T) => {
  94. item = Object.assign({}, item);
  95. if (item.children) item.children = filterRoutesFun(item.children);
  96. return item;
  97. });
  98. };
  99. // 设置菜单导航是否固定(移动端)
  100. const initMenuFixed = (clientWidth: number) => {
  101. state.clientWidth = clientWidth;
  102. };
  103. // 鼠标移入、移出
  104. const onAsideEnterLeave = (bool: Boolean) => {
  105. let { layout } = themeConfig.value;
  106. if (layout !== 'columns') return false;
  107. if (!bool) mittBus.emit('restoreDefault');
  108. // 开启 `分栏菜单鼠标悬停预加载` 才设置,防止 columnsAside.vue 监听 pinia.state
  109. if (themeConfig.value.isColumnsMenuHoverPreload) stores.setColumnsMenuHover(bool);
  110. };
  111. // 页面加载前
  112. onBeforeMount(() => {
  113. initMenuFixed(document.body.clientWidth);
  114. setFilterRoutes();
  115. // 此界面不需要取消监听(mittBus.off('setSendColumnsChildren))
  116. // 因为切换布局时有的监听需要使用,取消了监听,某些操作将不生效
  117. mittBus.on('setSendColumnsChildren', (res: MittMenu) => {
  118. state.menuList = res.children;
  119. });
  120. // 开启经典布局分割菜单时,设置菜单数据
  121. mittBus.on('setSendClassicChildren', (res: MittMenu) => {
  122. let { layout, isClassicSplitMenu } = themeConfig.value;
  123. if (layout === 'classic' && isClassicSplitMenu) {
  124. state.menuList = [];
  125. state.menuList = res.children;
  126. }
  127. });
  128. // 开启经典布局分割菜单时,重新处理菜单数据
  129. mittBus.on('getBreadcrumbIndexSetFilterRoutes', () => {
  130. setFilterRoutes();
  131. });
  132. // 监听窗口大小改变时(适配移动端)
  133. mittBus.on('layoutMobileResize', (res: LayoutMobileResize) => {
  134. initMenuFixed(res.clientWidth);
  135. closeLayoutAsideMobileMode();
  136. });
  137. });
  138. // 监听 themeConfig 配置文件的变化,更新菜单 el-scrollbar 的高度
  139. watch(themeConfig.value, (val) => {
  140. if (val.isShowLogoChange !== val.isShowLogo) {
  141. if (layoutAsideScrollbarRef.value) layoutAsideScrollbarRef.value.update();
  142. }
  143. });
  144. // 监听 pinia 值的变化,动态赋值给菜单中
  145. watch(
  146. pinia.state,
  147. (val) => {
  148. let { layout, isClassicSplitMenu } = val.themeConfig.themeConfig;
  149. if (layout === 'classic' && isClassicSplitMenu) return false;
  150. setFilterRoutes();
  151. },
  152. {
  153. deep: true,
  154. }
  155. );
  156. </script>