mirror of
https://github.com/ctwj/urldb.git
synced 2025-11-25 19:37:33 +08:00
118 lines
3.3 KiB
Vue
118 lines
3.3 KiB
Vue
<template>
|
|
<div v-if="shouldShowAnnouncement" class="announcement-container px-3 py-1">
|
|
<div class="flex items-center justify-between min-h-[24px]">
|
|
<div class="flex items-center gap-2 flex-1 overflow-hidden">
|
|
<i class="fas fa-bullhorn text-blue-600 dark:text-blue-400 text-sm flex-shrink-0"></i>
|
|
<div class="announcement-content overflow-hidden">
|
|
<div class="announcement-item active">
|
|
<span class="text-sm text-gray-700 dark:text-gray-300 truncate" v-html="validAnnouncements[currentIndex].content"></span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="flex items-center gap-1 text-xs text-gray-500 dark:text-gray-400 flex-shrink-0 ml-2">
|
|
<span>{{ (currentIndex + 1) }}/{{ validAnnouncements.length }}</span>
|
|
<button @click="nextAnnouncement" class="hover:text-blue-500 transition-colors">
|
|
<i class="fas fa-chevron-right text-xs"></i>
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
// 使用系统配置store获取公告数据
|
|
import { useSystemConfigStore } from '~/stores/systemConfig'
|
|
const systemConfigStore = useSystemConfigStore()
|
|
await systemConfigStore.initConfig(false, false)
|
|
const systemConfig = computed(() => systemConfigStore.config)
|
|
|
|
interface AnnouncementItem {
|
|
content: string
|
|
enabled: boolean
|
|
}
|
|
|
|
const currentIndex = ref(0)
|
|
const interval = ref<NodeJS.Timeout | null>(null)
|
|
|
|
// 计算有效公告(开启状态且有内容的公告)
|
|
const validAnnouncements = computed(() => {
|
|
if (!systemConfig.value?.announcements) return []
|
|
|
|
const announcements = Array.isArray(systemConfig.value.announcements)
|
|
? systemConfig.value.announcements
|
|
: JSON.parse(systemConfig.value.announcements || '[]')
|
|
|
|
return announcements.filter((item: AnnouncementItem) =>
|
|
item.enabled && item.content && item.content.trim()
|
|
)
|
|
})
|
|
|
|
// 判断是否应该显示公告
|
|
const shouldShowAnnouncement = computed(() => {
|
|
return systemConfig.value?.enable_announcements && validAnnouncements.value.length > 0
|
|
})
|
|
|
|
// 自动切换公告
|
|
const startAutoSwitch = () => {
|
|
if (validAnnouncements.value.length > 1) {
|
|
interval.value = setInterval(() => {
|
|
currentIndex.value = (currentIndex.value + 1) % validAnnouncements.value.length
|
|
}, 4000) // 每4秒切换一次
|
|
}
|
|
}
|
|
|
|
// 手动切换到下一条公告
|
|
const nextAnnouncement = () => {
|
|
currentIndex.value = (currentIndex.value + 1) % validAnnouncements.value.length
|
|
}
|
|
|
|
// 监听公告数据变化,重新开始自动切换
|
|
watch(() => validAnnouncements.value.length, (newLength) => {
|
|
if (newLength > 0) {
|
|
currentIndex.value = 0
|
|
stopAutoSwitch()
|
|
startAutoSwitch()
|
|
}
|
|
})
|
|
|
|
// 清理定时器
|
|
const stopAutoSwitch = () => {
|
|
if (interval.value) {
|
|
clearInterval(interval.value)
|
|
interval.value = null
|
|
}
|
|
}
|
|
|
|
onMounted(() => {
|
|
if (shouldShowAnnouncement.value) {
|
|
startAutoSwitch()
|
|
}
|
|
})
|
|
|
|
onUnmounted(() => {
|
|
stopAutoSwitch()
|
|
})
|
|
</script>
|
|
|
|
<style scoped>
|
|
.announcement-content {
|
|
position: relative;
|
|
height: 20px; /* 固定高度 */
|
|
}
|
|
|
|
.announcement-item {
|
|
opacity: 0;
|
|
transform: translateY(5px);
|
|
transition: all 0.5s ease-in-out;
|
|
}
|
|
|
|
.announcement-item.active {
|
|
opacity: 1;
|
|
transform: translateY(0);
|
|
}
|
|
|
|
/* 暗色主题适配 */
|
|
.dark-theme .announcement-container {
|
|
background: transparent;
|
|
}
|
|
</style> |