auth-web/src/layout/components/lay-sidebar/NavMix.vue

183 lines
5.5 KiB
Vue
Raw Normal View History

2024-09-26 09:38:02 +08:00
<script lang="ts" setup>
import { useRenderIcon } from '@/components/CommonIcon/src/hooks';
import { useNav } from '@/layout/hooks/useNav';
import { findRouteByPath, getParentPaths } from '@/router/utils';
import { usePermissionStoreHook } from '@/store/permission';
import { isAllEmpty } from '@pureadmin/utils';
import { nextTick, onMounted, ref, toRaw, watch } from 'vue';
import { useTranslationLang } from '../../hooks/useTranslationLang';
import LayNotice from '../lay-notice/index.vue';
import LaySearch from '../lay-search/index.vue';
import LaySidebarExtraIcon from '../lay-sidebar/components/SidebarExtraIcon.vue';
import LaySidebarFullScreen from '../lay-sidebar/components/SidebarFullScreen.vue';
2024-09-26 09:38:02 +08:00
import GlobalizationIcon from '@/assets/svg/globalization.svg?component';
import Check from '@iconify-icons/ep/check';
import LogoutCircleRLine from '@iconify-icons/ri/logout-circle-r-line';
import Setting from '@iconify-icons/ri/settings-3-line';
import { useI18n } from 'vue-i18n';
import { $t } from '@/plugins/i18n';
2024-09-26 09:38:02 +08:00
const menuRef = ref();
const defaultActive = ref(null);
const { t } = useI18n();
const { route, locale, translationCh, translationEn } = useTranslationLang(menuRef);
2025-04-24 13:43:37 +08:00
const {
device,
logout,
onPanel,
resolvePath,
username,
userAvatar,
getDivStyle,
avatarsStyle,
getDropdownItemStyle,
getDropdownItemClass,
} = useNav();
2024-09-26 09:38:02 +08:00
function getDefaultActive(routePath) {
2025-04-24 13:43:37 +08:00
const wholeMenus = usePermissionStoreHook().wholeMenus;
/** 当前路由的父级路径 */
const parentRoutes = getParentPaths(routePath, wholeMenus)[0];
defaultActive.value = !isAllEmpty(route.meta?.activePath)
? route.meta.activePath
: findRouteByPath(parentRoutes, wholeMenus)?.children[0]?.path;
2024-09-26 09:38:02 +08:00
}
onMounted(() => {
2025-04-24 13:43:37 +08:00
getDefaultActive(route.path);
2024-09-26 09:38:02 +08:00
});
nextTick(() => {
2025-04-24 13:43:37 +08:00
menuRef.value?.handleResize();
2024-09-26 09:38:02 +08:00
});
watch(
2025-04-24 13:43:37 +08:00
() => [route.path, usePermissionStoreHook().wholeMenus],
() => {
getDefaultActive(route.path);
}
2024-09-26 09:38:02 +08:00
);
</script>
<template>
2025-04-24 13:43:37 +08:00
<div
v-if="device !== 'mobile'"
v-loading="usePermissionStoreHook().wholeMenus.length === 0"
class="horizontal-header"
>
<el-menu
ref="menuRef"
:default-active="defaultActive"
class="horizontal-header-menu"
mode="horizontal"
popper-class="pure-scrollbar"
router
>
<el-menu-item
v-for="route in usePermissionStoreHook().wholeMenus"
:key="route.path"
:index="resolvePath(route) || route.redirect"
>
<template #title>
<div v-if="toRaw(route.meta.icon)" :class="['sub-menu-icon', route.meta.icon]">
<component :is="useRenderIcon(route.meta && toRaw(route.meta.icon))" />
</div>
<div :style="getDivStyle">
<span class="select-none">
{{ t(route.meta.title) }}
</span>
<LaySidebarExtraIcon :extraIcon="route.meta.extraIcon" />
</div>
</template>
</el-menu-item>
</el-menu>
<div class="horizontal-header-right">
<!-- 菜单搜索 -->
<LaySearch id="header-search" />
<!-- 国际化 -->
<el-dropdown id="header-translation" trigger="click">
<GlobalizationIcon class="navbar-bg-hover w-[40px] h-[48px] p-[11px] cursor-pointer outline-none" />
<template #dropdown>
<el-dropdown-menu class="translation">
<el-dropdown-item
:class="['dark:!text-white', getDropdownItemClass(locale, 'zh')]"
:style="getDropdownItemStyle(locale, 'zh')"
@click="translationCh"
>
<span v-show="locale === 'zh'" class="check-zh">
<IconifyIconOffline :icon="Check" />
</span>
简体中文
</el-dropdown-item>
<el-dropdown-item
:class="['dark:!text-white', getDropdownItemClass(locale, 'en')]"
:style="getDropdownItemStyle(locale, 'en')"
@click="translationEn"
>
<span v-show="locale === 'en'" class="check-en">
<IconifyIconOffline :icon="Check" />
</span>
English
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
<!-- 全屏 -->
<LaySidebarFullScreen id="full-screen" />
<!-- 消息通知 -->
<LayNotice id="header-notice" />
<!-- 退出登录 -->
<el-dropdown trigger="click">
<span class="el-dropdown-link navbar-bg-hover select-none">
<img :src="userAvatar" :style="avatarsStyle" />
<p v-if="username" class="dark:text-white">{{ username }}</p>
</span>
<template #dropdown>
<el-dropdown-menu class="logout">
<el-dropdown-item @click="logout">
<IconifyIconOffline :icon="LogoutCircleRLine" style="margin: 5px" />
{{ $t('buttons.pureLoginOut') }}
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
<span :title="$t('buttons.pureOpenSystemSet')" class="set-icon navbar-bg-hover" @click="onPanel">
<IconifyIconOffline :icon="Setting" />
</span>
</div>
</div>
2024-09-26 09:38:02 +08:00
</template>
<style lang="scss" scoped>
:deep(.el-loading-mask) {
2025-04-24 13:43:37 +08:00
opacity: 0.45;
2024-09-26 09:38:02 +08:00
}
.translation {
2025-04-24 13:43:37 +08:00
::v-deep(.el-dropdown-menu__item) {
padding: 5px 40px;
}
2024-09-26 09:38:02 +08:00
2025-04-24 13:43:37 +08:00
.check-zh {
position: absolute;
left: 20px;
}
2024-09-26 09:38:02 +08:00
2025-04-24 13:43:37 +08:00
.check-en {
position: absolute;
left: 20px;
}
2024-09-26 09:38:02 +08:00
}
.logout {
2025-04-24 13:43:37 +08:00
width: 120px;
2024-09-26 09:38:02 +08:00
2025-04-24 13:43:37 +08:00
::v-deep(.el-dropdown-menu__item) {
display: inline-flex;
flex-wrap: wrap;
min-width: 100%;
}
2024-09-26 09:38:02 +08:00
}
</style>