This commit is contained in:
Connor
2026-01-12 13:17:11 +08:00
parent 95851f8e69
commit 9600fc542c
132 changed files with 35734 additions and 5 deletions

View File

@@ -0,0 +1,667 @@
<template>
<div class="drama-management">
<div class="page-header">
<div class="header-left">
<el-button :icon="ArrowLeft" @click="router.back()">返回</el-button>
<div class="drama-info">
<h1>{{ drama?.title }}</h1>
</div>
</div>
</div>
<el-tabs v-model="activeTab" class="management-tabs">
<!-- 项目概览 -->
<el-tab-pane label="项目概览" name="overview">
<el-row :gutter="20">
<el-col :span="8">
<el-card shadow="hover">
<template #header>
<div class="card-header">
<el-icon :size="24" color="#409eff"><Document /></el-icon>
<span>章节统计</span>
</div>
</template>
<div class="stat-content">
<div class="stat-number">{{ episodesCount }}</div>
<div class="stat-label">已创建章节</div>
</div>
</el-card>
</el-col>
<el-col :span="8">
<el-card shadow="hover">
<template #header>
<div class="card-header">
<el-icon :size="24" color="#67c23a"><User /></el-icon>
<span>角色统计</span>
</div>
</template>
<div class="stat-content">
<div class="stat-number">{{ charactersCount }}</div>
<div class="stat-label">已创建角色</div>
</div>
</el-card>
</el-col>
<el-col :span="8">
<el-card shadow="hover">
<template #header>
<div class="card-header">
<el-icon :size="24" color="#e6a23c"><Picture /></el-icon>
<span>场景统计</span>
</div>
</template>
<div class="stat-content">
<div class="stat-number">{{ scenesCount }}</div>
<div class="stat-label">场景库数量</div>
</div>
</el-card>
</el-col>
</el-row>
<!-- 引导卡片无章节时显示 -->
<el-alert
v-if="episodesCount === 0"
title="开始创作您的第一个章节!"
type="info"
:closable="false"
style="margin-top: 20px;"
>
<template #default>
<p style="margin: 8px 0;">您的项目还没有章节请先创建一个章节开始制作</p>
<el-button type="primary" :icon="Plus" @click="createNewEpisode" style="margin-top: 8px;">
立即创建第一个章节
</el-button>
</template>
</el-alert>
<el-card shadow="never" style="margin-top: 20px;">
<template #header>
<h3>项目信息</h3>
</template>
<el-descriptions :column="2" border>
<el-descriptions-item label="项目名称">{{ drama?.title }}</el-descriptions-item>
<el-descriptions-item label="创建时间">{{ formatDate(drama?.created_at) }}</el-descriptions-item>
<el-descriptions-item label="项目描述" :span="2">
{{ drama?.description || '暂无描述' }}
</el-descriptions-item>
</el-descriptions>
</el-card>
</el-tab-pane>
<!-- 章节管理 -->
<el-tab-pane label="章节管理" name="episodes">
<div class="tab-header">
<h2>章节列表</h2>
<el-button type="primary" :icon="Plus" @click="createNewEpisode">创建新章节</el-button>
</div>
<!-- 空状态引导 -->
<el-empty
v-if="episodesCount === 0"
description="还没有章节"
style="margin-top: 40px;"
>
<template #image>
<el-icon :size="80" color="#409eff"><Document /></el-icon>
</template>
<el-button type="primary" :icon="Plus" @click="createNewEpisode">
创建第一个章节
</el-button>
</el-empty>
<el-table v-else :data="sortedEpisodes" border stripe style="margin-top: 16px;">
<el-table-column type="index" label="序号" width="80" />
<el-table-column prop="title" label="章节标题" min-width="200" />
<el-table-column label="状态" width="120">
<template #default="{ row }">
<el-tag :type="getEpisodeStatusType(row)">{{ getEpisodeStatusText(row) }}</el-tag>
</template>
</el-table-column>
<el-table-column label="分镜数" width="100">
<template #default="{ row }">
{{ row.shots?.length || 0 }}
</template>
</el-table-column>
<el-table-column label="创建时间" width="180">
<template #default="{ row }">
{{ formatDate(row.created_at) }}
</template>
</el-table-column>
<el-table-column label="操作" width="280" fixed="right">
<template #default="{ row }">
<el-button size="small" type="primary" @click="enterEpisodeWorkflow(row)">
进入制作
</el-button>
<el-button size="small" @click="editEpisode(row)">
编辑
</el-button>
<el-button size="small" type="danger" @click="deleteEpisode(row)">
删除
</el-button>
</template>
</el-table-column>
</el-table>
</el-tab-pane>
<!-- 角色管理 -->
<el-tab-pane label="角色管理" name="characters">
<div class="tab-header">
<h2>角色列表</h2>
<el-button type="primary" :icon="Plus" @click="openAddCharacterDialog">添加角色</el-button>
</div>
<el-row :gutter="16" style="margin-top: 16px;">
<el-col :span="6" v-for="character in drama?.characters" :key="character.id">
<el-card shadow="hover" class="character-card">
<div class="character-preview">
<img v-if="character.image_url" :src="fixImageUrl(character.image_url)" :alt="character.name" />
<el-avatar v-else :size="120">{{ character.name[0] }}</el-avatar>
</div>
<div class="character-info">
<h4>{{ character.name }}</h4>
<el-tag :type="character.role === 'main' ? 'danger' : 'info'" size="small">
{{ character.role === 'main' ? '主角' : character.role === 'supporting' ? '配角' : '次要' }}
</el-tag>
<p class="desc">{{ character.appearance || character.description }}</p>
</div>
<div class="character-actions">
<el-button size="small" @click="editCharacter(character)">编辑</el-button>
<el-button size="small" type="danger" @click="deleteCharacter(character)">删除</el-button>
</div>
</el-card>
</el-col>
</el-row>
<el-empty v-if="!drama?.characters || drama.characters.length === 0" description="暂无角色" />
</el-tab-pane>
<!-- 场景库管理 -->
<el-tab-pane label="场景库" name="scenes">
<div class="tab-header">
<h2>场景库</h2>
<el-button type="primary" :icon="Plus" @click="openAddSceneDialog">添加场景</el-button>
</div>
<el-row :gutter="16" style="margin-top: 16px;">
<el-col :span="6" v-for="scene in scenes" :key="scene.id">
<el-card shadow="hover" class="scene-card">
<div class="scene-preview">
<img v-if="scene.image_url" :src="fixImageUrl(scene.image_url)" :alt="scene.name" />
<div v-else class="scene-placeholder">
<el-icon :size="48"><Picture /></el-icon>
</div>
</div>
<div class="scene-info">
<h4>{{ scene.name }}</h4>
<p class="desc">{{ scene.description }}</p>
</div>
<div class="scene-actions">
<el-button size="small" @click="editScene(scene)">编辑</el-button>
<el-button size="small" type="danger" @click="deleteScene(scene)">删除</el-button>
</div>
</el-card>
</el-col>
</el-row>
<el-empty v-if="scenes.length === 0" description="暂无场景" />
</el-tab-pane>
</el-tabs>
<!-- 添加角色对话框 -->
<el-dialog v-model="addCharacterDialogVisible" title="添加角色" width="600px">
<el-form :model="newCharacter" label-width="100px">
<el-form-item label="角色名称">
<el-input v-model="newCharacter.name" placeholder="请输入角色名称" />
</el-form-item>
<el-form-item label="角色类型">
<el-select v-model="newCharacter.role" placeholder="请选择角色类型">
<el-option label="主角" value="main" />
<el-option label="配角" value="supporting" />
<el-option label="次要角色" value="minor" />
</el-select>
</el-form-item>
<el-form-item label="外貌特征">
<el-input v-model="newCharacter.appearance" type="textarea" :rows="3" placeholder="描述角色的外貌特征" />
</el-form-item>
<el-form-item label="性格特点">
<el-input v-model="newCharacter.personality" type="textarea" :rows="3" placeholder="描述角色的性格特点" />
</el-form-item>
<el-form-item label="角色描述">
<el-input v-model="newCharacter.description" type="textarea" :rows="3" placeholder="其他描述信息" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="addCharacterDialogVisible = false">取消</el-button>
<el-button type="primary" @click="addCharacter">确定</el-button>
</template>
</el-dialog>
<!-- 添加场景对话框 -->
<el-dialog v-model="addSceneDialogVisible" title="添加场景" width="600px">
<el-form :model="newScene" label-width="100px">
<el-form-item label="场景名称">
<el-input v-model="newScene.name" placeholder="请输入场景名称" />
</el-form-item>
<el-form-item label="场景描述">
<el-input v-model="newScene.description" type="textarea" :rows="4" placeholder="描述场景的特征" />
</el-form-item>
</el-form>
<template #footer>
<el-button @click="addSceneDialogVisible = false">取消</el-button>
<el-button type="primary" @click="addScene">确定</el-button>
</template>
</el-dialog>
</div>
</template>
<script setup lang="ts">
import { ref, computed, onMounted, watch } from 'vue'
import { useRouter, useRoute } from 'vue-router'
import { ElMessage, ElMessageBox } from 'element-plus'
import { ArrowLeft, Document, User, Picture, Plus } from '@element-plus/icons-vue'
import { dramaAPI } from '@/api/drama'
import type { Drama } from '@/types/drama'
const router = useRouter()
const route = useRoute()
const drama = ref<Drama>()
const activeTab = ref(route.query.tab as string || 'overview')
const scenes = ref<any[]>([])
const addCharacterDialogVisible = ref(false)
const addSceneDialogVisible = ref(false)
const newCharacter = ref({
name: '',
role: 'supporting',
appearance: '',
personality: '',
description: ''
})
const newScene = ref({
name: '',
description: ''
})
const episodesCount = computed(() => drama.value?.episodes?.length || 0)
const charactersCount = computed(() => drama.value?.characters?.length || 0)
const scenesCount = computed(() => scenes.value.length)
const sortedEpisodes = computed(() => {
if (!drama.value?.episodes) return []
return [...drama.value.episodes].sort((a, b) => a.episode_number - b.episode_number)
})
const loadDramaData = async () => {
try {
const data = await dramaAPI.get(route.params.id as string)
drama.value = data
loadScenes()
} catch (error: any) {
ElMessage.error(error.message || '加载项目数据失败')
}
}
const loadScenes = async () => {
// 场景数据已经在drama中加载了后端Preload了Scenes
if (drama.value?.scenes) {
scenes.value = drama.value.scenes
} else {
scenes.value = []
}
}
const getStatusType = (status?: string) => {
const map: Record<string, any> = {
draft: 'info',
in_progress: 'warning',
completed: 'success'
}
return map[status || 'draft'] || 'info'
}
const getStatusText = (status?: string) => {
const map: Record<string, string> = {
draft: '草稿',
in_progress: '制作中',
completed: '已完成'
}
return map[status || 'draft'] || '草稿'
}
const getEpisodeStatusType = (episode: any) => {
if (episode.shots && episode.shots.length > 0) return 'success'
if (episode.script_content) return 'warning'
return 'info'
}
const getEpisodeStatusText = (episode: any) => {
if (episode.shots && episode.shots.length > 0) return '已拆分'
if (episode.script_content) return '已创建'
return '草稿'
}
const formatDate = (date?: string) => {
if (!date) return '-'
return new Date(date).toLocaleString('zh-CN')
}
const fixImageUrl = (url: string) => {
if (!url) return ''
if (url.startsWith('http')) return url
return `${import.meta.env.VITE_API_BASE_URL}${url}`
}
const createNewEpisode = () => {
const nextEpisodeNumber = episodesCount.value + 1
router.push({
name: 'EpisodeWorkflowNew',
params: {
id: route.params.id,
episodeNumber: nextEpisodeNumber
}
})
}
const enterEpisodeWorkflow = (episode: any) => {
router.push({
name: 'EpisodeWorkflowNew',
params: {
id: route.params.id,
episodeNumber: episode.episode_number
}
})
}
const editEpisode = (episode: any) => {
ElMessage.info('编辑功能开发中')
}
const deleteEpisode = async (episode: any) => {
await ElMessageBox.confirm(
`确定要删除第${episode.episode_number}章吗?`,
'删除确认',
{
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}
)
try {
// TODO: 调用删除API
ElMessage.success('删除成功')
await loadDramaData()
} catch (error: any) {
ElMessage.error(error.message || '删除失败')
}
}
const openAddCharacterDialog = () => {
newCharacter.value = {
name: '',
role: 'supporting',
appearance: '',
personality: '',
description: ''
}
addCharacterDialogVisible.value = true
}
const addCharacter = async () => {
if (!newCharacter.value.name.trim()) {
ElMessage.warning('请输入角色名称')
return
}
try {
const existingCharacters = drama.value?.characters || []
const allCharacters = [
...existingCharacters.map(c => ({
name: c.name,
role: c.role,
appearance: c.appearance,
personality: c.personality,
description: c.description
})),
newCharacter.value
]
await dramaAPI.saveCharacters(drama.value!.id, allCharacters)
ElMessage.success('角色添加成功')
addCharacterDialogVisible.value = false
await loadDramaData()
} catch (error: any) {
ElMessage.error(error.message || '添加失败')
}
}
const editCharacter = (character: any) => {
ElMessage.info('编辑功能开发中')
}
const deleteCharacter = async (character: any) => {
if (character.library_id) {
ElMessage.warning('该角色来自角色库,请前往角色库进行删除')
return
}
await ElMessageBox.confirm(
`确定要删除角色"${character.name}"吗?`,
'删除确认',
{
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}
)
try {
const updatedCharacters = drama.value!.characters!.filter(c => c.id !== character.id)
await dramaAPI.saveCharacters(drama.value!.id, updatedCharacters.map(c => ({
name: c.name,
role: c.role,
appearance: c.appearance,
personality: c.personality,
description: c.description
})))
ElMessage.success('删除成功')
await loadDramaData()
} catch (error: any) {
ElMessage.error(error.message || '删除失败')
}
}
const openAddSceneDialog = () => {
newScene.value = {
name: '',
description: ''
}
addSceneDialogVisible.value = true
}
const addScene = async () => {
if (!newScene.value.name.trim()) {
ElMessage.warning('请输入场景名称')
return
}
try {
// TODO: 调用场景库API
ElMessage.success('场景添加成功')
addSceneDialogVisible.value = false
await loadScenes()
} catch (error: any) {
ElMessage.error(error.message || '添加失败')
}
}
const editScene = (scene: any) => {
ElMessage.info('编辑功能开发中')
}
const deleteScene = async (scene: any) => {
await ElMessageBox.confirm(
`确定要删除场景"${scene.name}"吗?`,
'删除确认',
{
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}
)
try {
// TODO: 调用删除API
ElMessage.success('删除成功')
await loadScenes()
} catch (error: any) {
ElMessage.error(error.message || '删除失败')
}
}
onMounted(() => {
loadDramaData()
loadScenes()
// 如果有query参数指定tab切换到对应tab
if (route.query.tab) {
activeTab.value = route.query.tab as string
}
})
</script>
<style scoped>
.drama-management {
padding: 20px;
background: #f5f7fa;
min-height: 100vh;
}
.page-header {
background: white;
padding: 20px;
border-radius: 8px;
margin-bottom: 20px;
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
}
.header-left {
display: flex;
align-items: center;
gap: 16px;
}
.drama-info {
display: flex;
align-items: center;
gap: 12px;
}
.drama-info h1 {
margin: 0;
font-size: 24px;
font-weight: 600;
}
.management-tabs {
background: white;
padding: 20px;
border-radius: 8px;
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.1);
}
.tab-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 16px;
}
.tab-header h2 {
margin: 0;
font-size: 20px;
font-weight: 600;
}
.card-header {
display: flex;
align-items: center;
gap: 8px;
font-weight: 600;
}
.stat-content {
text-align: center;
padding: 20px 0;
}
.stat-number {
font-size: 36px;
font-weight: 700;
color: #409eff;
margin-bottom: 8px;
}
.stat-label {
font-size: 14px;
color: #909399;
}
.character-card, .scene-card {
margin-bottom: 16px;
}
.character-preview, .scene-preview {
display: flex;
justify-content: center;
align-items: center;
height: 180px;
background: linear-gradient(135deg, #f5f7fa 0%, #e8eaf0 100%);
margin: -20px -20px 12px -20px;
overflow: hidden;
}
.character-preview img, .scene-preview img {
width: 100%;
height: 100%;
object-fit: cover;
}
.scene-placeholder {
color: #c0c4cc;
}
.character-info, .scene-info {
text-align: center;
margin-bottom: 12px;
}
.character-info h4, .scene-info h4 {
margin: 0 0 8px 0;
font-size: 16px;
font-weight: 600;
}
.desc {
font-size: 13px;
color: #606266;
margin: 8px 0;
overflow: hidden;
text-overflow: ellipsis;
display: -webkit-box;
-webkit-line-clamp: 2;
-webkit-box-orient: vertical;
}
.character-actions, .scene-actions {
display: flex;
gap: 8px;
justify-content: center;
}
</style>