forked from Mirrors/elk
62 lines
1.7 KiB
Vue
62 lines
1.7 KiB
Vue
<script setup lang="ts">
|
|
import type { RouteLocationRaw } from 'vue-router'
|
|
|
|
const { options, command, replace, preventScrollTop = false } = $defineProps<{
|
|
options: {
|
|
to: RouteLocationRaw
|
|
display: string
|
|
disabled?: boolean
|
|
name?: string
|
|
icon?: string
|
|
}[]
|
|
command?: boolean
|
|
replace?: boolean
|
|
preventScrollTop?: boolean
|
|
}>()
|
|
|
|
const router = useRouter()
|
|
const nuxtApp = useNuxtApp()
|
|
|
|
useCommands(() => command
|
|
? options.map(tab => ({
|
|
scope: 'Tabs',
|
|
|
|
name: tab.display,
|
|
icon: tab.icon ?? 'i-ri:file-list-2-line',
|
|
onActivate: () => router.replace(tab.to),
|
|
}))
|
|
: [],
|
|
)
|
|
|
|
const handleClick = (to: RouteLocationRaw) => {
|
|
if (preventScrollTop || nuxtApp.$preventScrollToTop(router.resolve(to).fullPath))
|
|
return
|
|
|
|
nuxtApp.$scrollToTop()
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<div flex w-full items-center lg:text-lg of-x-auto scrollbar-hide>
|
|
<template
|
|
v-for="(option, index) in options"
|
|
:key="option?.name || index"
|
|
>
|
|
<NuxtLink
|
|
v-if="!option.disabled"
|
|
:to="option.to"
|
|
:replace="replace"
|
|
relative flex flex-auto cursor-pointer sm:px6 px2 rounded transition-all
|
|
tabindex="1"
|
|
hover:bg-active transition-100
|
|
exact-active-class="children:(text-secondary !border-primary !op100 !text-base)"
|
|
@click="handleClick(option.to)"
|
|
>
|
|
<span ws-nowrap mxa sm:px2 sm:py3 xl:pb4 xl:pt5 py2 text-center border-b-3 text-secondary-light hover:text-secondary border-transparent>{{ option.display }}</span>
|
|
</NuxtLink>
|
|
<div v-else flex flex-auto sm:px6 px2 xl:pb4 xl:pt5>
|
|
<span ws-nowrap mxa sm:px2 sm:py3 py2 text-center text-secondary-light op50>{{ option.display }}</span>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</template>
|