fix: 主题闪烁修复

This commit is contained in:
kongweigen
2026-01-14 23:40:41 +08:00
parent d9cb2c0cf3
commit e91fbcda3a
4 changed files with 106 additions and 140 deletions

View File

@@ -1,41 +1,30 @@
<template> <template>
<!-- Project card component - Compact & refined design --> <!-- Project card component - Compact design with hover actions -->
<!-- 项目卡片组件 - 紧凑精致设计 --> <!-- 项目卡片组件 - 紧凑设计悬停显示操作 -->
<article <article
class="project-card" class="project-card"
@click="$emit('click')" @click="$emit('click')"
tabindex="0" tabindex="0"
@keydown.enter="$emit('click')" @keydown.enter="$emit('click')"
> >
<!-- Cover image section / 封面图片区域 --> <!-- Gradient header with icon / 渐变头部区域 -->
<div class="card-cover"> <div class="card-header">
<img <el-icon class="header-icon"><Film /></el-icon>
v-if="coverImage" <!-- Hover actions / 悬停操作区 -->
:src="coverImage" <div class="hover-actions" @click.stop>
:alt="title" <slot name="actions"></slot>
class="cover-image"
loading="lazy"
/>
<div v-else class="cover-placeholder">
<el-icon :size="20" class="placeholder-icon">
<Film />
</el-icon>
</div> </div>
<!-- Floating tag / 悬浮标签 -->
<span v-if="tag" class="floating-tag">{{ tag }}</span>
</div> </div>
<!-- Card content / 卡片内容 --> <!-- Card content / 卡片内容 -->
<div class="card-body"> <div class="card-body">
<div class="card-header"> <h3 class="card-title">{{ title }}</h3>
<h3 class="card-title">{{ title }}</h3> <p v-if="description" class="card-description">{{ description }}</p>
<span class="meta-time">{{ formattedDate }}</span>
</div>
<p class="card-description">{{ description || '无描述' }}</p>
<!-- Actions / 操作区 --> <!-- Footer section / 底部区域 -->
<div class="card-actions" @click.stop> <div class="card-footer">
<slot name="actions"></slot> <span class="meta-time">{{ formattedDate }}</span>
<span class="episode-label"> {{ episodeCount }} </span>
</div> </div>
</div> </div>
</article> </article>
@@ -49,50 +38,50 @@ import { Film } from '@element-plus/icons-vue'
* ProjectCard - Reusable project/drama card component * ProjectCard - Reusable project/drama card component
* 项目卡片组件 - 可复用的项目展示卡片 * 项目卡片组件 - 可复用的项目展示卡片
*/ */
const props = defineProps<{ const props = withDefaults(defineProps<{
title: string title: string
description?: string description?: string
coverImage?: string
tag?: string
updatedAt: string updatedAt: string
}>() episodeCount?: number
}>(), {
description: '',
episodeCount: 0
})
defineEmits<{ defineEmits<{
click: [] click: []
}>() }>()
// Format date to relative time / 格式化日期为相对时间 // Format date / 格式化日期
const formattedDate = computed(() => { const formattedDate = computed(() => {
const date = new Date(props.updatedAt) const date = new Date(props.updatedAt)
const now = new Date() const year = date.getFullYear()
const diff = now.getTime() - date.getTime() const month = String(date.getMonth() + 1).padStart(2, '0')
const days = Math.floor(diff / (1000 * 60 * 60 * 24)) const day = String(date.getDate()).padStart(2, '0')
const hours = String(date.getHours()).padStart(2, '0')
if (days === 0) return '今天' const minutes = String(date.getMinutes()).padStart(2, '0')
if (days === 1) return '昨天' const seconds = String(date.getSeconds()).padStart(2, '0')
if (days < 7) return `${days}天前` return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`
return date.toLocaleDateString('zh-CN')
}) })
</script> </script>
<style scoped> <style scoped>
/* Card Container / 卡片容器 */ /* Card Container / 卡片容器 */
.project-card { .project-card {
position: relative;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
background: var(--bg-card); background: var(--bg-card);
border: 1px solid var(--border-primary); border: 1px solid var(--border-primary);
border-radius: var(--radius-lg); border-radius: var(--radius-md);
overflow: hidden; overflow: hidden;
cursor: pointer; cursor: pointer;
transition: all var(--transition-normal); transition: all var(--transition-normal);
box-shadow: var(--shadow-card); width: 200px;
} }
.project-card:hover { .project-card:hover {
border-color: var(--accent); border-color: var(--accent);
box-shadow: var(--shadow-card-hover), 0 0 0 1px var(--accent);
transform: translateY(-2px);
} }
.project-card:focus-visible { .project-card:focus-visible {
@@ -100,125 +89,90 @@ const formattedDate = computed(() => {
outline-offset: 2px; outline-offset: 2px;
} }
/* Cover Section / 封面区域 */ /* Card Header / 卡片头部 */
.card-cover { .card-header {
position: relative; position: relative;
width: 100%; height: 120px;
aspect-ratio: 16 / 9;
overflow: hidden;
background: linear-gradient(135deg, var(--accent) 0%, #06b6d4 100%); background: linear-gradient(135deg, var(--accent) 0%, #06b6d4 100%);
}
.cover-image {
width: 100%;
height: 100%;
object-fit: cover;
transition: transform var(--transition-slow);
}
.project-card:hover .cover-image {
transform: scale(1.05);
}
.cover-placeholder {
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
height: 100%;
} }
.placeholder-icon { .header-icon {
color: rgba(255, 255, 255, 0.7); font-size: 28px;
color: rgba(255, 255, 255, 0.8);
} }
/* Floating tag / 悬浮标签 */ /* Hover Actions / 悬停操作区 */
.floating-tag { .hover-actions {
position: absolute; position: absolute;
top: var(--space-2); top: 8px;
left: var(--space-2); right: 8px;
padding: 0.2rem 0.5rem; display: flex;
background: rgba(0, 0, 0, 0.6); gap: 4px;
color: white; opacity: 0;
font-size: 0.625rem; transition: opacity var(--transition-fast);
font-weight: 600; z-index: 10;
border-radius: var(--radius-sm); }
backdrop-filter: blur(8px);
letter-spacing: 0.02em; .project-card:hover .hover-actions {
opacity: 1;
} }
/* Body Section / 内容区域 */ /* Body Section / 内容区域 */
.card-body { .card-body {
flex: 1;
display: flex; display: flex;
flex-direction: column; flex-direction: column;
padding: var(--space-3); padding: 12px;
gap: var(--space-1); gap: 10px;
}
.card-header {
display: flex;
align-items: center;
justify-content: space-between;
gap: var(--space-2);
} }
.card-title { .card-title {
margin: 0; margin: 0;
font-size: 0.8125rem; font-size: 1.2rem;
font-weight: 600; font-weight: 600;
color: var(--text-primary); color: var(--text-primary);
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
white-space: nowrap; white-space: nowrap;
flex: 1;
letter-spacing: -0.01em;
}
.meta-time {
font-size: 0.625rem;
color: var(--text-muted);
white-space: nowrap;
font-weight: 500;
} }
.card-description { .card-description {
margin: 0; margin: 0;
font-size: 0.6875rem; font-size: 0.85rem;
color: var(--text-muted); color: var(--text-secondary);
line-height: 1.4;
display: -webkit-box;
-webkit-line-clamp: 1;
line-clamp: 1;
-webkit-box-orient: vertical;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis;
display: -webkit-box;
-webkit-line-clamp: 2;
line-clamp: 2;
-webkit-box-orient: vertical;
line-height: 1.4;
} }
/* Actions / 操作区 */ /* Footer Section / 底部区域 */
.card-actions { .card-footer {
display: flex; margin-top: auto;
gap: var(--space-1); padding-top: 8px;
align-items: center;
margin-top: var(--space-1);
padding-top: var(--space-2);
border-top: 1px solid var(--border-primary); border-top: 1px solid var(--border-primary);
display: flex;
flex-direction: column;
gap: 4px;
}
.meta-time,
.episode-label {
font-size: 0.75rem;
color: var(--text-muted);
} }
:deep(.action-button) { :deep(.action-button) {
padding: 0.25rem !important; width: 28px !important;
min-width: 1.5rem !important; height: 28px !important;
height: 1.5rem !important; padding: 0 !important;
} background: var(--bg-secondary) !important;
border: none !important;
:deep(.action-button .el-icon) {
font-size: 0.75rem !important;
}
:deep(.action-button.primary:hover) {
color: var(--accent);
background: var(--accent-light);
}
:deep(.action-button.danger:hover) {
color: var(--error);
background: var(--error-light);
} }
</style> </style>

View File

@@ -130,6 +130,7 @@ export default {
create: '创建项目', create: '创建项目',
totalProjects: '共 {count} 个项目', totalProjects: '共 {count} 个项目',
createNew: '创建新项目', createNew: '创建新项目',
createDesc: '开始创作您的短剧项目',
aiConfig: 'AI配置', aiConfig: 'AI配置',
aiConfigTip: '请先配置 AI 服务后再创建项目', aiConfigTip: '请先配置 AI 服务后再创建项目',
empty: '暂无项目,点击上方按钮创建新项目', empty: '暂无项目,点击上方按钮创建新项目',

View File

@@ -11,6 +11,13 @@ import router from './router'
import i18n from './locales' import i18n from './locales'
import './assets/styles/main.css' import './assets/styles/main.css'
// Apply theme before app mounts to prevent flash
// 在应用挂载前应用主题,防止闪烁
const savedTheme = localStorage.getItem('theme')
if (savedTheme === 'dark' || (!savedTheme && window.matchMedia('(prefers-color-scheme: dark)').matches)) {
document.documentElement.classList.add('dark')
}
const app = createApp(App) const app = createApp(App)
app.use(createPinia()) app.use(createPinia())

View File

@@ -43,23 +43,16 @@
:key="drama.id" :key="drama.id"
:title="drama.title" :title="drama.title"
:description="drama.description" :description="drama.description"
:cover-image="drama.thumbnail"
:tag="drama.genre"
:updated-at="drama.updated_at" :updated-at="drama.updated_at"
:episode-count="drama.total_episodes || 0"
@click="viewDrama(drama.id)" @click="viewDrama(drama.id)"
> >
<template #actions> <template #actions>
<ActionButton <ActionButton
:icon="Edit" :icon="Edit"
:tooltip="$t('common.edit')" :tooltip="$t('common.edit')"
variant="primary"
@click="editDrama(drama.id)" @click="editDrama(drama.id)"
/> />
<ActionButton
:icon="View"
:tooltip="$t('common.view')"
@click="viewDrama(drama.id)"
/>
<el-popconfirm <el-popconfirm
:title="$t('drama.deleteConfirm')" :title="$t('drama.deleteConfirm')"
:confirm-button-text="$t('common.confirm')" :confirm-button-text="$t('common.confirm')"
@@ -67,15 +60,15 @@
@confirm="deleteDrama(drama.id)" @confirm="deleteDrama(drama.id)"
> >
<template #reference> <template #reference>
<ActionButton <el-button
:icon="Delete" :icon="Delete"
:tooltip="$t('common.delete')" class="action-button danger"
variant="danger" link
/> />
</template> </template>
</el-popconfirm> </el-popconfirm>
</template> </template>
</ProjectCard> </ProjectCard>
</div> </div>
<!-- Edit Dialog / 编辑对话框 --> <!-- Edit Dialog / 编辑对话框 -->
@@ -349,8 +342,8 @@ onMounted(() => {
Projects Grid / 项目网格 - 紧凑间距 Projects Grid / 项目网格 - 紧凑间距
======================================== */ ======================================== */
.projects-grid { .projects-grid {
display: grid; display: flex;
grid-template-columns: repeat(2, 1fr); /* grid-template-columns: repeat(2, 1fr); */
gap: var(--space-2); gap: var(--space-2);
margin-bottom: var(--space-4); margin-bottom: var(--space-4);
min-height: 300px; min-height: 300px;
@@ -507,4 +500,15 @@ onMounted(() => {
justify-content: flex-end; justify-content: flex-end;
gap: 0.75rem; gap: 0.75rem;
} }
/* Delete button style */
.action-button.danger {
padding: 0.5rem;
color: var(--text-muted);
}
.action-button.danger:hover {
color: #ef4444;
background: rgba(239, 68, 68, 0.1);
}
</style> </style>