mixNav.vue 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240
  1. <script setup lang="ts">
  2. import { useI18n } from "vue-i18n";
  3. import Search from "../search/index.vue";
  4. import Notice from "../notice/index.vue";
  5. import { useNav } from "../../hooks/nav";
  6. import { templateRef } from "@vueuse/core";
  7. import avatars from "/@/assets/avatars.jpg";
  8. import { transformI18n } from "/@/plugins/i18n";
  9. import screenfull from "../screenfull/index.vue";
  10. import { useRoute, useRouter } from "vue-router";
  11. import { deviceDetection } from "/@/utils/deviceDetection";
  12. import { useRenderIcon } from "/@/components/ReIcon/src/hooks";
  13. import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
  14. import { getParentPaths, findRouteByPath } from "/@/router/utils";
  15. import { usePermissionStoreHook } from "/@/store/modules/permission";
  16. import globalization from "/@/assets/svg/globalization.svg?component";
  17. import { ref, watch, nextTick, onMounted, getCurrentInstance } from "vue";
  18. const route = useRoute();
  19. const { locale, t } = useI18n();
  20. const routers = useRouter().options.routes;
  21. const menuRef = templateRef<ElRef | null>("menu", null);
  22. const instance =
  23. getCurrentInstance().appContext.config.globalProperties.$storage;
  24. const {
  25. logout,
  26. onPanel,
  27. changeTitle,
  28. toggleSideBar,
  29. handleResize,
  30. menuSelect,
  31. resolvePath,
  32. pureApp,
  33. usename,
  34. getDropdownItemStyle
  35. } = useNav();
  36. let defaultActive = ref(null);
  37. function getDefaultActive(routePath) {
  38. const wholeMenus = usePermissionStoreHook().wholeMenus;
  39. // 当前路由的父级路径
  40. const parentRoutes = getParentPaths(routePath, wholeMenus)[0];
  41. defaultActive.value = findRouteByPath(
  42. parentRoutes,
  43. wholeMenus
  44. )?.children[0]?.path;
  45. }
  46. onMounted(() => {
  47. getDefaultActive(route.path);
  48. nextTick(() => {
  49. handleResize(menuRef.value);
  50. });
  51. });
  52. watch(
  53. () => locale.value,
  54. () => {
  55. changeTitle(route.meta);
  56. }
  57. );
  58. watch(
  59. () => route.path,
  60. () => {
  61. getDefaultActive(route.path);
  62. }
  63. );
  64. function translationCh() {
  65. instance.locale = { locale: "zh" };
  66. locale.value = "zh";
  67. handleResize(menuRef.value);
  68. }
  69. function translationEn() {
  70. instance.locale = { locale: "en" };
  71. locale.value = "en";
  72. handleResize(menuRef.value);
  73. }
  74. </script>
  75. <template>
  76. <div class="horizontal-header">
  77. <div
  78. :class="classes.container"
  79. :title="pureApp.sidebar.opened ? '点击折叠' : '点击展开'"
  80. @click="toggleSideBar"
  81. >
  82. <svg
  83. :fill="useEpThemeStoreHook().fill"
  84. :class="[
  85. 'hamburger',
  86. pureApp.sidebar.opened ? 'is-active-hamburger' : ''
  87. ]"
  88. viewBox="0 0 1024 1024"
  89. xmlns="http://www.w3.org/2000/svg"
  90. width="64"
  91. height="64"
  92. >
  93. <path
  94. d="M408 442h480c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8H408c-4.4 0-8 3.6-8 8v56c0 4.4 3.6 8 8 8zm-8 204c0 4.4 3.6 8 8 8h480c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8H408c-4.4 0-8 3.6-8 8v56zm504-486H120c-4.4 0-8 3.6-8 8v56c0 4.4 3.6 8 8 8h784c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8zm0 632H120c-4.4 0-8 3.6-8 8v56c0 4.4 3.6 8 8 8h784c4.4 0 8-3.6 8-8v-56c0-4.4-3.6-8-8-8zM142.4 642.1L298.7 519a8.84 8.84 0 0 0 0-13.9L142.4 381.9c-5.8-4.6-14.4-.5-14.4 6.9v246.3a8.9 8.9 0 0 0 14.4 7z"
  95. />
  96. </svg>
  97. </div>
  98. <el-menu
  99. ref="menu"
  100. class="horizontal-header-menu"
  101. mode="horizontal"
  102. :default-active="defaultActive"
  103. router
  104. @select="indexPath => menuSelect(indexPath, routers)"
  105. >
  106. <el-menu-item
  107. v-for="route in usePermissionStoreHook().wholeMenus"
  108. :key="route.path"
  109. :index="resolvePath(route) || route.redirect"
  110. >
  111. <template #title>
  112. <div v-show="route.meta.icon" :class="['el-icon', route.meta.icon]">
  113. <component :is="useRenderIcon(route.meta && route.meta.icon)" />
  114. </div>
  115. <span>{{ transformI18n(route.meta.title, route.meta.i18n) }}</span>
  116. <FontIcon
  117. v-if="route.meta.extraIcon"
  118. width="30px"
  119. height="30px"
  120. style="position: absolute; right: 10px"
  121. :icon="route.meta.extraIcon.name"
  122. :svg="route.meta.extraIcon.svg ? true : false"
  123. />
  124. </template>
  125. </el-menu-item>
  126. </el-menu>
  127. <div class="horizontal-header-right">
  128. <!-- 菜单搜索 -->
  129. <Search />
  130. <!-- 通知 -->
  131. <Notice id="header-notice" />
  132. <!-- 全屏 -->
  133. <screenfull id="header-screenfull" v-show="!deviceDetection()" />
  134. <!-- 国际化 -->
  135. <el-dropdown id="header-translation" trigger="click">
  136. <globalization />
  137. <template #dropdown>
  138. <el-dropdown-menu class="translation">
  139. <el-dropdown-item
  140. :style="getDropdownItemStyle(locale, 'zh')"
  141. @click="translationCh"
  142. ><span class="check-zh" v-show="locale === 'zh'"
  143. ><IconifyIconOffline icon="check" /></span
  144. >简体中文</el-dropdown-item
  145. >
  146. <el-dropdown-item
  147. :style="getDropdownItemStyle(locale, 'en')"
  148. @click="translationEn"
  149. ><span class="check-en" v-show="locale === 'en'"
  150. ><IconifyIconOffline icon="check" /></span
  151. >English</el-dropdown-item
  152. >
  153. </el-dropdown-menu>
  154. </template>
  155. </el-dropdown>
  156. <!-- 退出登陆 -->
  157. <el-dropdown trigger="click">
  158. <span class="el-dropdown-link">
  159. <img :src="avatars" />
  160. <p>{{ usename }}</p>
  161. </span>
  162. <template #dropdown>
  163. <el-dropdown-menu class="logout">
  164. <el-dropdown-item @click="logout">
  165. <IconifyIconOffline
  166. icon="logout-circle-r-line"
  167. style="margin: 5px"
  168. />
  169. {{ t("buttons.hsLoginOut") }}</el-dropdown-item
  170. >
  171. </el-dropdown-menu>
  172. </template>
  173. </el-dropdown>
  174. <span
  175. class="el-icon-setting"
  176. :title="t('buttons.hssystemSet')"
  177. @click="onPanel"
  178. >
  179. <IconifyIconOffline icon="setting" />
  180. </span>
  181. </div>
  182. </div>
  183. </template>
  184. <style module="classes" scoped>
  185. .container {
  186. padding: 0 15px;
  187. }
  188. </style>
  189. <style lang="scss" scoped>
  190. .hamburger {
  191. width: 20px;
  192. height: 20px;
  193. &:hover {
  194. cursor: pointer;
  195. }
  196. }
  197. .is-active-hamburger {
  198. transform: rotate(180deg);
  199. }
  200. .translation {
  201. ::v-deep(.el-dropdown-menu__item) {
  202. padding: 5px 40px;
  203. }
  204. .check-zh {
  205. position: absolute;
  206. left: 20px;
  207. }
  208. .check-en {
  209. position: absolute;
  210. left: 20px;
  211. }
  212. }
  213. .logout {
  214. max-width: 120px;
  215. ::v-deep(.el-dropdown-menu__item) {
  216. min-width: 100%;
  217. display: inline-flex;
  218. flex-wrap: wrap;
  219. }
  220. }
  221. </style>