mixNav.vue 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180
  1. <script setup lang="ts">
  2. import Search from "../search/index.vue";
  3. import Notice from "../notice/index.vue";
  4. import avatars from "/@/assets/avatars.jpg";
  5. import { useNav } from "/@/layout/hooks/useNav";
  6. import { transformI18n } from "/@/plugins/i18n";
  7. import { ref, toRaw, watch, onMounted } from "vue";
  8. import { useRenderIcon } from "/@/components/ReIcon/src/hooks";
  9. import { getParentPaths, findRouteByPath } from "/@/router/utils";
  10. import { useTranslationLang } from "../../hooks/useTranslationLang";
  11. import { usePermissionStoreHook } from "/@/store/modules/permission";
  12. import globalization from "/@/assets/svg/globalization.svg?component";
  13. const menuRef = ref();
  14. let defaultActive = ref(null);
  15. const { t, route, locale, translationCh, translationEn } =
  16. useTranslationLang(menuRef);
  17. const {
  18. device,
  19. routers,
  20. logout,
  21. onPanel,
  22. menuSelect,
  23. resolvePath,
  24. username,
  25. avatarsStyle,
  26. getDropdownItemStyle,
  27. getDropdownItemClass
  28. } = useNav();
  29. function getDefaultActive(routePath) {
  30. const wholeMenus = usePermissionStoreHook().wholeMenus;
  31. /** 当前路由的父级路径 */
  32. const parentRoutes = getParentPaths(routePath, wholeMenus)[0];
  33. defaultActive.value = findRouteByPath(
  34. parentRoutes,
  35. wholeMenus
  36. )?.children[0]?.path;
  37. }
  38. onMounted(() => {
  39. getDefaultActive(route.path);
  40. });
  41. watch(
  42. () => route.path,
  43. () => {
  44. getDefaultActive(route.path);
  45. }
  46. );
  47. </script>
  48. <template>
  49. <div v-if="device !== 'mobile'" class="horizontal-header">
  50. <el-menu
  51. router
  52. ref="menuRef"
  53. mode="horizontal"
  54. class="horizontal-header-menu"
  55. :default-active="defaultActive"
  56. @select="indexPath => menuSelect(indexPath, routers)"
  57. >
  58. <el-menu-item
  59. v-for="route in usePermissionStoreHook().wholeMenus"
  60. :key="route.path"
  61. :index="resolvePath(route) || route.redirect"
  62. >
  63. <template #title>
  64. <div
  65. v-if="toRaw(route.meta.icon)"
  66. :class="['sub-menu-icon', route.meta.icon]"
  67. >
  68. <component
  69. :is="useRenderIcon(route.meta && toRaw(route.meta.icon))"
  70. />
  71. </div>
  72. <span class="select-none">{{ transformI18n(route.meta.title) }}</span>
  73. <FontIcon
  74. v-if="route.meta.extraIcon"
  75. width="30px"
  76. height="30px"
  77. style="position: absolute; right: 10px"
  78. :icon="route.meta.extraIcon.name"
  79. :svg="route.meta.extraIcon.svg ? true : false"
  80. />
  81. </template>
  82. </el-menu-item>
  83. </el-menu>
  84. <div class="horizontal-header-right">
  85. <!-- 菜单搜索 -->
  86. <Search />
  87. <!-- 通知 -->
  88. <Notice id="header-notice" />
  89. <!-- 国际化 -->
  90. <el-dropdown id="header-translation" trigger="click">
  91. <globalization
  92. class="navbar-bg-hover w-[40px] h-[48px] p-[11px] cursor-pointer outline-none"
  93. />
  94. <template #dropdown>
  95. <el-dropdown-menu class="translation">
  96. <el-dropdown-item
  97. :style="getDropdownItemStyle(locale, 'zh')"
  98. :class="['dark:!text-white', getDropdownItemClass(locale, 'zh')]"
  99. @click="translationCh"
  100. >
  101. <span class="check-zh" v-show="locale === 'zh'">
  102. <IconifyIconOffline icon="check" />
  103. </span>
  104. 简体中文
  105. </el-dropdown-item>
  106. <el-dropdown-item
  107. :style="getDropdownItemStyle(locale, 'en')"
  108. :class="['dark:!text-white', getDropdownItemClass(locale, 'en')]"
  109. @click="translationEn"
  110. >
  111. <span class="check-en" v-show="locale === 'en'">
  112. <IconifyIconOffline icon="check" />
  113. </span>
  114. English
  115. </el-dropdown-item>
  116. </el-dropdown-menu>
  117. </template>
  118. </el-dropdown>
  119. <!-- 退出登录 -->
  120. <el-dropdown trigger="click">
  121. <span class="el-dropdown-link navbar-bg-hover">
  122. <img v-if="avatars" :src="avatars" :style="avatarsStyle" />
  123. <p v-if="username" class="dark:text-white">{{ username }}</p>
  124. </span>
  125. <template #dropdown>
  126. <el-dropdown-menu class="logout">
  127. <el-dropdown-item @click="logout">
  128. <IconifyIconOffline
  129. icon="logout-circle-r-line"
  130. style="margin: 5px"
  131. />
  132. {{ t("buttons.hsLoginOut") }}
  133. </el-dropdown-item>
  134. </el-dropdown-menu>
  135. </template>
  136. </el-dropdown>
  137. <span
  138. class="set-icon navbar-bg-hover"
  139. :title="t('buttons.hssystemSet')"
  140. @click="onPanel"
  141. >
  142. <IconifyIconOffline icon="setting" />
  143. </span>
  144. </div>
  145. </div>
  146. </template>
  147. <style lang="scss" scoped>
  148. .translation {
  149. ::v-deep(.el-dropdown-menu__item) {
  150. padding: 5px 40px;
  151. }
  152. .check-zh {
  153. position: absolute;
  154. left: 20px;
  155. }
  156. .check-en {
  157. position: absolute;
  158. left: 20px;
  159. }
  160. }
  161. .logout {
  162. max-width: 120px;
  163. ::v-deep(.el-dropdown-menu__item) {
  164. min-width: 100%;
  165. display: inline-flex;
  166. flex-wrap: wrap;
  167. }
  168. }
  169. </style>