'use client'; import { useParams, useRouter } from 'next/navigation'; import { motion } from 'framer-motion'; import { ArrowLeft, Play, Sparkles, RefreshCw, Clock, CheckCircle2, AlertCircle, Loader2, FileText, Film, Trash2, MoreVertical, } from 'lucide-react'; import Link from 'next/link'; import { useState } from 'react'; import { useProject, useGenerateScript, useApproveAndQueue, useUpdateProject, useDeleteProject, useGenerateSceneImage, useUpscaleSceneImage, useRegenerateScene } from '@/hooks/use-api'; import { useRenderProgress } from '@/hooks/use-render-progress'; import { SceneCard } from '@/components/project/scene-card'; import { RenderProgress } from '@/components/project/render-progress'; import { VideoPlayer } from '@/components/project/video-player'; import { projectsApi } from '@/lib/api/api-service'; // X (Twitter) ikonunu burada da tanımlıyoruz const XIcon = ({ size = 16 }: { size?: number }) => ( ); const STATUS_MAP: Record = { DRAFT: { label: 'Taslak', color: 'text-slate-400', icon: FileText, bgClass: 'bg-slate-500/10 border-slate-500/20' }, GENERATING_SCRIPT: { label: 'Senaryo Üretiliyor', color: 'text-violet-400', icon: Sparkles, bgClass: 'bg-violet-500/10 border-violet-500/20' }, PENDING: { label: 'Kuyrukta', color: 'text-amber-400', icon: Clock, bgClass: 'bg-amber-500/10 border-amber-500/20' }, GENERATING_MEDIA: { label: 'Medya Üretiliyor', color: 'text-cyan-400', icon: Sparkles, bgClass: 'bg-cyan-500/10 border-cyan-500/20' }, RENDERING: { label: 'Video İşleniyor', color: 'text-blue-400', icon: Film, bgClass: 'bg-blue-500/10 border-blue-500/20' }, COMPLETED: { label: 'Tamamlandı', color: 'text-emerald-400', icon: CheckCircle2, bgClass: 'bg-emerald-500/10 border-emerald-500/20' }, FAILED: { label: 'Başarısız', color: 'text-red-400', icon: AlertCircle, bgClass: 'bg-red-500/10 border-red-500/20' }, }; const videoStyles = [ // Film & Sinema { id: "CINEMATIC", label: "Sinematik", emoji: "🎬", desc: "Film kalitesinde görseller", category: "Film & Sinema" }, { id: "DOCUMENTARY", label: "Belgesel", emoji: "📹", desc: "Bilimsel ve ciddi ton", category: "Film & Sinema" }, { id: "STORYTELLING", label: "Hikâye Anlatımı", emoji: "📖", desc: "Anlatı odaklı, sürükleyici", category: "Film & Sinema" }, { id: "NEWS", label: "Haber", emoji: "📰", desc: "Güncel ve bilgilendirici", category: "Film & Sinema" }, { id: "ARTISTIC", label: "Sanatsal", emoji: "🎨", desc: "Yaratıcı ve sıra dışı", category: "Film & Sinema" }, { id: "NOIR", label: "Film Noir", emoji: "🖤", desc: "Karanlık, dramatik", category: "Film & Sinema" }, { id: "VLOG", label: "Vlog", emoji: "📱", desc: "Günlük, samimi", category: "Film & Sinema" }, // Animasyon { id: "ANIME", label: "Anime", emoji: "⛩️", desc: "Japon animasyon stili", category: "Animasyon" }, { id: "ANIMATION_3D", label: "3D Animasyon", emoji: "🧊", desc: "Pixar kalitesi", category: "Animasyon" }, { id: "ANIMATION_2D", label: "2D Animasyon", emoji: "✏️", desc: "Klasik el çizimi", category: "Animasyon" }, { id: "STOP_MOTION", label: "Stop Motion", emoji: "🧸", desc: "Kare kare animasyon", category: "Animasyon" }, { id: "MOTION_COMIC", label: "Hareketli Çizgi Roman", emoji: "💥", desc: "Panel bazlı anlatım", category: "Animasyon" }, { id: "CARTOON", label: "Karikatür", emoji: "🎭", desc: "Çizgi film stili", category: "Animasyon" }, { id: "CLAYMATION", label: "Claymation", emoji: "🏺", desc: "Kil animasyon", category: "Animasyon" }, { id: "PIXEL_ART", label: "Pixel Art", emoji: "👾", desc: "8-bit retro oyun", category: "Animasyon" }, { id: "ISOMETRIC", label: "İzometrik", emoji: "🔷", desc: "İzometrik animasyon", category: "Animasyon" }, // Eğitim & Bilgi { id: "EDUCATIONAL", label: "Eğitim", emoji: "🎓", desc: "Öğretici ve açıklayıcı", category: "Eğitim & Bilgi" }, { id: "INFOGRAPHIC", label: "İnfografik", emoji: "📊", desc: "Veri görselleştirme", category: "Eğitim & Bilgi" }, { id: "WHITEBOARD", label: "Whiteboard", emoji: "📝", desc: "Tahta animasyonu", category: "Eğitim & Bilgi" }, { id: "EXPLAINER", label: "Explainer", emoji: "💡", desc: "Ürün/konsept anlatımı", category: "Eğitim & Bilgi" }, { id: "DATA_VIZ", label: "Veri Görselleştirme", emoji: "📈", desc: "Grafikler ve tablolar", category: "Eğitim & Bilgi" }, // Retro & Nostaljik { id: "RETRO_80S", label: "Retro 80s", emoji: "🕹️", desc: "Synthwave estetik", category: "Retro & Nostaljik" }, { id: "VINTAGE_FILM", label: "Vintage Film", emoji: "📽️", desc: "Super 8 filmi", category: "Retro & Nostaljik" }, { id: "VHS", label: "VHS", emoji: "📼", desc: "Kaset estetik", category: "Retro & Nostaljik" }, { id: "POLAROID", label: "Polaroid", emoji: "📸", desc: "Analog fotoğraf", category: "Retro & Nostaljik" }, { id: "RETRO_90S", label: "Retro 90s Y2K", emoji: "💿", desc: "Y2K & internet", category: "Retro & Nostaljik" }, // Sanat Akımları { id: "WATERCOLOR", label: "Suluboya", emoji: "🎨", desc: "Suluboya resim", category: "Sanat Akımları" }, { id: "OIL_PAINTING", label: "Yağlı Boya", emoji: "🖌️", desc: "Klasik tuval", category: "Sanat Akımları" }, { id: "IMPRESSIONIST", label: "Empresyonist", emoji: "🌅", desc: "Monet tarzı", category: "Sanat Akımları" }, { id: "POP_ART", label: "Pop Art", emoji: "🎯", desc: "Warhol stili", category: "Sanat Akımları" }, { id: "UKIYO_E", label: "Ukiyo-e", emoji: "🏯", desc: "Japon gravür", category: "Sanat Akımları" }, { id: "ART_DECO", label: "Art Deco", emoji: "✨", desc: "1920s zarafet", category: "Sanat Akımları" }, { id: "SURREAL", label: "Sürrealist", emoji: "🌀", desc: "Dalí tarzı", category: "Sanat Akımları" }, { id: "COMIC_BOOK", label: "Çizgi Roman", emoji: "💬", desc: "Marvel/DC stili", category: "Sanat Akımları" }, { id: "SKETCH", label: "Karakalem", emoji: "✍️", desc: "Kalem çizim", category: "Sanat Akımları" }, // Modern & Minimal { id: "MINIMALIST", label: "Minimalist", emoji: "⚪", desc: "Apple estetiği", category: "Modern & Minimal" }, { id: "GLASSMORPHISM", label: "Glassmorphism", emoji: "🔮", desc: "Cam efekti", category: "Modern & Minimal" }, { id: "NEON", label: "Neon Glow", emoji: "💜", desc: "Neon ışıkları", category: "Modern & Minimal" }, { id: "CYBERPUNK", label: "Cyberpunk", emoji: "🤖", desc: "Gelecek distopya", category: "Modern & Minimal" }, { id: "STEAMPUNK", label: "Steampunk", emoji: "⚙️", desc: "Viktoryan mekanik", category: "Modern & Minimal" }, { id: "ABSTRACT", label: "Soyut", emoji: "🔵", desc: "Abstract sanat", category: "Modern & Minimal" }, // Fotoğrafik { id: "PRODUCT", label: "Ürün Fotoğrafı", emoji: "📦", desc: "Studio çekim", category: "Fotoğrafik" }, { id: "FASHION", label: "Moda", emoji: "👗", desc: "Editöryal çekim", category: "Fotoğrafik" }, { id: "AERIAL", label: "Havadan", emoji: "🚁", desc: "Drone görüntüsü", category: "Fotoğrafik" }, { id: "MACRO", label: "Makro", emoji: "🔬", desc: "Yakın çekim", category: "Fotoğrafik" }, { id: "PORTRAIT", label: "Portre", emoji: "🧑", desc: "Portre fotoğraf", category: "Fotoğrafik" }, ]; const stagger = { hidden: { opacity: 0 }, show: { opacity: 1, transition: { staggerChildren: 0.06 } }, }; const fadeUp = { hidden: { opacity: 0, y: 12 }, show: { opacity: 1, y: 0, transition: { duration: 0.4, ease: [0.16, 1, 0.3, 1] as const } }, }; export default function ProjectDetailPage() { const { id } = useParams<{ id: string }>(); const router = useRouter(); const [showMenu, setShowMenu] = useState(false); const [regeneratingSceneId, setRegeneratingSceneId] = useState(null); // Veri hook'ları const { data: project, isLoading, error, refetch } = useProject(id); const generateScriptMutation = useGenerateScript(); const approveMutation = useApproveAndQueue(); const deleteMutation = useDeleteProject(); const generateImageMutation = useGenerateSceneImage(); const upscaleImageMutation = useUpscaleSceneImage(); const regenerateSceneMutation = useRegenerateScene(); const [generatingImageId, setGeneratingImageId] = useState(null); const [upscalingImageId, setUpscalingImageId] = useState(null); // WebSocket progress const renderState = useRenderProgress( project?.status && ['PENDING', 'GENERATING_MEDIA', 'RENDERING'].includes(project.status) ? id : undefined, ); // Sahne güncelleme const handleSceneUpdate = async (sceneId: string, data: { narrationText?: string; visualPrompt?: string; subtitleText?: string }) => { try { await projectsApi.update(`${id}/scenes/${sceneId}` as unknown as string, data as any); refetch(); } catch (err) { console.error('Sahne güncelleme hatası:', err); } }; // Sahne yeniden üretim const handleSceneRegenerate = async (sceneId: string) => { setRegeneratingSceneId(sceneId); regenerateSceneMutation.mutate( { projectId: id, sceneId }, { onSettled: () => setRegeneratingSceneId(null), onSuccess: () => refetch(), } ); }; // Sahne görseli oluşturma const handleGenerateImage = (sceneId: string, customPrompt?: string) => { setGeneratingImageId(sceneId); generateImageMutation.mutate( { projectId: id, sceneId, customPrompt }, { onSettled: () => setGeneratingImageId(null), onSuccess: () => refetch(), } ); }; // Sahne görseli upscale const handleUpscaleImage = (sceneId: string) => { setUpscalingImageId(sceneId); upscaleImageMutation.mutate( { projectId: id, sceneId }, { onSettled: () => setUpscalingImageId(null), onSuccess: () => refetch(), } ); }; // Senaryo üret const handleGenerateScript = () => { generateScriptMutation.mutate(id, { onSuccess: () => refetch(), }); }; // Onayla ve gönder const handleApprove = () => { approveMutation.mutate(id, { onSuccess: () => refetch(), }); }; // Sil const handleDelete = () => { if (confirm('Bu projeyi silmek istediğinize emin misiniz?')) { deleteMutation.mutate(id, { onSuccess: () => router.push('/dashboard/projects'), }); } }; // ── Loading ── if (isLoading) { return ( Proje yükleniyor... ); } // ── Error ── if (error || !project) { return ( Proje Bulunamadı Bu proje silinmiş veya erişim izniniz yok. Projelere Dön ); } const statusInfo = STATUS_MAP[project.status] || STATUS_MAP.DRAFT; const StatusIcon = statusInfo.icon; const isRendering = ['PENDING', 'GENERATING_MEDIA', 'RENDERING', 'GENERATING_SCRIPT'].includes(project.status); const isEditable = !isRendering; const hasScript = project.scenes && project.scenes.length > 0; const isCompleted = project.status === 'COMPLETED'; const tweetData = project.sourceTweetData as Record | undefined; const currentStyle = videoStyles.find(s => s.id === project.videoStyle); const handleStyleChange = async (newStyleId: string) => { if (newStyleId === project.videoStyle) return; try { await projectsApi.update(id, { videoStyle: newStyleId } as any); refetch(); } catch (err) { console.error('Üslup (Stil) değiştirme hatası:', err); } }; return ( {/* ── Üst Bar — Geri + Aksiyonlar ── */} Projeler setShowMenu(!showMenu)} className="w-8 h-8 rounded-lg flex items-center justify-center text-[var(--color-text-muted)] hover:bg-[var(--color-bg-elevated)] transition-colors" > {showMenu && ( { refetch(); setShowMenu(false); }} className="w-full flex items-center gap-2 px-3 py-2 text-sm text-[var(--color-text-secondary)] hover:bg-[var(--color-bg-elevated)] rounded-lg transition-colors" > Yenile { handleDelete(); setShowMenu(false); }} className="w-full flex items-center gap-2 px-3 py-2 text-sm text-red-400 hover:bg-red-500/10 rounded-lg transition-colors" > Sil )} {/* ── Proje Header ── */} {/* Durum badge */} {statusInfo.label} {/* Tweet kaynak badge */} {project.sourceType === 'X_TWEET' && ( Tweet )} {project.title} {project.description && ( {project.description} )} {/* Meta bilgiler */} {project.targetDuration}s {isEditable ? ( handleStyleChange(e.target.value)} className="bg-[var(--color-bg-base)] border border-[var(--color-border-faint)] rounded-md px-2 py-0.5 text-xs text-[var(--color-text-secondary)] focus:outline-none focus:border-violet-500/50 cursor-pointer" > {videoStyles.map((s) => ( {s.emoji} {s.label} ))} ) : ( {currentStyle ? `${currentStyle.emoji} ${currentStyle.label}` : project.videoStyle} )} {project.language} {new Date(project.createdAt).toLocaleDateString('tr-TR', { day: 'numeric', month: 'short', year: 'numeric', })} {/* Tweet kaynak bilgisi */} {tweetData && ( @{tweetData.authorUsername as string} {tweetData.viralScore && ( 🔥 {String(tweetData.viralScore)}/100 )} {tweetData.text as string} )} {/* Aksiyon butonları */} {/* Senaryo üret (draft, senaryo yok) */} {isEditable && !hasScript && ( {generateScriptMutation.isPending ? ( ) : ( )} AI ile Senaryo Üret )} {/* Senaryo yeniden üret (draft/failed, senaryo var) */} {isEditable && hasScript && ( {generateScriptMutation.isPending ? ( ) : ( )} Yeniden Üret )} {/* Onayla ve video üretimini başlat */} {isEditable && hasScript && ( {approveMutation.isPending ? ( ) : ( )} Onayla & Video Üret )} {/* Hata mesajı */} {project.errorMessage && ( Hata {project.errorMessage} )} {/* ── Render Progress (WebSocket) ── */} {isRendering && ( )} {/* ── Video Player (tamamlandıysa) ── */} {isCompleted && project.finalVideoUrl && ( )} {/* ── Sahneler ── */} {hasScript && ( Senaryo — {project.scenes!.length} sahne Toplam: {project.scenes!.reduce((sum, s) => sum + s.duration, 0)}s {project.scenes!.map((scene) => ( ))} )} {/* ── Boş durum (senaryo yok) ── */} {!hasScript && isEditable && ( Henüz senaryo üretilmedi AI'ın projeniz için etkileyici bir video senaryosu oluşturmasını sağlayın. {generateScriptMutation.isPending ? ( ) : ( )} Senaryo Üret )} {/* ── Render Geçmişi ── */} {project.renderJobs && project.renderJobs.length > 0 && ( Render Geçmişi {project.renderJobs.map((job) => ( Deneme #{job.attemptNumber} {job.status} {job.processingTimeMs && ( {(job.processingTimeMs / 1000).toFixed(1)}s )} {new Date(job.createdAt).toLocaleTimeString('tr-TR', { hour: '2-digit', minute: '2-digit' })} ))} )} ); }
Proje yükleniyor...
Bu proje silinmiş veya erişim izniniz yok.
{project.description}
{tweetData.text as string}
{project.errorMessage}
AI'ın projeniz için etkileyici bir video senaryosu oluşturmasını sağlayın.