'use client' import { useState, useEffect, useCallback } from 'react' import { useRouter, useParams } from 'next/navigation' import { useToast } from '@/components/ui/Toast' import { ArrowLeft, Play, Pause, AlertTriangle, Shield, Radio, Loader2 } from 'lucide-react' import { Card, CardContent, CardHeader, CardTitle } from '@/components/ui/Card' import { Button } from '@/components/ui/Button' import { SuccessTag, WarningTag, ErrorTag } from '@/components/ui/Tag' import { Modal, ConfirmModal } from '@/components/ui/Modal' import { ReviewSteps, getAgencyReviewSteps } from '@/components/ui/ReviewSteps' import { api } from '@/lib/api' import { USE_MOCK } from '@/contexts/AuthContext' import { useSSE } from '@/contexts/SSEContext' import type { TaskResponse, AIReviewResult } from '@/types/task' // ==================== Mock 数据 ==================== const mockTask: TaskResponse = { id: 'task-001', name: '夏日护肤推广', sequence: 1, stage: 'script_agency_review', project: { id: 'proj-001', name: 'XX品牌618推广', brand_name: 'XX护肤品牌' }, agency: { id: 'ag-001', name: '优创代理' }, creator: { id: 'cr-001', name: '小美护肤' }, script_ai_score: 85, script_ai_result: { score: 85, violations: [ { type: '违禁词', content: '效果最好', severity: 'high', suggestion: '建议替换为"效果显著"', timestamp: 15.5, source: 'speech', }, { type: '竞品露出', content: '疑似竞品Logo', severity: 'high', suggestion: '需人工确认是否为竞品露出', timestamp: 42.0, source: 'visual', }, ], soft_warnings: [ { type: '油腻预警', content: '达人表情过于夸张,建议检查', suggestion: '软性风险仅作提示' }, ], summary: '视频整体合规,发现2处硬性问题和1处舆情提示需人工确认', }, video_ai_score: 85, video_ai_result: { score: 85, violations: [ { type: '违禁词', content: '效果最好', severity: 'high', suggestion: '建议替换为"效果显著"', timestamp: 15.5, source: 'speech', }, { type: '竞品露出', content: '疑似竞品Logo', severity: 'high', suggestion: '需人工确认是否为竞品露出', timestamp: 42.0, source: 'visual', }, ], soft_warnings: [ { type: '油腻预警', content: '达人表情过于夸张,建议检查', suggestion: '软性风险仅作提示' }, ], summary: '视频整体合规,发现2处硬性问题和1处舆情提示需人工确认', }, appeal_count: 0, is_appeal: false, created_at: '2026-02-03T10:30:00Z', updated_at: '2026-02-03T10:35:00Z', } // ==================== 工具函数 ==================== function getReviewStepStatus(task: TaskResponse): string { if (task.stage.includes('agency_review')) return 'agent_reviewing' if (task.stage.includes('brand_review')) return 'brand_reviewing' if (task.stage === 'completed') return 'completed' return 'agent_reviewing' } function formatTimestamp(seconds: number): string { const mins = Math.floor(seconds / 60) const secs = Math.floor(seconds % 60) return `${mins}:${secs.toString().padStart(2, '0')}` } // ==================== 子组件 ==================== function ReviewProgressBar({ taskStatus }: { taskStatus: string }) { const steps = getAgencyReviewSteps(taskStatus) const currentStep = steps.find(s => s.status === 'current') return (
审核流程 当前:{currentStep?.label || '代理商审核'}
) } function RiskLevelTag({ level }: { level: string }) { if (level === 'high') return 高风险 if (level === 'medium') return 中风险 return 低风险 } function ReviewSkeleton() { return (
) } // ==================== 主页面 ==================== export default function ReviewPage() { const router = useRouter() const params = useParams() const toast = useToast() const taskId = params.id as string const { subscribe } = useSSE() const [task, setTask] = useState(null) const [loading, setLoading] = useState(true) const [submitting, setSubmitting] = useState(false) const [isPlaying, setIsPlaying] = useState(false) const [showApproveModal, setShowApproveModal] = useState(false) const [showRejectModal, setShowRejectModal] = useState(false) const [showForcePassModal, setShowForcePassModal] = useState(false) const [rejectReason, setRejectReason] = useState('') const [forcePassReason, setForcePassReason] = useState('') const [saveAsException, setSaveAsException] = useState(false) const [checkedViolations, setCheckedViolations] = useState>({}) const loadTask = useCallback(async () => { if (USE_MOCK) { setTask(mockTask) setLoading(false) return } try { const data = await api.getTask(taskId) setTask(data) } catch (err) { console.error('Failed to load task:', err) toast.error('加载任务失败') } finally { setLoading(false) } }, [taskId, toast]) useEffect(() => { loadTask() }, [loadTask]) useEffect(() => { const unsub1 = subscribe('task_updated', (data: any) => { if (data?.task_id === taskId) loadTask() }) const unsub2 = subscribe('review_completed', (data: any) => { if (data?.task_id === taskId) loadTask() }) return () => { unsub1(); unsub2() } }, [subscribe, taskId, loadTask]) if (loading || !task) return // Determine if this is script or video review const isVideoReview = task.stage.includes('video') const aiResult: AIReviewResult | null | undefined = isVideoReview ? task.video_ai_result : task.script_ai_result const aiScore = isVideoReview ? task.video_ai_score : task.script_ai_score const violations = aiResult?.violations || [] const softWarnings = aiResult?.soft_warnings || [] const aiSummary = aiResult?.summary || '暂无 AI 分析总结' const handleApprove = async () => { setSubmitting(true) try { if (!USE_MOCK) { if (isVideoReview) { await api.reviewVideo(taskId, { action: 'pass' }) } else { await api.reviewScript(taskId, { action: 'pass' }) } } toast.success('审核已通过') setShowApproveModal(false) router.push('/agency/review') } catch (err) { console.error('Failed to approve:', err) toast.error('操作失败,请重试') } finally { setSubmitting(false) } } const handleReject = async () => { if (!rejectReason.trim()) { toast.error('请填写驳回原因') return } setSubmitting(true) try { if (!USE_MOCK) { if (isVideoReview) { await api.reviewVideo(taskId, { action: 'reject', comment: rejectReason }) } else { await api.reviewScript(taskId, { action: 'reject', comment: rejectReason }) } } toast.success('已驳回') setShowRejectModal(false) router.push('/agency/review') } catch (err) { console.error('Failed to reject:', err) toast.error('操作失败,请重试') } finally { setSubmitting(false) } } const handleForcePass = async () => { if (!forcePassReason.trim()) { toast.error('请填写强制通过原因') return } setSubmitting(true) try { if (!USE_MOCK) { if (isVideoReview) { await api.reviewVideo(taskId, { action: 'force_pass', comment: forcePassReason }) } else { await api.reviewScript(taskId, { action: 'force_pass', comment: forcePassReason }) } } toast.success('已强制通过') setShowForcePassModal(false) router.push('/agency/review') } catch (err) { console.error('Failed to force pass:', err) toast.error('操作失败,请重试') } finally { setSubmitting(false) } } // 时间线标记 const timelineMarkers = [ ...violations.filter(v => v.timestamp != null).map(v => ({ time: v.timestamp!, type: 'hard' as const })), ].sort((a, b) => a.time - b.time) const maxTime = Math.max(120, ...timelineMarkers.map(m => m.time + 10)) return (
{/* 顶部导航 */}

{task.name}

{task.creator.name} · {task.project.brand_name || task.project.name} · {isVideoReview ? '视频审核' : '脚本审核'}

{task.is_appeal && ( 申诉重审 )}
{/* 申诉理由 */} {task.is_appeal && task.appeal_reason && (

申诉理由

{task.appeal_reason}

)} {/* 审核流程进度条 */}
{/* 左侧:视频/脚本播放器 (3/5) */}
{isVideoReview ? (
) : (

脚本预览区域

{task.script_file_name || '脚本文件'}

)} {/* 智能进度条(仅视频且有时间标记时显示) */} {isVideoReview && timelineMarkers.length > 0 && (
智能进度条(点击跳转)
{timelineMarkers.map((marker, idx) => (
0:00 {formatTimestamp(maxTime)}
硬性问题 舆情提示 卖点覆盖
)}
{/* AI 分析总结 */}
AI 分析总结 {aiScore != null && ( = 80 ? 'text-accent-green' : aiScore >= 60 ? 'text-yellow-400' : 'text-accent-coral'}`}> {aiScore}分 )}

{aiSummary}

{/* 右侧:AI 检查单 (2/5) */}
{/* 硬性合规 */} 硬性合规 ({violations.length}) {violations.length > 0 ? violations.map((v, idx) => { const key = `v-${idx}` return (
setCheckedViolations((prev) => ({ ...prev, [key]: !prev[key] }))} className="mt-1 accent-accent-indigo" />
{v.type} {v.timestamp != null && ( {formatTimestamp(v.timestamp)} )}

「{v.content}」

{v.suggestion}

) }) : (
无硬性违规
)}
{/* 舆情雷达 */} {softWarnings.length > 0 && ( 舆情雷达(仅提示) {softWarnings.map((w, idx) => (
{w.type}

{w.content}

软性风险仅作提示,不强制拦截

))}
)}
{/* 底部决策栏 */}
已检查 {Object.values(checkedViolations).filter(Boolean).length}/{violations.length} 个问题
{/* 通过确认弹窗 */} setShowApproveModal(false)} onConfirm={handleApprove} title="确认通过" message={`确定要通过此${isVideoReview ? '视频' : '脚本'}的审核吗?通过后达人将收到通知。`} confirmText="确认通过" /> {/* 驳回弹窗 */} setShowRejectModal(false)} title="驳回审核">

请填写驳回原因,已勾选的问题将自动打包发送给达人。

已选问题 ({Object.values(checkedViolations).filter(Boolean).length})

{violations.filter((_, idx) => checkedViolations[`v-${idx}`]).map((v, idx) => (
- {v.type}: {v.content}
))} {Object.values(checkedViolations).filter(Boolean).length === 0 && (
未选择任何问题
)}