jointrashposs/components/g/Nav.vue

151 lines
7.4 KiB
Vue
Raw Normal View History

2023-07-08 10:36:02 +02:00
<template>
2023-09-29 13:30:13 +02:00
<div :class="['sticky top-0 z-[9900] transition', { 'shadow bg-opacity-90': (!disableShadow && scrollPos <= -40), 'bg-white dark:bg-gray-950': (disableShadow || scrollPos <= -40)}, (slim ? 'h-16 border-b border-slate-300 dark:border-slate-800' : 'h-16 lg:h-20')]">
2023-09-26 09:55:14 +02:00
<nav class="container mx-auto max-w-screen-xl grid items-center grid-cols-2 lg:grid-cols-6 p-4 h-full transition-[height]">
2023-07-09 20:09:50 +02:00
<div class="">
<GNuxtLink :to="localePath('/')" class="flex items-center space-x-2 hover:opacity-80">
<MiIcon class="h-8 w-8" />
<div class="font-title font-bold text-lg">{{ $t('_seo.siteName') }}</div>
2023-07-09 11:58:53 +02:00
</GNuxtLink>
2023-07-09 20:09:50 +02:00
</div>
2023-09-26 09:09:19 +02:00
<ul
2023-09-29 13:30:13 +02:00
class="fixed z-[9902] top-16 right-0 text-right p-4 w-[80vw] sm:w-[50vw] bg-slate-100/90 dark:bg-slate-950/90 shadow-lg space-y-2 transition-[transform,border-radius] lg:transition-none lg:translate-x-0 lg:backdrop-blur-none lg:w-auto lg:rounded-none lg:shadow-none lg:space-y-0 lg:p-0 lg:relative lg:top-0 lg:right-auto lg:bg-transparent dark:lg:bg-transparent lg:col-span-4 lg:space-x-8 xl:space-x-10 lg:flex lg:justify-center"
2023-09-26 09:09:19 +02:00
:class="[(scrollPos <= -40) ? 'rounded-bl-lg' : 'rounded-l-lg', navOpen ? 'translate-x-0' : 'translate-x-full']"
>
2023-07-09 20:09:50 +02:00
<li v-for="item in NavData.center">
2023-09-29 13:30:13 +02:00
<GNuxtLink :to="localePath(item.to)" @click.native="navOpen = !navOpen" :class="['block rounded-full px-4 py-2 lg:px-4 lg:py-1.5 hover:bg-slate-300 dark:hover:bg-slate-800', { 'bg-slate-300 dark:bg-slate-800 font-bold': currentPath.includes(item.to) }]">
2023-07-09 11:58:53 +02:00
<component v-if="item.icon" :is="item.icon" class="h-5 w-5" />
<template v-else>
{{ $t(item.i18n) }}
</template>
</GNuxtLink>
</li>
2023-09-26 09:09:19 +02:00
<li class="lg:hidden px-4 py-2 flex justify-end items-center space-x-4">
2023-09-29 13:30:13 +02:00
<button class="hover:opacity-80 relative before:absolute before:-z-10 before:-top-2 before:-left-2 before:w-9 before:h-9 before:rounded-full hover:before:bg-slate-300 dark:hover:before:bg-slate-600 h-5 w-5" @click="rotateColorMode()" aria-label="Change Color Mode">
2023-09-26 09:09:19 +02:00
<ClientOnly>
<SunIcon class="h-5 w-5" v-if="colorMode.preference === 'light'" />
<MoonIcon class="h-5 w-5" v-else-if="colorMode.preference === 'dark'" />
<DisplayIcon class="h-5 w-5" v-else />
</ClientOnly>
</button>
<div class="input-group">
<span class="input-group-text !rounded-l-full"><I18nIcon class="h-5 w-5" /><span class="sr-only">{{ $t('_nav.switchLang') }}</span></span>
<select class="form-select !rounded-r-full" v-model="spLocaleOption" @change="changeLocale()">
<option v-for="locale in locales" :value="locale.code">{{ locale.name }}</option>
</select>
</div>
</li>
2023-07-09 11:58:53 +02:00
</ul>
2023-09-26 09:09:19 +02:00
<div class="text-right">
<button class="p-1 lg:hidden" @click="navOpen = !navOpen">
<XIcon v-if="navOpen" class="h-5 w-5" />
<MenuIcon v-else class="h-5 w-5" />
</button>
2023-07-09 20:09:50 +02:00
<ul class="hidden lg:col-span-4 lg:space-x-4 lg:flex justify-center">
<li>
2023-09-26 09:09:19 +02:00
<button :class="['hover:opacity-80', { 'text-white': (landing && scrollPos >= -40) }]" @click="rotateColorMode()" aria-label="Change Color Mode">
2023-07-09 20:09:50 +02:00
<ClientOnly>
<SunIcon class="h-5 w-5" v-if="colorMode.preference === 'light'" />
<MoonIcon class="h-5 w-5" v-else-if="colorMode.preference === 'dark'" />
<DisplayIcon class="h-5 w-5" v-else />
2023-07-09 20:09:50 +02:00
</ClientOnly>
</button>
</li>
2023-07-16 10:53:42 +02:00
<li class="relative group">
2023-10-29 15:08:57 +01:00
<button class="hover:opacity-80"><I18nIcon :class="['h-5 w-5', { 'text-white': (landing && scrollPos >= -40) }]" /><span class="sr-only">{{ $t('_nav.switchLang') }}</span></button>
<div class="absolute top-6 right-0 hidden group-hover:block z-[9955]">
2023-07-09 20:09:50 +02:00
<ul class="px-4 py-2 bg-slate-50 dark:bg-slate-800 rounded-lg shadow-lg space-y-1">
<li v-for="locale in locales">
<GNuxtLink :to="switchLocalePath(locale.code)" :class="['hover:text-accent-600 py-1', {'text-accent-600 font-bold': currentLocale === locale.code}]">
{{ locale.name }}
</GNuxtLink>
</li>
</ul>
</div>
</li>
<li class="border-l"></li>
2023-07-15 10:34:06 +02:00
<li v-for="item in NavData.right" :class="['transition-colors', { 'text-white': (landing && scrollPos >= -40) }]">
2023-07-09 20:09:50 +02:00
<GNuxtLink :to="item.to" class="hover:opacity-80">
<component v-if="item.icon" :is="item.icon" class="h-5 w-5" />
<template v-else>
{{ $t(item.i18n) }}
</template>
</GNuxtLink>
</li>
</ul>
</div>
</nav>
</div>
2023-07-08 10:36:02 +02:00
</template>
2023-07-11 19:19:32 +02:00
<script setup lang="ts">
2023-07-09 11:58:53 +02:00
import MiIcon from '@/assets/svg/misskey_mi_bi.svg';
import I18nIcon from 'bi/translate.svg';
import SunIcon from 'bi/sun.svg';
import MoonIcon from 'bi/moon-stars.svg';
import DisplayIcon from 'bi/display.svg';
2023-09-26 09:09:19 +02:00
import MenuIcon from 'bi/list.svg';
import XIcon from 'bi/x.svg';
2023-07-09 11:58:53 +02:00
import NavData from '@/assets/data/nav';
2023-07-08 10:36:02 +02:00
2023-07-11 19:19:32 +02:00
const props = withDefaults(defineProps<{
disableShadow?: boolean;
slim?: boolean;
2023-07-15 10:34:06 +02:00
landing?: boolean;
2023-07-11 19:19:32 +02:00
}>(), {
disableShadow: false,
slim: false,
2023-07-15 10:34:06 +02:00
landing: false,
2023-07-11 19:19:32 +02:00
});
2023-09-26 09:09:19 +02:00
const navOpen = ref(false);
2023-07-09 20:09:50 +02:00
const { locales, locale: currentLocale } = useI18n();
2023-07-09 11:58:53 +02:00
const { path } = useRoute();
2023-09-26 09:09:19 +02:00
const { afterEach, push } = useRouter();
2023-09-09 08:09:04 +02:00
const currentPath = ref(path);
2023-09-23 20:02:04 +02:00
2023-09-09 08:09:04 +02:00
afterEach((to) => {
currentPath.value = to.path;
2023-09-23 20:02:04 +02:00
});
2023-09-23 20:16:27 +02:00
onMounted(() => {
const _route = useRoute();
currentPath.value = _route.path;
});
2023-07-09 11:58:53 +02:00
const switchLocalePath = useSwitchLocalePath();
const localePath = useLocalePath();
2023-09-26 09:09:19 +02:00
const spLocaleOption = ref<string>(currentLocale.value);
function changeLocale() {
const path = switchLocalePath(spLocaleOption.value);
push(path);
}
2023-07-09 20:09:50 +02:00
const colorMode = useColorMode();
function rotateColorMode() {
const values = ['system', 'light', 'dark'];
const index = values.indexOf(colorMode.preference);
const next = (index + 1) % values.length;
colorMode.preference = values[next];
}
const scrollPos = ref(0);
2023-07-10 05:55:57 +02:00
async function updatePos() {
scrollPos.value = document.body.getBoundingClientRect().y;
}
2023-09-26 09:09:19 +02:00
if (process.client) {
2023-07-10 05:55:57 +02:00
window.addEventListener('scroll', updatePos);
window.addEventListener('resize', updatePos);
2023-07-09 20:09:50 +02:00
}
onUnmounted(() => {
2023-09-26 09:09:19 +02:00
if (process.client) {
2023-07-10 05:55:57 +02:00
window.removeEventListener('scroll', updatePos);
window.removeEventListener('resize', updatePos);
2023-07-09 20:09:50 +02:00
}
});
2023-07-09 11:58:53 +02:00
</script>