feat: 集成tags多标签功能
This commit is contained in:
parent
c180cf54a8
commit
0d240f083a
87
src/layout/components/tags/index.vue
Normal file
87
src/layout/components/tags/index.vue
Normal file
@ -0,0 +1,87 @@
|
|||||||
|
<template>
|
||||||
|
<div class="tags-wrapper" :style="{ height: useTheme.tag.height + 'px' }">
|
||||||
|
<n-space>
|
||||||
|
<n-tag
|
||||||
|
v-for="tag in useTag.tags"
|
||||||
|
:key="tag.path"
|
||||||
|
:type="useTag.activeTag === tag.path ? 'primary' : 'default'"
|
||||||
|
:closable="useTag.tags.length > 1"
|
||||||
|
@click="handleTagClick(tag.path)"
|
||||||
|
@close.stop="handleClose(tag.path)"
|
||||||
|
>
|
||||||
|
{{ tag.title }}
|
||||||
|
</n-tag>
|
||||||
|
</n-space>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<script setup name="Tags">
|
||||||
|
import { watch } from 'vue'
|
||||||
|
import { useRoute, useRouter } from 'vue-router'
|
||||||
|
import { useTagStore } from '@/store/modules/tag'
|
||||||
|
import { useThemeStore } from '@/store/modules/theme'
|
||||||
|
|
||||||
|
const route = useRoute()
|
||||||
|
const router = useRouter()
|
||||||
|
const useTag = useTagStore()
|
||||||
|
const useTheme = useThemeStore()
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => route.path,
|
||||||
|
() => {
|
||||||
|
const { name, path } = route
|
||||||
|
const title = route.meta?.title
|
||||||
|
useTag.addTag({ name, path, title })
|
||||||
|
useTag.setActiveTag(path)
|
||||||
|
},
|
||||||
|
{ immediate: true }
|
||||||
|
)
|
||||||
|
|
||||||
|
const handleTagClick = (path) => {
|
||||||
|
useTag.setActiveTag(path)
|
||||||
|
router.push(path)
|
||||||
|
}
|
||||||
|
|
||||||
|
const handleClose = (path) => {
|
||||||
|
if (path === useTag.activeTag) {
|
||||||
|
const activeIndex = useTag.tags.findIndex((item) => item.path === path)
|
||||||
|
if (activeIndex > 0) {
|
||||||
|
router.push(useTag.tags[activeIndex - 1].path)
|
||||||
|
} else {
|
||||||
|
router.push(useTag.tags[activeIndex + 1].path)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
useTag.removeTag(path)
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style lang="scss">
|
||||||
|
.tags-wrapper {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
background-color: #f5f6fb;
|
||||||
|
padding: 0 10px;
|
||||||
|
position: sticky;
|
||||||
|
top: 0;
|
||||||
|
z-index: 9;
|
||||||
|
.n-tag {
|
||||||
|
padding: 0 15px;
|
||||||
|
cursor: pointer;
|
||||||
|
.n-tag__close {
|
||||||
|
margin-left: 5px;
|
||||||
|
box-sizing: content-box;
|
||||||
|
font-size: 12px;
|
||||||
|
padding: 2px;
|
||||||
|
border-radius: 50%;
|
||||||
|
transition: all 0.7s;
|
||||||
|
&:hover {
|
||||||
|
color: #fff;
|
||||||
|
background-color: $primaryColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:hover {
|
||||||
|
color: $primaryColor;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
@ -1,20 +1,16 @@
|
|||||||
<script setup>
|
|
||||||
import AppHeader from './components/header/index.vue'
|
|
||||||
import SideMenu from './components/sidebar/index.vue'
|
|
||||||
import AppMain from './components/AppMain.vue'
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="layout">
|
<div class="layout">
|
||||||
<n-layout has-sider position="absolute">
|
<n-layout has-sider position="absolute">
|
||||||
<n-layout-sider :width="200" :collapsed-width="0" :native-scrollbar="false">
|
<n-layout-sider :width="200" :collapsed-width="0" :native-scrollbar="false">
|
||||||
<SideMenu />
|
<SideBar />
|
||||||
</n-layout-sider>
|
</n-layout-sider>
|
||||||
<n-layout>
|
<n-layout>
|
||||||
<n-layout-header>
|
<n-layout-header>
|
||||||
<AppHeader />
|
<AppHeader />
|
||||||
</n-layout-header>
|
</n-layout-header>
|
||||||
|
|
||||||
<n-layout position="absolute" style="top: 60px; background-color: #f5f6fb" :native-scrollbar="false">
|
<n-layout position="absolute" style="top: 60px; background-color: #f5f6fb" :native-scrollbar="false">
|
||||||
|
<AppTags v-if="useTheme.tag.visible" />
|
||||||
<AppMain />
|
<AppMain />
|
||||||
</n-layout>
|
</n-layout>
|
||||||
</n-layout>
|
</n-layout>
|
||||||
@ -22,6 +18,16 @@ import AppMain from './components/AppMain.vue'
|
|||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
|
<script setup>
|
||||||
|
import AppHeader from './components/header/index.vue'
|
||||||
|
import SideBar from './components/sidebar/index.vue'
|
||||||
|
import AppMain from './components/AppMain.vue'
|
||||||
|
import AppTags from './components/tags/index.vue'
|
||||||
|
import { useThemeStore } from '@/store/modules/theme'
|
||||||
|
|
||||||
|
const useTheme = useThemeStore()
|
||||||
|
</script>
|
||||||
|
|
||||||
<style lang="scss" scoped>
|
<style lang="scss" scoped>
|
||||||
.n-layout-header {
|
.n-layout-header {
|
||||||
height: 60px;
|
height: 60px;
|
||||||
|
22
src/store/modules/tag.js
Normal file
22
src/store/modules/tag.js
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
|
||||||
|
export const useTagStore = defineStore('tag', {
|
||||||
|
state() {
|
||||||
|
return {
|
||||||
|
tags: [],
|
||||||
|
activeTag: '',
|
||||||
|
}
|
||||||
|
},
|
||||||
|
actions: {
|
||||||
|
setActiveTag(path) {
|
||||||
|
this.activeTag = path
|
||||||
|
},
|
||||||
|
addTag(tag = {}) {
|
||||||
|
if (this.tags.some((item) => item.path === tag.path)) return
|
||||||
|
this.tags.push(tag)
|
||||||
|
},
|
||||||
|
removeTag(path) {
|
||||||
|
this.tags = this.tags.filter((tag) => tag.path !== path)
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
17
src/store/modules/theme/index.js
Normal file
17
src/store/modules/theme/index.js
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
|
||||||
|
export const useThemeStore = defineStore('theme', {
|
||||||
|
state() {
|
||||||
|
return {
|
||||||
|
tag: {
|
||||||
|
visible: true,
|
||||||
|
height: 50,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
},
|
||||||
|
actions: {
|
||||||
|
setTabVisible(visible) {
|
||||||
|
this.tag.visible = visible
|
||||||
|
},
|
||||||
|
},
|
||||||
|
})
|
Loading…
Reference in New Issue
Block a user