mixNav.vue 6.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  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. <el-icon v-show="route.meta.icon" :class="route.meta.icon">
  113. <component
  114. :is="useRenderIcon(route.meta && route.meta.icon)"
  115. ></component>
  116. </el-icon>
  117. <span>{{ transformI18n(route.meta.title, route.meta.i18n) }}</span>
  118. <FontIcon
  119. v-if="route.meta.extraIcon"
  120. width="30px"
  121. height="30px"
  122. style="position: absolute; right: 10px"
  123. :icon="route.meta.extraIcon.name"
  124. :svg="route.meta.extraIcon.svg ? true : false"
  125. ></FontIcon>
  126. </template>
  127. </el-menu-item>
  128. </el-menu>
  129. <div class="horizontal-header-right">
  130. <!-- 菜单搜索 -->
  131. <Search />
  132. <!-- 通知 -->
  133. <Notice id="header-notice" />
  134. <!-- 全屏 -->
  135. <screenfull id="header-screenfull" v-show="!deviceDetection()" />
  136. <!-- 国际化 -->
  137. <el-dropdown id="header-translation" trigger="click">
  138. <globalization />
  139. <template #dropdown>
  140. <el-dropdown-menu class="translation">
  141. <el-dropdown-item
  142. :style="getDropdownItemStyle(locale, 'zh')"
  143. @click="translationCh"
  144. ><el-icon class="check-zh" v-show="locale === 'zh'"
  145. ><IconifyIconOffline icon="check" /></el-icon
  146. >简体中文</el-dropdown-item
  147. >
  148. <el-dropdown-item
  149. :style="getDropdownItemStyle(locale, 'en')"
  150. @click="translationEn"
  151. ><el-icon class="check-en" v-show="locale === 'en'"
  152. ><IconifyIconOffline icon="check" /></el-icon
  153. >English</el-dropdown-item
  154. >
  155. </el-dropdown-menu>
  156. </template>
  157. </el-dropdown>
  158. <!-- 退出登陆 -->
  159. <el-dropdown trigger="click">
  160. <span class="el-dropdown-link">
  161. <img :src="avatars" />
  162. <p>{{ usename }}</p>
  163. </span>
  164. <template #dropdown>
  165. <el-dropdown-menu class="logout">
  166. <el-dropdown-item @click="logout">
  167. <IconifyIconOffline
  168. icon="logout-circle-r-line"
  169. style="margin: 5px"
  170. />
  171. {{ t("buttons.hsLoginOut") }}</el-dropdown-item
  172. >
  173. </el-dropdown-menu>
  174. </template>
  175. </el-dropdown>
  176. <el-icon
  177. class="el-icon-setting"
  178. :title="t('buttons.hssystemSet')"
  179. @click="onPanel"
  180. >
  181. <IconifyIconOffline icon="setting" />
  182. </el-icon>
  183. </div>
  184. </div>
  185. </template>
  186. <style module="classes" scoped>
  187. .container {
  188. padding: 0 15px;
  189. }
  190. </style>
  191. <style lang="scss" scoped>
  192. .hamburger {
  193. width: 20px;
  194. height: 20px;
  195. &:hover {
  196. cursor: pointer;
  197. }
  198. }
  199. .is-active-hamburger {
  200. transform: rotate(180deg);
  201. }
  202. .translation {
  203. ::v-deep(.el-dropdown-menu__item) {
  204. padding: 5px 40px;
  205. }
  206. .check-zh {
  207. position: absolute;
  208. left: 20px;
  209. }
  210. .check-en {
  211. position: absolute;
  212. left: 20px;
  213. }
  214. }
  215. .logout {
  216. max-width: 120px;
  217. ::v-deep(.el-dropdown-menu__item) {
  218. min-width: 100%;
  219. display: inline-flex;
  220. flex-wrap: wrap;
  221. }
  222. }
  223. </style>