123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242 |
- <script setup lang="ts">
- import { useI18n } from "vue-i18n";
- import Search from "../search/index.vue";
- import Notice from "../notice/index.vue";
- import { useNav } from "../../hooks/nav";
- import { templateRef } from "@vueuse/core";
- import avatars from "/@/assets/avatars.jpg";
- import { transformI18n } from "/@/plugins/i18n";
- import screenfull from "../screenfull/index.vue";
- import { useRoute, useRouter } from "vue-router";
- import { deviceDetection } from "/@/utils/deviceDetection";
- import { useRenderIcon } from "/@/components/ReIcon/src/hooks";
- import { useEpThemeStoreHook } from "/@/store/modules/epTheme";
- import { getParentPaths, findRouteByPath } from "/@/router/utils";
- import { usePermissionStoreHook } from "/@/store/modules/permission";
- import globalization from "/@/assets/svg/globalization.svg?component";
- import { ref, watch, nextTick, onMounted, getCurrentInstance } from "vue";
- const route = useRoute();
- const { locale, t } = useI18n();
- const routers = useRouter().options.routes;
- const menuRef = templateRef<ElRef | null>("menu", null);
- const instance =
- getCurrentInstance().appContext.config.globalProperties.$storage;
- const {
- logout,
- onPanel,
- changeTitle,
- toggleSideBar,
- handleResize,
- menuSelect,
- resolvePath,
- pureApp,
- usename,
- getDropdownItemStyle
- } = useNav();
- let defaultActive = ref(null);
- function getDefaultActive(routePath) {
- const wholeMenus = usePermissionStoreHook().wholeMenus;
- // 当前路由的父级路径
- const parentRoutes = getParentPaths(routePath, wholeMenus)[0];
- defaultActive.value = findRouteByPath(
- parentRoutes,
- wholeMenus
- )?.children[0]?.path;
- }
- onMounted(() => {
- getDefaultActive(route.path);
- nextTick(() => {
- handleResize(menuRef.value);
- });
- });
- watch(
- () => locale.value,
- () => {
- changeTitle(route.meta);
- }
- );
- watch(
- () => route.path,
- () => {
- getDefaultActive(route.path);
- }
- );
- function translationCh() {
- instance.locale = { locale: "zh" };
- locale.value = "zh";
- handleResize(menuRef.value);
- }
- function translationEn() {
- instance.locale = { locale: "en" };
- locale.value = "en";
- handleResize(menuRef.value);
- }
- </script>
- <template>
- <div class="horizontal-header">
- <div
- :class="classes.container"
- :title="pureApp.sidebar.opened ? '点击折叠' : '点击展开'"
- @click="toggleSideBar"
- >
- <svg
- :fill="useEpThemeStoreHook().fill"
- :class="[
- 'hamburger',
- pureApp.sidebar.opened ? 'is-active-hamburger' : ''
- ]"
- viewBox="0 0 1024 1024"
- xmlns="http://www.w3.org/2000/svg"
- width="64"
- height="64"
- >
- <path
- 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"
- />
- </svg>
- </div>
- <el-menu
- ref="menu"
- class="horizontal-header-menu"
- mode="horizontal"
- :default-active="defaultActive"
- router
- @select="indexPath => menuSelect(indexPath, routers)"
- >
- <el-menu-item
- v-for="route in usePermissionStoreHook().wholeMenus"
- :key="route.path"
- :index="resolvePath(route) || route.redirect"
- >
- <template #title>
- <el-icon v-show="route.meta.icon" :class="route.meta.icon">
- <component
- :is="useRenderIcon(route.meta && route.meta.icon)"
- ></component>
- </el-icon>
- <span>{{ transformI18n(route.meta.title, route.meta.i18n) }}</span>
- <FontIcon
- v-if="route.meta.extraIcon"
- width="30px"
- height="30px"
- style="position: absolute; right: 10px"
- :icon="route.meta.extraIcon.name"
- :svg="route.meta.extraIcon.svg ? true : false"
- ></FontIcon>
- </template>
- </el-menu-item>
- </el-menu>
- <div class="horizontal-header-right">
- <!-- 菜单搜索 -->
- <Search />
- <!-- 通知 -->
- <Notice id="header-notice" />
- <!-- 全屏 -->
- <screenfull id="header-screenfull" v-show="!deviceDetection()" />
- <!-- 国际化 -->
- <el-dropdown id="header-translation" trigger="click">
- <globalization />
- <template #dropdown>
- <el-dropdown-menu class="translation">
- <el-dropdown-item
- :style="getDropdownItemStyle(locale, 'zh')"
- @click="translationCh"
- ><el-icon class="check-zh" v-show="locale === 'zh'"
- ><IconifyIconOffline icon="check" /></el-icon
- >简体中文</el-dropdown-item
- >
- <el-dropdown-item
- :style="getDropdownItemStyle(locale, 'en')"
- @click="translationEn"
- ><el-icon class="check-en" v-show="locale === 'en'"
- ><IconifyIconOffline icon="check" /></el-icon
- >English</el-dropdown-item
- >
- </el-dropdown-menu>
- </template>
- </el-dropdown>
- <!-- 退出登陆 -->
- <el-dropdown trigger="click">
- <span class="el-dropdown-link">
- <img :src="avatars" />
- <p>{{ usename }}</p>
- </span>
- <template #dropdown>
- <el-dropdown-menu class="logout">
- <el-dropdown-item @click="logout">
- <IconifyIconOffline
- icon="logout-circle-r-line"
- style="margin: 5px"
- />
- {{ t("buttons.hsLoginOut") }}</el-dropdown-item
- >
- </el-dropdown-menu>
- </template>
- </el-dropdown>
- <el-icon
- class="el-icon-setting"
- :title="t('buttons.hssystemSet')"
- @click="onPanel"
- >
- <IconifyIconOffline icon="setting" />
- </el-icon>
- </div>
- </div>
- </template>
- <style module="classes" scoped>
- .container {
- padding: 0 15px;
- }
- </style>
- <style lang="scss" scoped>
- .hamburger {
- width: 20px;
- height: 20px;
- &:hover {
- cursor: pointer;
- }
- }
- .is-active-hamburger {
- transform: rotate(180deg);
- }
- .translation {
- ::v-deep(.el-dropdown-menu__item) {
- padding: 5px 40px;
- }
- .check-zh {
- position: absolute;
- left: 20px;
- }
- .check-en {
- position: absolute;
- left: 20px;
- }
- }
- .logout {
- max-width: 120px;
- ::v-deep(.el-dropdown-menu__item) {
- min-width: 100%;
- display: inline-flex;
- flex-wrap: wrap;
- }
- }
- </style>
|