This commit is contained in:
chansee97 2025-01-20 22:47:46 +08:00
commit 2f2d8726d4
3 changed files with 84 additions and 46 deletions

View File

@ -1,17 +1,17 @@
import BackTop from './common/BackTop.vue' import BackTop from './common/BackTop.vue'
import Setting from './common/Setting.vue' import Setting from './common/Setting.vue'
import SettingDrawer from './common/SettingDrawer.vue' import SettingDrawer from './common/SettingDrawer.vue'
import Breadcrumb from './header/Breadcrumb.vue' import Breadcrumb from './header/Breadcrumb.vue'
import CollapaseButton from './header/CollapaseButton.vue' import CollapaseButton from './header/CollapaseButton.vue'
import FullScreen from './header/FullScreen.vue' import FullScreen from './header/FullScreen.vue'
import Notices from './header/Notices.vue' import Notices from './header/Notices.vue'
import Search from './header/Search.vue' import Search from './header/Search.vue'
import UserCenter from './header/UserCenter.vue' import UserCenter from './header/UserCenter.vue'
import Logo from './sider/Logo.vue' import Logo from './sider/Logo.vue'
import Menu from './sider/Menu.vue' import Menu from './sider/Menu.vue'
import TabBar from './tab/TabBar.vue' import TabBar from './tab/TabBar.vue'
export { export {

View File

@ -1,6 +1,7 @@
<script setup lang="ts"> <script setup lang="ts">
import type { RouteLocationNormalized } from 'vue-router' import type { RouteLocationNormalized } from 'vue-router'
import { useAppStore, useTabStore } from '@/store' import { useAppStore, useTabStore } from '@/store'
import { useDraggable } from 'vue-draggable-plus'
import IconClose from '~icons/icon-park-outline/close' import IconClose from '~icons/icon-park-outline/close'
import IconDelete from '~icons/icon-park-outline/delete-four' import IconDelete from '~icons/icon-park-outline/delete-four'
import IconFullwith from '~icons/icon-park-outline/fullwidth' import IconFullwith from '~icons/icon-park-outline/fullwidth'
@ -10,6 +11,7 @@ import IconRight from '~icons/icon-park-outline/to-right'
import ContentFullScreen from './ContentFullScreen.vue' import ContentFullScreen from './ContentFullScreen.vue'
import DropTabs from './DropTabs.vue' import DropTabs from './DropTabs.vue'
import Reload from './Reload.vue' import Reload from './Reload.vue'
import TabBarItem from './TabBarItem.vue'
const tabStore = useTabStore() const tabStore = useTabStore()
const appStore = useAppStore() const appStore = useAppStore()
@ -98,54 +100,49 @@ function handleContextMenu(e: MouseEvent, route: RouteLocationNormalized) {
function onClickoutside() { function onClickoutside() {
showDropdown.value = false showDropdown.value = false
} }
// const [DefineTabItem, ReuseTabItem] = createReusableTemplate<{ route: RouteLocationNormalized }>()
const el = ref()
useDraggable(el, tabStore.tabs, {
animation: 150,
ghostClass: 'ghost',
})
</script> </script>
<template> <template>
<div class="wh-full flex items-end"> <div class="p-l-2 flex w-full relative">
<n-tabs <div class="flex items-end">
type="card" <TabBarItem
size="small" v-for="item in tabStore.pinTabs" :key="item.fullPath" :value="tabStore.currentTabPath" :route="item"
:tabs-padding="10" @click="handleTab(item)"
:value="tabStore.currentTabPath" />
@close="tabStore.closeTab" </div>
> <div ref="el" class="flex items-end flex-1">
<n-tab <TabBarItem
v-for="item in tabStore.pinTabs" v-for="item in tabStore.tabs" :key="item.fullPath" :value="tabStore.currentTabPath" :route="item" closable
:key="item.fullPath" @close="tabStore.closeTab"
:name="item.fullPath"
@click="router.push(item.fullPath)"
>
<div class="flex-x-center gap-2">
<nova-icon :icon="item.meta.icon" /> {{ $t(`route.${String(item.name)}`, item.meta.title) }}
</div>
</n-tab>
<n-tab
v-for="item in tabStore.tabs"
:key="item.fullPath"
closable
:name="item.fullPath"
@click="handleTab(item)" @click="handleTab(item)"
@contextmenu="handleContextMenu($event, item)" @contextmenu="handleContextMenu($event, item)"
> />
<div class="flex-x-center gap-2"> <n-dropdown
<nova-icon :icon="item.meta.icon" /> {{ $t(`route.${String(item.name)}`, item.meta.title) }} placement="bottom-start" trigger="manual" :x="x" :y="y" :options="options" :show="showDropdown"
</div> :on-clickoutside="onClickoutside" @select="handleSelect"
</n-tab> />
<template #suffix> </div>
<Reload /> <!-- <span class="m-l-auto" /> -->
<ContentFullScreen /> <n-el class="absolute right-0 flex items-center gap-1 bg-[var(--base-color)] h-full">
<DropTabs /> <Reload />
</template> <ContentFullScreen />
</n-tabs> <DropTabs />
<n-dropdown </n-el>
placement="bottom-start"
trigger="manual"
:x="x"
:y="y"
:options="options"
:show="showDropdown"
:on-clickoutside="onClickoutside"
@select="handleSelect"
/>
</div> </div>
</template> </template>
<style scoped>
.ghost {
opacity: 0.5;
background: #c4f6d5;
}
</style>

View File

@ -0,0 +1,41 @@
<script setup lang="ts">
import type { RouteLocationNormalized } from 'vue-router'
const { route, value, closable = false } = defineProps<{
route: RouteLocationNormalized
value: string
closable?: boolean
}>()
const emit = defineEmits<{
close: [string]
}>()
</script>
<template>
<n-el
class="cursor-pointer p-x-4 p-y-2 m-x-2px b b-[--divider-color] b-b-[#0000] rounded-[--border-radius]"
:class="[
value === route.fullPath ? 'c-[--primary-color]' : 'c-[--text-color-2]',
value === route.fullPath ? 'bg-[#0000]' : 'bg-[--tab-color]',
closable && 'p-r-2',
]"
style="transition: box-shadow .3s var(--n-bezier), color .3s var(--n-bezier), background-color .3s var(--n-bezier), border-color .3s var(--n-bezier);"
>
<div class="flex-center gap-2 text-nowrap">
<nova-icon :icon="route.meta.icon" />
<span>{{ $t(`route.${String(route.name)}`, route.meta.title) }}</span>
<button
v-if="closable"
type="button"
class="bg-transparent h-18px w-18px flex-center text-[var(--close-icon-color)] hover:bg-[var(--close-color-hover)] rounded-3px"
style="transition: background-color .3s var(--n-bezier), color .3s var(--n-bezier);"
@click.stop="emit('close', route.fullPath)"
>
<n-icon size="14">
<svg viewBox="0 0 12 12" version="1.1" xmlns="http://www.w3.org/2000/svg" aria-hidden="true"><g stroke="none" stroke-width="1" fill="none" fill-rule="evenodd"><g fill="currentColor" fill-rule="nonzero"><path d="M2.08859116,2.2156945 L2.14644661,2.14644661 C2.32001296,1.97288026 2.58943736,1.95359511 2.7843055,2.08859116 L2.85355339,2.14644661 L6,5.293 L9.14644661,2.14644661 C9.34170876,1.95118446 9.65829124,1.95118446 9.85355339,2.14644661 C10.0488155,2.34170876 10.0488155,2.65829124 9.85355339,2.85355339 L6.707,6 L9.85355339,9.14644661 C10.0271197,9.32001296 10.0464049,9.58943736 9.91140884,9.7843055 L9.85355339,9.85355339 C9.67998704,10.0271197 9.41056264,10.0464049 9.2156945,9.91140884 L9.14644661,9.85355339 L6,6.707 L2.85355339,9.85355339 C2.65829124,10.0488155 2.34170876,10.0488155 2.14644661,9.85355339 C1.95118446,9.65829124 1.95118446,9.34170876 2.14644661,9.14644661 L5.293,6 L2.14644661,2.85355339 C1.97288026,2.67998704 1.95359511,2.41056264 2.08859116,2.2156945 L2.14644661,2.14644661 L2.08859116,2.2156945 Z" /></g></g></svg>
</n-icon>
</button>
</div>
</n-el>
</template>