668 lines
19 KiB
Vue
668 lines
19 KiB
Vue
<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>
|