use-mixed-menu.ts 3.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  1. import type { MenuRecordRaw } from '@vben/types';
  2. import { computed, onBeforeMount, ref, watch } from 'vue';
  3. import { useRoute } from 'vue-router';
  4. import { preferences, usePreferences } from '@vben/preferences';
  5. import { useAccessStore } from '@vben/stores';
  6. import { findRootMenuByPath } from '@vben/utils';
  7. import { useNavigation } from './use-navigation';
  8. function useMixedMenu() {
  9. const { navigation } = useNavigation();
  10. const accessStore = useAccessStore();
  11. const route = useRoute();
  12. const splitSideMenus = ref<MenuRecordRaw[]>([]);
  13. const rootMenuPath = ref<string>('');
  14. const { isMixedNav } = usePreferences();
  15. const needSplit = computed(
  16. () => preferences.navigation.split && isMixedNav.value,
  17. );
  18. const sidebarVisible = computed(() => {
  19. const enableSidebar = preferences.sidebar.enable;
  20. if (needSplit.value) {
  21. return enableSidebar && splitSideMenus.value.length > 0;
  22. }
  23. return enableSidebar;
  24. });
  25. const menus = computed(() => accessStore.accessMenus);
  26. /**
  27. * 头部菜单
  28. */
  29. const headerMenus = computed(() => {
  30. if (!needSplit.value) {
  31. return menus.value;
  32. }
  33. return menus.value.map((item) => {
  34. return {
  35. ...item,
  36. children: [],
  37. };
  38. });
  39. });
  40. /**
  41. * 侧边菜单
  42. */
  43. const sidebarMenus = computed(() => {
  44. return needSplit.value ? splitSideMenus.value : menus.value;
  45. });
  46. /**
  47. * 侧边菜单激活路径
  48. */
  49. const sidebarActive = computed(() => {
  50. return (route?.meta?.activePath as string) ?? route.path;
  51. });
  52. /**
  53. * 头部菜单激活路径
  54. */
  55. const headerActive = computed(() => {
  56. if (!needSplit.value) {
  57. return route.path;
  58. }
  59. return rootMenuPath.value;
  60. });
  61. /**
  62. * 菜单点击事件处理
  63. * @param key 菜单路径
  64. * @param mode 菜单模式
  65. */
  66. const handleMenuSelect = (key: string, mode?: string) => {
  67. if (!needSplit.value || mode === 'vertical') {
  68. navigation(key);
  69. return;
  70. }
  71. const rootMenu = menus.value.find((item) => item.path === key);
  72. rootMenuPath.value = rootMenu?.path ?? '';
  73. splitSideMenus.value = rootMenu?.children ?? [];
  74. if (splitSideMenus.value.length === 0) {
  75. navigation(key);
  76. }
  77. };
  78. /**
  79. * 计算侧边菜单
  80. * @param path 路由路径
  81. */
  82. function calcSideMenus(path: string = route.path) {
  83. let { rootMenu } = findRootMenuByPath(menus.value, path);
  84. if (!rootMenu) {
  85. rootMenu = menus.value.find((item) => item.path === path);
  86. }
  87. rootMenuPath.value = rootMenu?.path ?? '';
  88. splitSideMenus.value = rootMenu?.children ?? [];
  89. }
  90. watch(
  91. () => route.path,
  92. (path) => {
  93. const currentPath = (route?.meta?.activePath as string) ?? path;
  94. calcSideMenus(currentPath);
  95. },
  96. { immediate: true },
  97. );
  98. // 初始化计算侧边菜单
  99. onBeforeMount(() => {
  100. calcSideMenus();
  101. });
  102. return {
  103. handleMenuSelect,
  104. headerActive,
  105. headerMenus,
  106. sidebarActive,
  107. sidebarMenus,
  108. sidebarVisible,
  109. };
  110. }
  111. export { useMixedMenu };