diff --git a/app/create/script-to-video/page.tsx b/app/create/script-to-video/page.tsx deleted file mode 100644 index 2e38915..0000000 --- a/app/create/script-to-video/page.tsx +++ /dev/null @@ -1,5 +0,0 @@ -import { ScriptToVideo } from '@/components/pages/script-to-video'; - -export default function ScriptToVideoPage() { - return ; -} \ No newline at end of file diff --git a/app/create/video-to-video/page.tsx b/app/create/video-to-video/page.tsx deleted file mode 100644 index 9a8685b..0000000 --- a/app/create/video-to-video/page.tsx +++ /dev/null @@ -1,5 +0,0 @@ -import { VideoToVideo } from '@/components/pages/video-to-video'; - -export default function VideoToVideoPage() { - return ; -} \ No newline at end of file diff --git a/app/history/page.tsx b/app/history/page.tsx index 300dd4d..cf84544 100644 --- a/app/history/page.tsx +++ b/app/history/page.tsx @@ -1,10 +1,11 @@ import { DashboardLayout } from '@/components/layout/dashboard-layout'; -import { HistoryPage } from '@/components/pages/history-page'; export default function History() { return ( - + + History + ); } \ No newline at end of file diff --git a/components/common/EmptyStateAnimation.tsx b/components/common/EmptyStateAnimation.tsx deleted file mode 100644 index 86254fd..0000000 --- a/components/common/EmptyStateAnimation.tsx +++ /dev/null @@ -1,328 +0,0 @@ -"use client"; - -import React, { useState, useEffect, useRef, useCallback } from 'react'; -import { ArrowUp } from 'lucide-react'; -import gsap from 'gsap'; -import { ImageWave } from '@/components/ui/ImageWave'; - -interface AnimationStageProps { - shouldStart: boolean; - onComplete: () => void; -} - -// 动画1:模拟输入和点击 -const InputAnimation: React.FC = ({ shouldStart, onComplete }) => { - const containerRef = useRef(null); - const inputRef = useRef(null); - const cursorRef = useRef(null); - const buttonRef = useRef(null); - const mouseRef = useRef(null); - const [displayText, setDisplayText] = useState(''); - - const demoText = "a cute capybara with an orange on its head"; - - useEffect(() => { - if (!shouldStart || !containerRef.current) return; - - // 重置状态 - setDisplayText(''); - - const tl = gsap.timeline({ - onComplete: () => { - setTimeout(onComplete, 500); - } - }); - - // 1. 显示输入框和鼠标 - tl.fromTo([inputRef.current, mouseRef.current], { - scale: 0.9, - opacity: 0 - }, { - scale: 1, - opacity: 1, - duration: 0.3, - ease: "back.out(1.7)", - stagger: 0.1 - }); - - // 2. 鼠标移动到输入框中心 - tl.to(mouseRef.current, { - x: 20, - y: 0, - duration: 0.3 - }); - - // 3. 输入框聚焦效果 - tl.to(inputRef.current, { - scale: 1.02, - boxShadow: '0 0 0 2px rgba(59, 130, 246, 0.5)', - duration: 0.2 - }); - - // 4. 打字动画 - const typingDuration = demoText.length * 0.05; - tl.to({}, { - duration: typingDuration, - onUpdate: function() { - const progress = this.progress(); - const targetChar = Math.floor(progress * demoText.length); - setDisplayText(demoText.slice(0, targetChar)); - } - }); - - // 6. 鼠标移动到按钮位置(调整移动时间和缓动函数) - tl.to(mouseRef.current, { - x: 650, - y: 0, - duration: 0.8, - ease: "power2.inOut" - }); - - // 7. 等待一小段时间 - tl.to({}, { duration: 0.2 }); - - // 8. 点击效果 - tl.to(mouseRef.current, { - scale: 0.8, - duration: 0.1, - yoyo: true, - repeat: 1 - }).to(buttonRef.current, { - scale: 0.95, - duration: 0.1, - yoyo: true, - repeat: 1 - }, "<"); - - // 9. 等待一小段时间 - tl.to({}, { duration: 0.3 }); - - // 10. 整体淡出 - tl.to(containerRef.current, { - opacity: 0, - y: -20, - duration: 0.3 - }); - - }, [shouldStart, onComplete]); - - return ( - - - - - {displayText} - - - - - - - Action - - - - - - - - - - ); -}; - -// 动画2:ImageWave 动画展示 -const WaveAnimation: React.FC = ({ shouldStart, onComplete }) => { - const containerRef = useRef(null); - const [showWave, setShowWave] = useState(false); - const [autoAnimate, setAutoAnimate] = useState(false); - - const imageUrls = [ - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-1.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-2.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-3.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-4.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-1.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-2.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-3.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-4.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-3.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-2.jpg', - 'https://d3phaj0sisr2ct.cloudfront.net/app/gen4/object-reference/welcome-ref-1.jpg', - ]; - - useEffect(() => { - if (!shouldStart) { - setShowWave(false); - setAutoAnimate(false); - return; - } - - // 显示 ImageWave - setShowWave(true); - - // 延迟开始自动动画 - const startTimeout = setTimeout(() => { - setAutoAnimate(true); - }, 300); - - return () => { - clearTimeout(startTimeout); - }; - }, [shouldStart]); - - const handleAnimationComplete = () => { - // 动画完成后淡出并触发完成回调 - gsap.to(containerRef.current, { - opacity: 0, - scale: 0.9, - duration: 0.3, - onComplete: () => { - setAutoAnimate(false); - setShowWave(false); - onComplete(); - } - }); - }; - - return ( - - - - ); -}; - -// 动画3:图片墙打破,显示视频 -const FinalAnimation: React.FC = ({ shouldStart, onComplete }) => { - const containerRef = useRef(null); - const videoRef = useRef(null); - const [showVideo, setShowVideo] = useState(false); - - const videoUrl = 'https://cdn.qikongjian.com/videos/1750385931_99a8fb42-af89-4ae9-841a-a49869f026bd_text_to_video_0.mp4'; - - useEffect(() => { - if (!shouldStart || !containerRef.current) return; - - const tl = gsap.timeline({ - onComplete: () => { - setTimeout(() => { - // 淡出视频 - gsap.to(containerRef.current, { - opacity: 0, - scale: 0.9, - duration: 0.3, - onComplete - }); - }, 3000); - } - }); - - // 显示容器 - tl.fromTo(containerRef.current, - { opacity: 0, scale: 0.9 }, - { opacity: 1, scale: 1, duration: 0.3 } - ); - - // 显示视频 - setShowVideo(true); - - }, [shouldStart, onComplete]); - - return ( - - {showVideo && ( - - )} - - ); -}; - -// 主组件 -export const EmptyStateAnimation = ({ className }: { className: string }) => { - const [currentStage, setCurrentStage] = useState<'input' | 'wave' | 'final'>('input'); - const [animationCycle, setAnimationCycle] = useState(0); - const [isReady, setIsReady] = useState(true); - - const handleStageComplete = useCallback(() => { - // 先将当前阶段标记为不可执行 - setIsReady(false); - - // 延迟切换到下一个阶段 - setTimeout(() => { - switch (currentStage) { - case 'input': - setCurrentStage('wave'); - break; - case 'wave': - setCurrentStage('final'); - break; - case 'final': - setAnimationCycle(prev => prev + 1); - setCurrentStage('input'); - break; - } - - // 给下一个阶段一些准备时间 - setTimeout(() => { - setIsReady(true); - }, 100); - }, 300); - }, [currentStage]); - - return ( - - - - - - ); -}; \ No newline at end of file diff --git a/components/pages/history-page.tsx b/components/pages/history-page.tsx deleted file mode 100644 index c590883..0000000 --- a/components/pages/history-page.tsx +++ /dev/null @@ -1,224 +0,0 @@ -"use client"; - -import { useState, useEffect } from 'react'; -import { Card } from '@/components/ui/card'; -import { Button } from '@/components/ui/button'; -import { Badge } from '@/components/ui/badge'; -import { Input } from '@/components/ui/input'; -import { - Search, - Filter, - Video, - Calendar, - Eye, - Edit, - Trash2, - MoreHorizontal, - Play -} from 'lucide-react'; -import { - DropdownMenu, - DropdownMenuContent, - DropdownMenuItem, - DropdownMenuTrigger -} from '@/components/ui/dropdown-menu'; - -interface HistoryItem { - id: number; - title: string; - type: 'script-to-video' | 'video-to-video'; - status: 'completed' | 'processing' | 'failed'; - createdAt: string; - duration?: string; - thumbnail?: string; -} - -export function HistoryPage() { - const [searchTerm, setSearchTerm] = useState(''); - const [historyItems, setHistoryItems] = useState([]); - const [filteredItems, setFilteredItems] = useState([]); - - // Mock data - replace with actual API call - useEffect(() => { - const mockData: HistoryItem[] = [ - { - id: 1, - title: "Sample Script Video", - type: "script-to-video", - status: "completed", - createdAt: "2024-01-15", - duration: "2:30", - thumbnail: "/assets/empty_video.png" - }, - { - id: 2, - title: "Video Enhancement Project", - type: "video-to-video", - status: "processing", - createdAt: "2024-01-14", - duration: "1:45" - }, - { - id: 3, - title: "Marketing Video", - type: "script-to-video", - status: "completed", - createdAt: "2024-01-13", - duration: "3:15", - thumbnail: "/assets/empty_video.png" - } - ]; - setHistoryItems(mockData); - setFilteredItems(mockData); - }, []); - - // Filter items based on search term - useEffect(() => { - const filtered = historyItems.filter(item => - item.title.toLowerCase().includes(searchTerm.toLowerCase()) - ); - setFilteredItems(filtered); - }, [searchTerm, historyItems]); - - const getStatusBadgeVariant = (status: string) => { - switch (status) { - case 'completed': - return 'default'; - case 'processing': - return 'secondary'; - case 'failed': - return 'destructive'; - default: - return 'default'; - } - }; - - const getTypeIcon = (type: string) => { - return type === 'script-to-video' ? '📝' : '🎥'; - }; - - return ( - - {/* Header */} - - - Project History - View and manage your video projects - - - {/* Search and Filter */} - - - - setSearchTerm(e.target.value)} - className="pl-10 bg-white/5 border-white/20 text-white placeholder:text-white/40" - /> - - - - Filter - - - - - {/* Project Grid */} - {filteredItems.length === 0 ? ( - - - - {searchTerm ? 'No projects found' : 'No projects yet'} - - - {searchTerm - ? 'Try adjusting your search terms' - : 'Create your first video project to see it here' - } - - - ) : ( - - {filteredItems.map((item) => ( - - {/* Thumbnail */} - - {item.thumbnail ? ( - - ) : ( - - - - )} - - {/* Play overlay */} - - - - - - - {/* Type indicator */} - - {getTypeIcon(item.type)} - - - {/* Duration */} - {item.duration && ( - - {item.duration} - - )} - - - {/* Content */} - - - - {item.title} - - - - - - - - - - - View - - - - Edit - - - - Delete - - - - - - - - - {item.createdAt} - - - {item.status} - - - - - ))} - - )} - - ); -} \ No newline at end of file diff --git a/components/pages/home-page2.tsx b/components/pages/home-page2.tsx index 77b8644..9a55abf 100644 --- a/components/pages/home-page2.tsx +++ b/components/pages/home-page2.tsx @@ -88,8 +88,8 @@ export function HomePage2() { const projectData: CreateScriptProjectRequest = { title: "script default", // 默认剧本名称 project_type: projectType, - mode: ModeEnum.MANUAL, - resolution: ResolutionEnum.FULL_HD_1080P + mode: ModeEnum.MANUAL === 'manual' ? 1 : 2, // 1 表示手动模式,2 表示自动模式 + resolution: 1080 // 1080p 分辨率 }; const projectResponse = await createScriptProject(projectData); @@ -121,7 +121,7 @@ export function HomePage2() { {/* 工具栏-列表形式切换 */} { + onClick={(e: React.MouseEvent) => { e.stopPropagation(); handleToolChange(activeTool === "stretch" ? "right" : "left"); }} diff --git a/components/pages/script-to-video.tsx b/components/pages/script-to-video.tsx deleted file mode 100644 index bf8f914..0000000 --- a/components/pages/script-to-video.tsx +++ /dev/null @@ -1,168 +0,0 @@ -"use client"; - -import { useState } from 'react'; -import { Button } from '@/components/ui/button'; -import { Card } from '@/components/ui/card'; -import { Textarea } from '@/components/ui/textarea'; -import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; -import { ArrowLeft, Loader2 } from 'lucide-react'; -import { useRouter } from 'next/navigation'; -import { ModeEnum, ResolutionEnum } from "@/app/model/enums"; -import { createScriptEpisode, CreateScriptEpisodeRequest, updateScriptEpisode, UpdateScriptEpisodeRequest } from "@/api/script_episode"; -import { convertScriptToScene } from "@/api/video_flow"; - -export function ScriptToVideo() { - const router = useRouter(); - const [script, setScript] = useState(''); - const [selectedMode, setSelectedMode] = useState(ModeEnum.AUTOMATIC); - const [selectedResolution, setSelectedResolution] = useState(ResolutionEnum.HD_720P); - const [isCreating, setIsCreating] = useState(false); - - const handleBack = () => { - router.push('/create'); - }; - - const handleCreate = async () => { - if (!script.trim()) { - alert('请输入剧本内容'); - return; - } - - try { - setIsCreating(true); - - // Create episode - const episodeData: CreateScriptEpisodeRequest = { - title: "Script Episode", - script_id: 0, // This should come from a project - status: 1, - summary: script - }; - - const episodeResponse = await createScriptEpisode(episodeData); - if (episodeResponse.code !== 0) { - alert(`创建剧集失败: ${episodeResponse.message}`); - return; - } - - const episodeId = episodeResponse.data.id; - - // Convert script to scenes - const convertResponse = await convertScriptToScene(script, episodeId, 0); - - if (convertResponse.code === 0) { - // Update episode with generated data - const updateEpisodeData: UpdateScriptEpisodeRequest = { - id: episodeId, - atmosphere: convertResponse.data.atmosphere, - summary: convertResponse.data.summary, - scene: convertResponse.data.scene, - characters: convertResponse.data.characters, - }; - - await updateScriptEpisode(updateEpisodeData); - - // Navigate to workflow - router.push(`/create/work-flow?episodeId=${episodeId}`); - } else { - alert(`转换失败: ${convertResponse.message}`); - } - } catch (error) { - console.error('创建过程出错:', error); - alert("创建项目时发生错误,请稍后重试"); - } finally { - setIsCreating(false); - } - }; - - return ( - - - {/* Header */} - - - - - Script to Video - - - {/* Main Content */} - - - {/* Script Input */} - - - Script Content - - setScript(e.target.value)} - placeholder="Enter your script here..." - className="min-h-[200px] bg-white/5 border-white/20 text-white placeholder:text-white/50" - rows={10} - /> - - - {/* Settings */} - - - - Mode - - setSelectedMode(Number(value) as ModeEnum)}> - - - - - Automatic - Manual - - - - - - - Resolution - - setSelectedResolution(Number(value) as ResolutionEnum)}> - - - - - 720P HD - 1080P Full HD - 2K UHD - 4K UHD - - - - - - {/* Create Button */} - - - {isCreating ? ( - <> - - Creating... - > - ) : ( - 'Create Video' - )} - - - - - - - ); -} \ No newline at end of file diff --git a/components/pages/video-to-video.tsx b/components/pages/video-to-video.tsx deleted file mode 100644 index ee9a341..0000000 --- a/components/pages/video-to-video.tsx +++ /dev/null @@ -1,233 +0,0 @@ -"use client"; - -import { useState } from 'react'; -import { Button } from '@/components/ui/button'; -import { Card } from '@/components/ui/card'; -import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"; -import { ArrowLeft, Upload, Loader2 } from 'lucide-react'; -import { useRouter } from 'next/navigation'; -import { ModeEnum, ResolutionEnum } from "@/app/model/enums"; -import { createScriptEpisode, CreateScriptEpisodeRequest, updateScriptEpisode, UpdateScriptEpisodeRequest } from "@/api/script_episode"; -import { convertVideoToScene } from "@/api/video_flow"; -import { getUploadToken, uploadToQiniu } from "@/api/common"; - -export function VideoToVideo() { - const router = useRouter(); - const [videoUrl, setVideoUrl] = useState(''); - const [selectedMode, setSelectedMode] = useState(ModeEnum.AUTOMATIC); - const [selectedResolution, setSelectedResolution] = useState(ResolutionEnum.HD_720P); - const [isUploading, setIsUploading] = useState(false); - const [isCreating, setIsCreating] = useState(false); - - const handleBack = () => { - router.push('/create'); - }; - - const handleUploadVideo = async () => { - const input = document.createElement('input'); - input.type = 'file'; - input.accept = 'video/*'; - input.onchange = async (e) => { - const file = (e.target as HTMLInputElement).files?.[0]; - if (file) { - try { - setIsUploading(true); - - // Get upload token - const { token } = await getUploadToken(); - - // Upload to Qiniu - const uploadedVideoUrl = await uploadToQiniu(file, token); - - setVideoUrl(uploadedVideoUrl); - } catch (error) { - console.error('Upload error:', error); - alert('Upload failed, please try again'); - } finally { - setIsUploading(false); - } - } - }; - input.click(); - }; - - const handleCreate = async () => { - if (!videoUrl) { - alert('请先上传视频'); - return; - } - - try { - setIsCreating(true); - - // Create episode - const episodeData: CreateScriptEpisodeRequest = { - title: "Video Episode", - script_id: 0, // This should come from a project - status: 1, - summary: "Video conversion" - }; - - const episodeResponse = await createScriptEpisode(episodeData); - if (episodeResponse.code !== 0) { - alert(`创建剧集失败: ${episodeResponse.message}`); - return; - } - - const episodeId = episodeResponse.data.id; - - // Convert video to scenes - const convertResponse = await convertVideoToScene(videoUrl, episodeId, 0); - - if (convertResponse.code === 0) { - // Update episode with generated data - const updateEpisodeData: UpdateScriptEpisodeRequest = { - id: episodeId, - atmosphere: convertResponse.data.atmosphere, - summary: convertResponse.data.summary, - scene: convertResponse.data.scene, - characters: convertResponse.data.characters, - }; - - await updateScriptEpisode(updateEpisodeData); - - // Navigate to workflow - router.push(`/create/work-flow?episodeId=${episodeId}`); - } else { - alert(`转换失败: ${convertResponse.message}`); - } - } catch (error) { - console.error('创建过程出错:', error); - alert("创建项目时发生错误,请稍后重试"); - } finally { - setIsCreating(false); - } - }; - - return ( - - - {/* Header */} - - - - - Video to Video - - - {/* Main Content */} - - - {/* Video Upload */} - - - Upload Video - - - {videoUrl ? ( - - - - Replace Video - - - ) : ( - - - - Click to upload a video file - - {isUploading ? ( - <> - - Uploading... - > - ) : ( - <> - - Choose Video - > - )} - - - - )} - - - - {/* Settings */} - - - - Mode - - setSelectedMode(Number(value) as ModeEnum)}> - - - - - Automatic - Manual - - - - - - - Resolution - - setSelectedResolution(Number(value) as ResolutionEnum)}> - - - - - 720P HD - 1080P Full HD - 2K UHD - 4K UHD - - - - - - {/* Create Button */} - - - {isCreating ? ( - <> - - Creating... - > - ) : ( - 'Create Video' - )} - - - - - - - ); -} \ No newline at end of file diff --git a/components/pages/work-flow.tsx b/components/pages/work-flow.tsx index 8bcf00b..10293ba 100644 --- a/components/pages/work-flow.tsx +++ b/components/pages/work-flow.tsx @@ -8,7 +8,7 @@ import { ErrorBoundary } from "@/components/ui/error-boundary"; import { TaskInfo } from "./work-flow/task-info"; import { MediaViewer } from "./work-flow/media-viewer"; import { ThumbnailGrid } from "./work-flow/thumbnail-grid"; -import { useWorkflowData } from "./work-flow/use-workflow-data.ts"; +import { useWorkflowData } from "./work-flow/use-workflow-data"; import { usePlaybackControls } from "./work-flow/use-playback-controls"; import { AlertCircle, RefreshCw } from "lucide-react"; import { motion } from "framer-motion"; diff --git a/components/pages/work-flow/use-api-data.ts b/components/pages/work-flow/use-api-data.ts index 0d48c5c..f726e56 100644 --- a/components/pages/work-flow/use-api-data.ts +++ b/components/pages/work-flow/use-api-data.ts @@ -1,6 +1,7 @@ import { useState, useEffect, useCallback } from 'react'; import { detailScriptEpisodeNew, getScriptTitle, getRunningStreamData, StreamData } from '@/api/video_flow'; import { useSearchParams } from 'next/navigation'; +import { ApiResponse } from '@/api/common'; // 步骤映射 const STEP_MAP = { @@ -56,8 +57,8 @@ export const useApiData = () => { const [streamInterval, setStreamInterval] = useState(null); // 处理流式数据 - const handleStreamData = useCallback((streamData: StreamData) => { - const { category, message, data, status } = streamData; + const handleStreamData = useCallback((streamData: ApiResponse) => { + const { category, message, data, status } = streamData.data; // 更新加载文本 setCurrentLoadingText(message); @@ -114,7 +115,7 @@ export const useApiData = () => { }); // 如果状态为 completed,停止获取流式数据 - if (status === 'completed' || streamData.all_completed) { + if (status === 'completed' || streamData.data.all_completed) { setNeedStreamData(false); } }, []); @@ -124,7 +125,7 @@ export const useApiData = () => { if (!episodeId || !needStreamData) return; try { - const streamData = await getRunningStreamData({ episodeId }); + const streamData = await getRunningStreamData({ project_id: episodeId }); handleStreamData(streamData); } catch (error) { console.error('获取流式数据失败:', error); diff --git a/components/pages/work-flow/use-workflow-data.ts b/components/pages/work-flow/use-workflow-data.ts deleted file mode 100644 index 4b566f1..0000000 --- a/components/pages/work-flow/use-workflow-data.ts +++ /dev/null @@ -1,406 +0,0 @@ -'use client'; - -import { useState, useEffect } from 'react'; -import { useSearchParams } from 'next/navigation'; -import { detailScriptEpisodeNew, getScriptTitle, getRunningStreamData } from '@/api/video_flow'; - -// 步骤映射 -const STEP_MAP = { - 'initializing': '0', - 'sketch': '1', - 'character': '2', - 'video': '3', - 'music': '4', - 'final_video': '6' -} as const; -// 执行loading文字映射 -const LOADING_TEXT_MAP = { - initializing: 'initializing...', - sketch: (count: number, total: number) => `Generating sketch ${count + 1 > total ? total : count + 1}/${total}...`, - sketchComplete: 'Sketch generation complete', - character: 'Drawing characters...', - newCharacter: (count: number, total: number) => `Drawing character ${count + 1 > total ? total : count + 1}/${total}...`, - video: (count: number, total: number) => `Generating video ${count + 1 > total ? total : count + 1}/${total}...`, - videoComplete: 'Video generation complete', - audio: 'Generating background audio...', - postProduction: (step: string) => `Post-production: ${step}...`, - final: 'Generating final product...', - complete: 'Task completed' -} as const; - -type ApiStep = keyof typeof STEP_MAP; - -// 添加 TaskObject 接口 -interface TaskObject { - taskStatus: string; - title: string; - currentLoadingText: string; - sketchCount?: number; - totalSketchCount?: number; - isGeneratingSketch?: boolean; - isGeneratingVideo?: boolean; - roles?: any[]; - music?: any[]; - final?: any; -} - -export function useWorkflowData() { - const searchParams = useSearchParams(); - const episodeId = searchParams.get('episodeId'); - - // 更新 taskObject 的类型 - const [taskObject, setTaskObject] = useState(null); - const [taskSketch, setTaskSketch] = useState([]); - const [taskVideos, setTaskVideos] = useState([]); - const [sketchCount, setSketchCount] = useState(0); - const [isLoading, setIsLoading] = useState(true); - const [currentStep, setCurrentStep] = useState('0'); - const [currentSketchIndex, setCurrentSketchIndex] = useState(0); - const [isGeneratingSketch, setIsGeneratingSketch] = useState(false); - const [isGeneratingVideo, setIsGeneratingVideo] = useState(false); - const [currentLoadingText, setCurrentLoadingText] = useState('正在加载项目数据...'); - const [totalSketchCount, setTotalSketchCount] = useState(0); - const [roles, setRoles] = useState([]); - const [music, setMusic] = useState([]); - const [final, setFinal] = useState(null); - const [dataLoadError, setDataLoadError] = useState(null); - const [needStreamData, setNeedStreamData] = useState(false); - - // 获取流式数据 - const fetchStreamData = async () => { - if (!episodeId || !needStreamData) return; - - try { - const response = await getRunningStreamData({ project_id: episodeId }); - if (!response.successful) { - throw new Error(response.message); - } - - let loadingText: any = LOADING_TEXT_MAP.initializing; - let finalStep = '1', sketchCount = 0; - const all_task_data = response.data; - // all_task_data 下标0 和 下标1 换位置 - const temp = all_task_data[0]; - all_task_data[0] = all_task_data[1]; - all_task_data[1] = temp; - - console.log('all_task_data', all_task_data); - for (const task of all_task_data) { - - // 如果有已完成的数据,同步到状态 - if (task.task_name === 'generate_sketch' && task.task_result) { - if (task.task_result.data.length >= 0 && taskSketch.length !== task.task_result.data.length) { - // 正在生成草图中 替换 sketch 数据 - const sketchList = []; - for (const sketch of task.task_result.data) { - sketchList.push({ - url: sketch.image_path, - script: sketch.sketch_name - }); - } - setTaskSketch(sketchList); - setSketchCount(sketchList.length); - setIsGeneratingSketch(true); - setCurrentSketchIndex(sketchList.length - 1); - loadingText = LOADING_TEXT_MAP.sketch(sketchList.length, task.task_result.total_count); - } - if (task.task_status === 'COMPLETED') { - // 草图生成完成 - setIsGeneratingSketch(false); - sketchCount = task.task_result.total_count; - console.log('----------草图生成完成', sketchCount); - loadingText = LOADING_TEXT_MAP.sketchComplete; - finalStep = '2'; - } - setTotalSketchCount(task.task_result.total_count); - } - if (task.task_name === 'generate_character' && task.task_result) { - if (task.task_result.data.length >= 0 && roles.length !== task.task_result.data.length) { - // 正在生成角色中 替换角色数据 - const characterList = []; - for (const character of task.task_result.data) { - characterList.push({ - name: character.character_name, - url: character.image_path, - sound: null, - soundDescription: '', - roleDescription: '' - }); - } - setRoles(characterList); - loadingText = LOADING_TEXT_MAP.newCharacter(characterList.length, task.task_result.total_count); - } - if (task.task_status === 'COMPLETED') { - console.log('----------角色生成完成,有几个分镜', sketchCount); - // 角色生成完成 - finalStep = '3'; - - loadingText = LOADING_TEXT_MAP.video(0, sketchCount); - } - } - if (task.task_name === 'generate_videos' && task.task_result) { - if (task.task_result.data.length >= 0 && taskVideos.length !== task.task_result.data.length) { - // 正在生成视频中 替换视频数据 - const videoList = []; - for (const video of task.task_result.data) { - // 每一项 video 有多个视频 先默认取第一个 - videoList.push({ - url: video[0].qiniuVideoUrl, - script: video[0].operation.metadata.video.prompt, - audio: null, - }); - } - setTaskVideos(videoList); - setIsGeneratingVideo(true); - setCurrentSketchIndex(videoList.length - 1); - loadingText = LOADING_TEXT_MAP.video(videoList.length, task.task_result.total_count); - } - if (task.task_status === 'COMPLETED') { - console.log('----------视频生成完成'); - // 视频生成完成 - setIsGeneratingVideo(false); - finalStep = '4'; - - // 暂时没有音频生成 直接跳过 - finalStep = '5'; - loadingText = LOADING_TEXT_MAP.postProduction('post-production...'); - } - } - if (task.task_name === 'generate_final_video') { - if (task.task_result && task.task_result.video) { - setFinal({ - url: task.task_result.video, - }) - finalStep = '6'; - loadingText = LOADING_TEXT_MAP.complete; - - // 停止轮询 - setNeedStreamData(false); - } - } - } - - console.log('----------finalStep', finalStep); - // 设置步骤 - setCurrentStep(finalStep); - setTaskObject(prev => { - if (!prev) return null; - return { - ...prev, - taskStatus: finalStep - }; - }); - setCurrentLoadingText(loadingText); - - } catch (error) { - console.error('获取数据失败:', error); - } - }; - - // 轮询获取流式数据 - useEffect(() => { - let interval: NodeJS.Timeout; - - if (needStreamData) { - interval = setInterval(fetchStreamData, 10000); - fetchStreamData(); // 立即执行一次 - } - - return () => { - if (interval) { - clearInterval(interval); - } - }; - }, [needStreamData]); - - // 初始化数据 - const initializeWorkflow = async () => { - if (!episodeId) { - setDataLoadError('缺少必要的参数'); - setIsLoading(false); - return; - } - - try { - setIsLoading(true); - setCurrentLoadingText('正在加载项目数据...'); - - // 获取剧集详情 - const response = await detailScriptEpisodeNew({ project_id: episodeId }); - if (!response.successful) { - throw new Error(response.message); - } - - const { name, status, data } = response.data; - setIsLoading(false); - - // 设置初始数据 - setTaskObject({ - taskStatus: '0', - title: name || 'generating...', - currentLoadingText: status === 'COMPLETED' ? LOADING_TEXT_MAP.complete : LOADING_TEXT_MAP.initializing - }); - - // 设置标题 - if (!name) { - // 如果没有标题,轮询获取 - const titleResponse = await getScriptTitle({ project_id: episodeId }); - console.log('titleResponse', titleResponse); - if (titleResponse.successful) { - setTaskObject((prev: TaskObject | null) => ({ - ...(prev || {}), - title: titleResponse.data.name - } as TaskObject)); - } - } - - let loadingText: any = LOADING_TEXT_MAP.initializing; - if (status === 'COMPLETED') { - loadingText = LOADING_TEXT_MAP.complete; - } - - // 如果有已完成的数据,同步到状态 - let finalStep = '1'; - if (data) { - if (data.sketch && data.sketch.data && data.sketch.data.length > 0) { - const sketchList = []; - for (const sketch of data.sketch.data) { - sketchList.push({ - url: sketch.image_path, - script: sketch.sketch_name, - }); - } - setTaskSketch(sketchList); - setSketchCount(sketchList.length); - setTotalSketchCount(data.sketch.total_count); - // 设置为最后一个草图 - if (data.sketch.total_count > data.sketch.data.length) { - setIsGeneratingSketch(true); - setCurrentSketchIndex(data.sketch.data.length - 1); - loadingText = LOADING_TEXT_MAP.sketch(data.sketch.data.length, data.sketch.total_count); - } else { - finalStep = '2'; - if (!data.character || !data.character.data || !data.character.data.length) { - loadingText = LOADING_TEXT_MAP.newCharacter(0, data.character.total_count); - } - } - } - if (data.character && data.character.data && data.character.data.length > 0) { - const characterList = []; - for (const character of data.character.data) { - characterList.push({ - name: character.character_name, - url: character.image_path, - sound: null, - soundDescription: '', - roleDescription: '' - }); - } - setRoles(characterList); - if (data.character.total_count > data.character.data.length) { - loadingText = LOADING_TEXT_MAP.newCharacter(data.character.data.length, data.character.total_count); - } else { - finalStep = '3'; - if (!data.video || !data.video.data || !data.video.data.length) { - loadingText = LOADING_TEXT_MAP.video(0, data.video.total_count || data.sketch.total_count); - } - } - } - if (data.video && data.video.data && data.video.data.length > 0) { - const videoList = []; - for (const video of data.video.data) { - // 每一项 video 有多个视频 先默认取第一个 - videoList.push({ - url: video[0].qiniuVideoUrl, - script: video[0].operation.metadata.video.prompt, - audio: null, - }); - } - setTaskVideos(videoList); - // 如果在视频步骤,设置为最后一个视频 - if (data.video.total_count > data.video.data.length) { - setIsGeneratingVideo(true); - setCurrentSketchIndex(data.video.data.length - 1); - loadingText = LOADING_TEXT_MAP.video(data.video.data.length, data.video.total_count); - } else { - finalStep = '4'; - loadingText = LOADING_TEXT_MAP.audio; - - // 暂时没有音频生成 直接跳过 - finalStep = '5'; - loadingText = LOADING_TEXT_MAP.postProduction('post-production...'); - } - } - if (data.final_video && data.final_video.video) { - setFinal({ - url: data.final_video.video - }); - finalStep = '6'; - loadingText = LOADING_TEXT_MAP.complete; - } - } - - // 设置步骤 - setCurrentStep(finalStep); - setTaskObject(prev => { - if (!prev) return null; - return { - ...prev, - taskStatus: finalStep - }; - }); - console.log('---------loadingText', loadingText); - setCurrentLoadingText(loadingText); - - // 设置是否需要获取流式数据 - setNeedStreamData(status !== 'COMPLETED' && finalStep !== '6'); - - } catch (error) { - console.error('初始化失败:', error); - setDataLoadError('加载失败,请重试'); - setIsLoading(false); - } - }; - - // 重试加载数据 - const retryLoadData = () => { - setDataLoadError(null); - // 重置所有状态 - setTaskSketch([]); - setTaskVideos([]); - setSketchCount(0); - setTotalSketchCount(0); - setRoles([]); - setMusic([]); - setFinal(null); - setCurrentSketchIndex(0); - setCurrentStep('0'); - // 重新初始化 - initializeWorkflow(); - }; - - // 初始化 - useEffect(() => { - initializeWorkflow(); - }, [episodeId]); - - return { - taskObject, - taskSketch, - taskVideos, - sketchCount, - isLoading, - currentStep, - currentSketchIndex, - isGeneratingSketch, - isGeneratingVideo, - currentLoadingText, - totalSketchCount, - roles, - music, - final, - dataLoadError, - setCurrentSketchIndex, - retryLoadData, - }; -} \ No newline at end of file diff --git a/components/pages/work-flow/use-workflow-data.tsx b/components/pages/work-flow/use-workflow-data.tsx index 5610a08..4b566f1 100644 --- a/components/pages/work-flow/use-workflow-data.tsx +++ b/components/pages/work-flow/use-workflow-data.tsx @@ -1,15 +1,56 @@ 'use client'; -import { useState, useEffect, useRef, useCallback } from 'react'; -import { getRandomMockData, STEP_MESSAGES, MOCK_DELAY_TIME, MOCK_DATA } from '@/components/work-flow/constants'; +import { useState, useEffect } from 'react'; +import { useSearchParams } from 'next/navigation'; +import { detailScriptEpisodeNew, getScriptTitle, getRunningStreamData } from '@/api/video_flow'; -// 当前选择的mock数据 -let selectedMockData: any = null; +// 步骤映射 +const STEP_MAP = { + 'initializing': '0', + 'sketch': '1', + 'character': '2', + 'video': '3', + 'music': '4', + 'final_video': '6' +} as const; +// 执行loading文字映射 +const LOADING_TEXT_MAP = { + initializing: 'initializing...', + sketch: (count: number, total: number) => `Generating sketch ${count + 1 > total ? total : count + 1}/${total}...`, + sketchComplete: 'Sketch generation complete', + character: 'Drawing characters...', + newCharacter: (count: number, total: number) => `Drawing character ${count + 1 > total ? total : count + 1}/${total}...`, + video: (count: number, total: number) => `Generating video ${count + 1 > total ? total : count + 1}/${total}...`, + videoComplete: 'Video generation complete', + audio: 'Generating background audio...', + postProduction: (step: string) => `Post-production: ${step}...`, + final: 'Generating final product...', + complete: 'Task completed' +} as const; + +type ApiStep = keyof typeof STEP_MAP; + +// 添加 TaskObject 接口 +interface TaskObject { + taskStatus: string; + title: string; + currentLoadingText: string; + sketchCount?: number; + totalSketchCount?: number; + isGeneratingSketch?: boolean; + isGeneratingVideo?: boolean; + roles?: any[]; + music?: any[]; + final?: any; +} export function useWorkflowData() { - const [taskObject, setTaskObject] = useState(null); + const searchParams = useSearchParams(); + const episodeId = searchParams.get('episodeId'); + + // 更新 taskObject 的类型 + const [taskObject, setTaskObject] = useState(null); const [taskSketch, setTaskSketch] = useState([]); - const [taskRoles, setTaskRoles] = useState([]); const [taskVideos, setTaskVideos] = useState([]); const [sketchCount, setSketchCount] = useState(0); const [isLoading, setIsLoading] = useState(true); @@ -18,321 +59,347 @@ export function useWorkflowData() { const [isGeneratingSketch, setIsGeneratingSketch] = useState(false); const [isGeneratingVideo, setIsGeneratingVideo] = useState(false); const [currentLoadingText, setCurrentLoadingText] = useState('正在加载项目数据...'); + const [totalSketchCount, setTotalSketchCount] = useState(0); + const [roles, setRoles] = useState([]); + const [music, setMusic] = useState([]); + const [final, setFinal] = useState(null); const [dataLoadError, setDataLoadError] = useState(null); - const [isLoadingData, setIsLoadingData] = useState(false); + const [needStreamData, setNeedStreamData] = useState(false); + + // 获取流式数据 + const fetchStreamData = async () => { + if (!episodeId || !needStreamData) return; - // 异步加载数据 - 改进错误处理和fallback机制 - const loadMockData = async () => { try { - setIsLoadingData(true); - setDataLoadError(null); - setCurrentLoadingText('正在从服务器获取项目数据...'); - - // 尝试从接口获取数据 - selectedMockData = await getRandomMockData(); - - console.log('成功从接口获取数据:', selectedMockData); - setCurrentLoadingText('项目数据加载完成'); - + const response = await getRunningStreamData({ project_id: episodeId }); + if (!response.successful) { + throw new Error(response.message); + } + + let loadingText: any = LOADING_TEXT_MAP.initializing; + let finalStep = '1', sketchCount = 0; + const all_task_data = response.data; + // all_task_data 下标0 和 下标1 换位置 + const temp = all_task_data[0]; + all_task_data[0] = all_task_data[1]; + all_task_data[1] = temp; + + console.log('all_task_data', all_task_data); + for (const task of all_task_data) { + + // 如果有已完成的数据,同步到状态 + if (task.task_name === 'generate_sketch' && task.task_result) { + if (task.task_result.data.length >= 0 && taskSketch.length !== task.task_result.data.length) { + // 正在生成草图中 替换 sketch 数据 + const sketchList = []; + for (const sketch of task.task_result.data) { + sketchList.push({ + url: sketch.image_path, + script: sketch.sketch_name + }); + } + setTaskSketch(sketchList); + setSketchCount(sketchList.length); + setIsGeneratingSketch(true); + setCurrentSketchIndex(sketchList.length - 1); + loadingText = LOADING_TEXT_MAP.sketch(sketchList.length, task.task_result.total_count); + } + if (task.task_status === 'COMPLETED') { + // 草图生成完成 + setIsGeneratingSketch(false); + sketchCount = task.task_result.total_count; + console.log('----------草图生成完成', sketchCount); + loadingText = LOADING_TEXT_MAP.sketchComplete; + finalStep = '2'; + } + setTotalSketchCount(task.task_result.total_count); + } + if (task.task_name === 'generate_character' && task.task_result) { + if (task.task_result.data.length >= 0 && roles.length !== task.task_result.data.length) { + // 正在生成角色中 替换角色数据 + const characterList = []; + for (const character of task.task_result.data) { + characterList.push({ + name: character.character_name, + url: character.image_path, + sound: null, + soundDescription: '', + roleDescription: '' + }); + } + setRoles(characterList); + loadingText = LOADING_TEXT_MAP.newCharacter(characterList.length, task.task_result.total_count); + } + if (task.task_status === 'COMPLETED') { + console.log('----------角色生成完成,有几个分镜', sketchCount); + // 角色生成完成 + finalStep = '3'; + + loadingText = LOADING_TEXT_MAP.video(0, sketchCount); + } + } + if (task.task_name === 'generate_videos' && task.task_result) { + if (task.task_result.data.length >= 0 && taskVideos.length !== task.task_result.data.length) { + // 正在生成视频中 替换视频数据 + const videoList = []; + for (const video of task.task_result.data) { + // 每一项 video 有多个视频 先默认取第一个 + videoList.push({ + url: video[0].qiniuVideoUrl, + script: video[0].operation.metadata.video.prompt, + audio: null, + }); + } + setTaskVideos(videoList); + setIsGeneratingVideo(true); + setCurrentSketchIndex(videoList.length - 1); + loadingText = LOADING_TEXT_MAP.video(videoList.length, task.task_result.total_count); + } + if (task.task_status === 'COMPLETED') { + console.log('----------视频生成完成'); + // 视频生成完成 + setIsGeneratingVideo(false); + finalStep = '4'; + + // 暂时没有音频生成 直接跳过 + finalStep = '5'; + loadingText = LOADING_TEXT_MAP.postProduction('post-production...'); + } + } + if (task.task_name === 'generate_final_video') { + if (task.task_result && task.task_result.video) { + setFinal({ + url: task.task_result.video, + }) + finalStep = '6'; + loadingText = LOADING_TEXT_MAP.complete; + + // 停止轮询 + setNeedStreamData(false); + } + } + } + + console.log('----------finalStep', finalStep); + // 设置步骤 + setCurrentStep(finalStep); + setTaskObject(prev => { + if (!prev) return null; + return { + ...prev, + taskStatus: finalStep + }; + }); + setCurrentLoadingText(loadingText); + } catch (error) { - // 报错 - } finally { - setIsLoadingData(false); + console.error('获取数据失败:', error); + } + }; + + // 轮询获取流式数据 + useEffect(() => { + let interval: NodeJS.Timeout; + + if (needStreamData) { + interval = setInterval(fetchStreamData, 10000); + fetchStreamData(); // 立即执行一次 + } + + return () => { + if (interval) { + clearInterval(interval); + } + }; + }, [needStreamData]); + + // 初始化数据 + const initializeWorkflow = async () => { + if (!episodeId) { + setDataLoadError('缺少必要的参数'); + setIsLoading(false); + return; + } + + try { + setIsLoading(true); + setCurrentLoadingText('正在加载项目数据...'); + + // 获取剧集详情 + const response = await detailScriptEpisodeNew({ project_id: episodeId }); + if (!response.successful) { + throw new Error(response.message); + } + + const { name, status, data } = response.data; + setIsLoading(false); + + // 设置初始数据 + setTaskObject({ + taskStatus: '0', + title: name || 'generating...', + currentLoadingText: status === 'COMPLETED' ? LOADING_TEXT_MAP.complete : LOADING_TEXT_MAP.initializing + }); + + // 设置标题 + if (!name) { + // 如果没有标题,轮询获取 + const titleResponse = await getScriptTitle({ project_id: episodeId }); + console.log('titleResponse', titleResponse); + if (titleResponse.successful) { + setTaskObject((prev: TaskObject | null) => ({ + ...(prev || {}), + title: titleResponse.data.name + } as TaskObject)); + } + } + + let loadingText: any = LOADING_TEXT_MAP.initializing; + if (status === 'COMPLETED') { + loadingText = LOADING_TEXT_MAP.complete; + } + + // 如果有已完成的数据,同步到状态 + let finalStep = '1'; + if (data) { + if (data.sketch && data.sketch.data && data.sketch.data.length > 0) { + const sketchList = []; + for (const sketch of data.sketch.data) { + sketchList.push({ + url: sketch.image_path, + script: sketch.sketch_name, + }); + } + setTaskSketch(sketchList); + setSketchCount(sketchList.length); + setTotalSketchCount(data.sketch.total_count); + // 设置为最后一个草图 + if (data.sketch.total_count > data.sketch.data.length) { + setIsGeneratingSketch(true); + setCurrentSketchIndex(data.sketch.data.length - 1); + loadingText = LOADING_TEXT_MAP.sketch(data.sketch.data.length, data.sketch.total_count); + } else { + finalStep = '2'; + if (!data.character || !data.character.data || !data.character.data.length) { + loadingText = LOADING_TEXT_MAP.newCharacter(0, data.character.total_count); + } + } + } + if (data.character && data.character.data && data.character.data.length > 0) { + const characterList = []; + for (const character of data.character.data) { + characterList.push({ + name: character.character_name, + url: character.image_path, + sound: null, + soundDescription: '', + roleDescription: '' + }); + } + setRoles(characterList); + if (data.character.total_count > data.character.data.length) { + loadingText = LOADING_TEXT_MAP.newCharacter(data.character.data.length, data.character.total_count); + } else { + finalStep = '3'; + if (!data.video || !data.video.data || !data.video.data.length) { + loadingText = LOADING_TEXT_MAP.video(0, data.video.total_count || data.sketch.total_count); + } + } + } + if (data.video && data.video.data && data.video.data.length > 0) { + const videoList = []; + for (const video of data.video.data) { + // 每一项 video 有多个视频 先默认取第一个 + videoList.push({ + url: video[0].qiniuVideoUrl, + script: video[0].operation.metadata.video.prompt, + audio: null, + }); + } + setTaskVideos(videoList); + // 如果在视频步骤,设置为最后一个视频 + if (data.video.total_count > data.video.data.length) { + setIsGeneratingVideo(true); + setCurrentSketchIndex(data.video.data.length - 1); + loadingText = LOADING_TEXT_MAP.video(data.video.data.length, data.video.total_count); + } else { + finalStep = '4'; + loadingText = LOADING_TEXT_MAP.audio; + + // 暂时没有音频生成 直接跳过 + finalStep = '5'; + loadingText = LOADING_TEXT_MAP.postProduction('post-production...'); + } + } + if (data.final_video && data.final_video.video) { + setFinal({ + url: data.final_video.video + }); + finalStep = '6'; + loadingText = LOADING_TEXT_MAP.complete; + } + } + + // 设置步骤 + setCurrentStep(finalStep); + setTaskObject(prev => { + if (!prev) return null; + return { + ...prev, + taskStatus: finalStep + }; + }); + console.log('---------loadingText', loadingText); + setCurrentLoadingText(loadingText); + + // 设置是否需要获取流式数据 + setNeedStreamData(status !== 'COMPLETED' && finalStep !== '6'); + + } catch (error) { + console.error('初始化失败:', error); + setDataLoadError('加载失败,请重试'); + setIsLoading(false); } }; // 重试加载数据 - const retryLoadData = async () => { - console.log('用户点击重试,重新加载数据...'); - selectedMockData = null; // 重置数据 + const retryLoadData = () => { setDataLoadError(null); - setIsLoading(true); - setCurrentStep('0'); - - // 重新初始化整个流程 - await initializeWorkflow(); - }; - - // 模拟接口请求 获取任务详情 - const getTaskDetail = async (taskId: string) => { - // 确保已经加载了数据 - if (!selectedMockData) { - console.warn('selectedMockData为空,重新加载数据'); - await loadMockData(); - } - - // 确保数据结构正确 - if (!selectedMockData || !selectedMockData.detail) { - throw new Error('数据结构不正确'); - } - - const data = { - projectId: selectedMockData.detail.projectId, - projectName: selectedMockData.detail.projectName, - taskId: taskId, - taskName: selectedMockData.detail.taskName, - taskDescription: selectedMockData.detail.taskDescription, - taskStatus: selectedMockData.detail.taskStatus, - taskProgress: 0, - mode: selectedMockData.detail.mode, - resolution: selectedMockData.detail.resolution.toString(), - }; - return data; - }; - - // 模拟接口请求 每次获取一个分镜草图 轮询获取 - const getTaskSketch = async (taskId: string) => { - if (isGeneratingSketch || taskSketch.length > 0) return; - - setIsGeneratingSketch(true); + // 重置所有状态 setTaskSketch([]); - - const sketchData = selectedMockData.sketch; - const totalSketches = sketchData.length; - - // 模拟分批获取分镜草图 - for (let i = 0; i < totalSketches; i++) { - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.sketch)); // 10s - - const newSketch = { - id: `sketch-${i}`, - url: sketchData[i].url, - script: sketchData[i].script, - bg_rgb: sketchData[i].bg_rgb, - status: 'done' - }; - - setTaskSketch(prev => { - if (prev.find(sketch => sketch.id === newSketch.id)) { - return prev; - } - return [...prev, newSketch]; - }); - setCurrentSketchIndex(i); - setSketchCount(i + 1); - } - - // 等待最后一个动画完成再设置生成状态为false - await new Promise(resolve => setTimeout(resolve, 1500)); - setIsGeneratingSketch(false); - }; - - // 模拟接口请求 每次获取一个角色 轮询获取 - const getTaskRole = async (taskId: string) => { - setTaskRoles([]); - const roleData = selectedMockData.roles; - const totalRoles = roleData.length; - - for (let i = 0; i < totalRoles; i++) { - // 先更新loading文字显示当前正在生成的角色 - setCurrentLoadingText(STEP_MESSAGES.newCharacter(i, totalRoles)); - - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.character)); // 2s 一个角色 - - // 添加角色到列表 - setTaskRoles(prev => [...prev, roleData[i]]); - - // 更新loading文字显示已完成的角色数量 - setCurrentLoadingText(STEP_MESSAGES.newCharacter(i + 1, totalRoles)); - - // 如果不是最后一个角色,稍微延迟一下让用户看到更新 - if (i < totalRoles - 1) { - await new Promise(resolve => setTimeout(resolve, 500)); - } - } - }; - - // 模拟接口请求 获取背景音 - const getTaskBackgroundAudio = async (taskId: string) => { - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.audio)); // 10s - }; - - // 模拟接口请求 获取最终成品 - const getTaskFinalProduct = async (taskId: string) => { - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.final)); // 50s - }; - - // 模拟接口请求 每次获取一个分镜视频 轮询获取 - const getTaskVideo = async (taskId: string) => { - setIsGeneratingVideo(true); setTaskVideos([]); - - const videoData = selectedMockData.video; - const totalVideos = videoData.length; - - // 模拟分批获取分镜视频 - for (let i = 0; i < totalVideos; i++) { - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.video)); // 60s - - const newVideo = { - id: `video-${i}`, - url: videoData[i].url, - script: videoData[i].script, - status: 'done' - }; - - setTaskVideos(prev => { - if (prev.find(video => video.id === newVideo.id)) { - return prev; - } - return [...prev, newVideo]; - }); - setCurrentSketchIndex(i); - } - - // 等待最后一个动画完成再设置生成状态为false - await new Promise(resolve => setTimeout(resolve, 1500)); - setIsGeneratingVideo(false); + setSketchCount(0); + setTotalSketchCount(0); + setRoles([]); + setMusic([]); + setFinal(null); + setCurrentSketchIndex(0); + setCurrentStep('0'); + // 重新初始化 + initializeWorkflow(); }; - // 更新加载文字 - useEffect(() => { - if (isLoading) { - // 在初始加载阶段,保持当前loading文字不变 - return; - } - - const totalSketches = selectedMockData?.sketch?.length || 0; - const totalVideos = selectedMockData?.video?.length || 0; - const totalCharacters = selectedMockData?.roles?.length || 0; - - if (currentStep === '1') { - if (isGeneratingSketch) { - setCurrentLoadingText(STEP_MESSAGES.sketch(sketchCount, totalSketches)); - } else { - setCurrentLoadingText(STEP_MESSAGES.sketchComplete); - } - } else if (currentStep === '2') { - // 在角色生成阶段,loading文字已经在 getTaskRole 函数中直接管理 - // 这里不需要额外设置,避免覆盖 - if (taskRoles.length === totalCharacters) { - setCurrentLoadingText(STEP_MESSAGES.newCharacter(totalCharacters, totalCharacters)); - } - } else if (currentStep === '3') { - if (isGeneratingVideo) { - setCurrentLoadingText(STEP_MESSAGES.video(taskVideos.length, totalVideos)); - } else { - setCurrentLoadingText(STEP_MESSAGES.videoComplete); - } - } else if (currentStep === '4') { - setCurrentLoadingText(STEP_MESSAGES.audio); - } else if (currentStep === '5') { - setCurrentLoadingText(STEP_MESSAGES.final); - } else { - setCurrentLoadingText(STEP_MESSAGES.complete); - } - }, [isLoading, currentStep, isGeneratingSketch, sketchCount, isGeneratingVideo, taskVideos.length, taskSketch.length, taskRoles.length]); - - // 工作流初始化函数 - const initializeWorkflow = async () => { - try { - setIsLoading(true); - setCurrentLoadingText('正在初始化工作流...'); - - const taskId = (typeof window !== 'undefined' ? localStorage.getItem("taskId") : null) || "taskId-123"; - - // 首先加载数据 - await loadMockData(); - - // 然后获取任务详情 - setCurrentLoadingText('正在加载任务详情...'); - const data = await getTaskDetail(taskId); - setTaskObject(data); - - // 数据加载完成,进入工作流 - setIsLoading(false); - setCurrentStep('1'); - - // 只在任务详情加载完成后获取分镜草图 - await getTaskSketch(taskId); - - // 修改 taskObject 下的 taskStatus 为 '2' - setTaskObject((prev: any) => ({ - ...prev, - taskStatus: '2' - })); - setCurrentStep('2'); - - // 获取分镜草图后,开始绘制角色 - await getTaskRole(taskId); - - // 修改 taskObject 下的 taskStatus 为 '3' - setTaskObject((prev: any) => ({ - ...prev, - taskStatus: '3' - })); - setCurrentStep('3'); - - // 获取绘制角色后,开始获取分镜视频 - await getTaskVideo(taskId); - - // 修改 taskObject 下的 taskStatus 为 '4' - setTaskObject((prev: any) => ({ - ...prev, - taskStatus: '4' - })); - setCurrentStep('4'); - - // 获取分镜视频后,开始获取背景音 - await getTaskBackgroundAudio(taskId); - // 后期制作:抽卡中 对口型中 配音中 一致性处理中 - setCurrentLoadingText(STEP_MESSAGES.postProduction('Selecting optimal frames')); - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.postProduction)); - setCurrentLoadingText(STEP_MESSAGES.postProduction('Aligning lip sync')); - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.postProduction)); - setCurrentLoadingText(STEP_MESSAGES.postProduction('Adding background audio')); - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.postProduction)); - setCurrentLoadingText(STEP_MESSAGES.postProduction('Consistency processing')); - await new Promise(resolve => setTimeout(resolve, MOCK_DELAY_TIME.postProduction)); - - // 修改 taskObject 下的 taskStatus 为 '5' - setTaskObject((prev: any) => ({ - ...prev, - taskStatus: '5' - })); - setCurrentStep('5'); - - // 获取背景音后,开始获取最终成品 - await getTaskFinalProduct(taskId); - await new Promise(resolve => setTimeout(resolve, 2000)); - - // 修改 taskObject 下的 taskStatus 为 '6' - setTaskObject((prev: any) => ({ - ...prev, - taskStatus: '6' - })); - setCurrentStep('6'); - - } catch (error) { - console.error('工作流初始化失败:', error); - setDataLoadError('工作流初始化失败,请刷新页面重试'); - setIsLoading(false); - } - }; - - // 初始化数据 + // 初始化 useEffect(() => { initializeWorkflow(); - }, []); + }, [episodeId]); return { - // 状态数据 taskObject, taskSketch, taskVideos, sketchCount, - isLoading: isLoading || isLoadingData, // 合并loading状态 + isLoading, currentStep, currentSketchIndex, isGeneratingSketch, isGeneratingVideo, currentLoadingText, - totalSketchCount: selectedMockData?.sketch?.length || 0, - roles: selectedMockData?.roles || [], - music: selectedMockData?.music || {}, - final: selectedMockData?.final || {}, + totalSketchCount, + roles, + music, + final, dataLoadError, - // 操作方法 setCurrentSketchIndex, retryLoadData, }; diff --git a/components/ui/liquid-glass-button.tsx b/components/ui/liquid-glass-button.tsx index ef15663..447f293 100644 --- a/components/ui/liquid-glass-button.tsx +++ b/components/ui/liquid-glass-button.tsx @@ -91,46 +91,64 @@ const liquidbuttonVariants = cva( } ) -function LiquidButton({ - className, - variant, - size, - asChild = false, - children, - ...props -}: React.ComponentProps<"button"> & - VariantProps & { - asChild?: boolean - }) { - const Comp = asChild ? Slot : "button" - - return ( - <> - - - - - - {children} - - - - > - ) +type PolymorphicRef = React.ComponentPropsWithRef["ref"] +type AsProp = { + as?: C } +type PropsToOmit = keyof (AsProp & P) +type PolymorphicComponentProp< + C extends React.ElementType, + Props = {} +> = React.PropsWithChildren> & + Omit, PropsToOmit> + +type PolymorphicComponentPropWithRef< + C extends React.ElementType, + Props = {} +> = PolymorphicComponentProp & { ref?: PolymorphicRef } + +interface LiquidButtonProps extends VariantProps { + asChild?: boolean +} + +const LiquidButton = React.forwardRef( + ( + { className, variant, size, asChild = false, children, ...props }: PolymorphicComponentPropWithRef, + ref?: PolymorphicRef + ) => { + const Comp = asChild ? Slot : "button" + + return ( + <> + + + + + + {children} + + + + > + ) + } +); + +(LiquidButton as any).displayName = "LiquidButton" function GlassFilter() { diff --git a/plugins/liquid-glass/index.tsx b/plugins/liquid-glass/index.tsx deleted file mode 100644 index a0e6b72..0000000 --- a/plugins/liquid-glass/index.tsx +++ /dev/null @@ -1,612 +0,0 @@ -import { type CSSProperties, forwardRef, useCallback, useEffect, useId, useRef, useState } from "react" -import { ShaderDisplacementGenerator, fragmentShaders } from "./shader-utils" -import { displacementMap, polarDisplacementMap, prominentDisplacementMap } from "./utils" - -// Generate shader-based displacement map using shaderUtils -const generateShaderDisplacementMap = (width: number, height: number): string => { - const generator = new ShaderDisplacementGenerator({ - width, - height, - fragment: fragmentShaders.liquidGlass, - }) - - const dataUrl = generator.updateShader() - generator.destroy() - - return dataUrl -} - -const getMap = (mode: "standard" | "polar" | "prominent" | "shader", shaderMapUrl?: string) => { - switch (mode) { - case "standard": - return displacementMap - case "polar": - return polarDisplacementMap - case "prominent": - return prominentDisplacementMap - case "shader": - return shaderMapUrl || displacementMap - default: - throw new Error(`Invalid mode: ${mode}`) - } -} - -/* ---------- SVG filter (edge-only displacement) ---------- */ -const GlassFilter: React.FC<{ id: string; displacementScale: number; aberrationIntensity: number; width: number; height: number; mode: "standard" | "polar" | "prominent" | "shader"; shaderMapUrl?: string }> = ({ - id, - displacementScale, - aberrationIntensity, - width, - height, - mode, - shaderMapUrl, -}) => ( - - - - - - - - - - - {/* Create edge mask using the displacement map itself */} - - - - - - {/* Original undisplaced image for center */} - - - {/* Red channel displacement with slight offset */} - - - - {/* Green channel displacement */} - - - - {/* Blue channel displacement with slight offset */} - - - - {/* Combine all channels with screen blend mode for chromatic aberration */} - - - - {/* Add slight blur to soften the aberration effect */} - - - {/* Apply edge mask to aberration effect */} - - - {/* Create inverted mask for center */} - - - - - - {/* Combine edge aberration with clean center */} - - - - -) - -/* ---------- container ---------- */ -const GlassContainer = forwardRef< - HTMLDivElement, - React.PropsWithChildren<{ - className?: string - style?: React.CSSProperties - displacementScale?: number - blurAmount?: number - saturation?: number - aberrationIntensity?: number - mouseOffset?: { x: number; y: number } - onMouseLeave?: () => void - onMouseEnter?: () => void - onMouseDown?: () => void - onMouseUp?: () => void - active?: boolean - overLight?: boolean - cornerRadius?: number - padding?: string - glassSize?: { width: number; height: number } - onClick?: () => void - mode?: "standard" | "polar" | "prominent" | "shader" - }> ->( - ( - { - children, - className = "", - style, - displacementScale = 25, - blurAmount = 12, - saturation = 180, - aberrationIntensity = 2, - onMouseEnter, - onMouseLeave, - onMouseDown, - onMouseUp, - active = false, - overLight = false, - cornerRadius = 999, - padding = "24px 32px", - glassSize = { width: 270, height: 69 }, - onClick, - mode = "standard", - }, - ref, - ) => { - const filterId = useId() - const [shaderMapUrl, setShaderMapUrl] = useState("") - - const isFirefox = navigator.userAgent.toLowerCase().includes("firefox") - - // Generate shader displacement map when in shader mode - useEffect(() => { - if (mode === "shader") { - const url = generateShaderDisplacementMap(glassSize.width, glassSize.height) - setShaderMapUrl(url) - } - }, [mode, glassSize.width, glassSize.height]) - - const backdropStyle = { - filter: isFirefox ? null : `url(#${filterId})`, - backdropFilter: `blur(${(overLight ? 12 : 4) + blurAmount * 32}px) saturate(${saturation}%)`, - } - - return ( - - - - - {/* backdrop layer that gets wiggly */} - - - {/* user content stays sharp */} - - {children} - - - - ) - }, -) - -GlassContainer.displayName = "GlassContainer" - -interface LiquidGlassProps { - children: React.ReactNode - displacementScale?: number - blurAmount?: number - saturation?: number - aberrationIntensity?: number - elasticity?: number - cornerRadius?: number - globalMousePos?: { x: number; y: number } - mouseOffset?: { x: number; y: number } - mouseContainer?: React.RefObject | null - className?: string - padding?: string - style?: React.CSSProperties - overLight?: boolean - mode?: "standard" | "polar" | "prominent" | "shader" - onClick?: () => void -} - -export default function LiquidGlass({ - children, - displacementScale = 70, - blurAmount = 0.0625, - saturation = 140, - aberrationIntensity = 2, - elasticity = 0.15, - cornerRadius = 999, - globalMousePos: externalGlobalMousePos, - mouseOffset: externalMouseOffset, - mouseContainer = null, - className = "", - padding = "24px 32px", - overLight = false, - style = {}, - mode = "standard", - onClick, -}: LiquidGlassProps) { - const glassRef = useRef(null) - const [isHovered, setIsHovered] = useState(false) - const [isActive, setIsActive] = useState(false) - const [glassSize, setGlassSize] = useState({ width: 270, height: 69 }) - const [internalGlobalMousePos, setInternalGlobalMousePos] = useState({ x: 0, y: 0 }) - const [internalMouseOffset, setInternalMouseOffset] = useState({ x: 0, y: 0 }) - - // Use external mouse position if provided, otherwise use internal - const globalMousePos = externalGlobalMousePos || internalGlobalMousePos - const mouseOffset = externalMouseOffset || internalMouseOffset - - // Internal mouse tracking - const handleMouseMove = useCallback( - (e: MouseEvent) => { - const container = mouseContainer?.current || glassRef.current - if (!container) { - return - } - - const rect = container.getBoundingClientRect() - const centerX = rect.left + rect.width / 2 - const centerY = rect.top + rect.height / 2 - - setInternalMouseOffset({ - x: ((e.clientX - centerX) / rect.width) * 100, - y: ((e.clientY - centerY) / rect.height) * 100, - }) - - setInternalGlobalMousePos({ - x: e.clientX, - y: e.clientY, - }) - }, - [mouseContainer], - ) - - // Set up mouse tracking if no external mouse position is provided - useEffect(() => { - if (externalGlobalMousePos && externalMouseOffset) { - // External mouse tracking is provided, don't set up internal tracking - return - } - - const container = mouseContainer?.current || glassRef.current - if (!container) { - return - } - - container.addEventListener("mousemove", handleMouseMove) - - return () => { - container.removeEventListener("mousemove", handleMouseMove) - } - }, [handleMouseMove, mouseContainer, externalGlobalMousePos, externalMouseOffset]) - - // Calculate directional scaling based on mouse position - const calculateDirectionalScale = useCallback(() => { - if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) { - return "scale(1)" - } - - const rect = glassRef.current.getBoundingClientRect() - const pillCenterX = rect.left + rect.width / 2 - const pillCenterY = rect.top + rect.height / 2 - const pillWidth = glassSize.width - const pillHeight = glassSize.height - - const deltaX = globalMousePos.x - pillCenterX - const deltaY = globalMousePos.y - pillCenterY - - // Calculate distance from mouse to pill edges (not center) - const edgeDistanceX = Math.max(0, Math.abs(deltaX) - pillWidth / 2) - const edgeDistanceY = Math.max(0, Math.abs(deltaY) - pillHeight / 2) - const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY) - - // Activation zone: 200px from edges - const activationZone = 200 - - // If outside activation zone, no effect - if (edgeDistance > activationZone) { - return "scale(1)" - } - - // Calculate fade-in factor (1 at edge, 0 at activation zone boundary) - const fadeInFactor = 1 - edgeDistance / activationZone - - // Normalize the deltas for direction - const centerDistance = Math.sqrt(deltaX * deltaX + deltaY * deltaY) - if (centerDistance === 0) { - return "scale(1)" - } - - const normalizedX = deltaX / centerDistance - const normalizedY = deltaY / centerDistance - - // Calculate stretch factors with fade-in - const stretchIntensity = Math.min(centerDistance / 300, 1) * elasticity * fadeInFactor - - // X-axis scaling: stretch horizontally when moving left/right, compress when moving up/down - const scaleX = 1 + Math.abs(normalizedX) * stretchIntensity * 0.3 - Math.abs(normalizedY) * stretchIntensity * 0.15 - - // Y-axis scaling: stretch vertically when moving up/down, compress when moving left/right - const scaleY = 1 + Math.abs(normalizedY) * stretchIntensity * 0.3 - Math.abs(normalizedX) * stretchIntensity * 0.15 - - return `scaleX(${Math.max(0.8, scaleX)}) scaleY(${Math.max(0.8, scaleY)})` - }, [globalMousePos, elasticity, glassSize]) - - // Helper function to calculate fade-in factor based on distance from element edges - const calculateFadeInFactor = useCallback(() => { - if (!globalMousePos.x || !globalMousePos.y || !glassRef.current) { - return 0 - } - - const rect = glassRef.current.getBoundingClientRect() - const pillCenterX = rect.left + rect.width / 2 - const pillCenterY = rect.top + rect.height / 2 - const pillWidth = glassSize.width - const pillHeight = glassSize.height - - const edgeDistanceX = Math.max(0, Math.abs(globalMousePos.x - pillCenterX) - pillWidth / 2) - const edgeDistanceY = Math.max(0, Math.abs(globalMousePos.y - pillCenterY) - pillHeight / 2) - const edgeDistance = Math.sqrt(edgeDistanceX * edgeDistanceX + edgeDistanceY * edgeDistanceY) - - const activationZone = 200 - return edgeDistance > activationZone ? 0 : 1 - edgeDistance / activationZone - }, [globalMousePos, glassSize]) - - // Helper function to calculate elastic translation - const calculateElasticTranslation = useCallback(() => { - if (!glassRef.current) { - return { x: 0, y: 0 } - } - - const fadeInFactor = calculateFadeInFactor() - const rect = glassRef.current.getBoundingClientRect() - const pillCenterX = rect.left + rect.width / 2 - const pillCenterY = rect.top + rect.height / 2 - - return { - x: (globalMousePos.x - pillCenterX) * elasticity * 0.1 * fadeInFactor, - y: (globalMousePos.y - pillCenterY) * elasticity * 0.1 * fadeInFactor, - } - }, [globalMousePos, elasticity, calculateFadeInFactor]) - - // Update glass size whenever component mounts or window resizes - useEffect(() => { - const updateGlassSize = () => { - if (glassRef.current) { - const rect = glassRef.current.getBoundingClientRect() - setGlassSize({ width: rect.width, height: rect.height }) - } - } - - updateGlassSize() - window.addEventListener("resize", updateGlassSize) - return () => window.removeEventListener("resize", updateGlassSize) - }, []) - - const transformStyle = `translate(calc(-50% + ${calculateElasticTranslation().x}px), calc(-50% + ${calculateElasticTranslation().y}px)) ${isActive && Boolean(onClick) ? "scale(0.96)" : calculateDirectionalScale()}` - - const baseStyle = { - ...style, - transform: transformStyle, - transition: "all ease-out 0.2s", - } - - const positionStyles = { - position: baseStyle.position || "relative", - top: baseStyle.top || "50%", - left: baseStyle.left || "50%", - } - - return ( - <> - {/* Over light effect */} - - - - setIsHovered(true)} - onMouseLeave={() => setIsHovered(false)} - onMouseDown={() => setIsActive(true)} - onMouseUp={() => setIsActive(false)} - active={isActive} - overLight={overLight} - onClick={onClick} - mode={mode} - > - {children} - - - {/* Border layer 1 - extracted from glass container */} - - - {/* Border layer 2 - duplicate with mix-blend-overlay */} - - - {/* Hover effects */} - {Boolean(onClick) && ( - <> - - - - > - )} - > - ) -} diff --git a/plugins/liquid-glass/shader-utils.ts b/plugins/liquid-glass/shader-utils.ts deleted file mode 100644 index e0957ff..0000000 --- a/plugins/liquid-glass/shader-utils.ts +++ /dev/null @@ -1,134 +0,0 @@ -// Adapted from https://github.com/shuding/liquid-glass - -export interface Vec2 { - x: number - y: number -} - -export interface ShaderOptions { - width: number - height: number - fragment: (uv: Vec2, mouse?: Vec2) => Vec2 - mousePosition?: Vec2 -} - -function smoothStep(a: number, b: number, t: number): number { - t = Math.max(0, Math.min(1, (t - a) / (b - a))) - return t * t * (3 - 2 * t) -} - -function length(x: number, y: number): number { - return Math.sqrt(x * x + y * y) -} - -function roundedRectSDF(x: number, y: number, width: number, height: number, radius: number): number { - const qx = Math.abs(x) - width + radius - const qy = Math.abs(y) - height + radius - return Math.min(Math.max(qx, qy), 0) + length(Math.max(qx, 0), Math.max(qy, 0)) - radius -} - -function texture(x: number, y: number): Vec2 { - return { x, y } -} - -// Shader fragment functions for different effects -export const fragmentShaders = { - liquidGlass: (uv: Vec2): Vec2 => { - const ix = uv.x - 0.5 - const iy = uv.y - 0.5 - const distanceToEdge = roundedRectSDF(ix, iy, 0.3, 0.2, 0.6) - const displacement = smoothStep(0.8, 0, distanceToEdge - 0.15) - const scaled = smoothStep(0, 1, displacement) - return texture(ix * scaled + 0.5, iy * scaled + 0.5) - }, -} - -export type FragmentShaderType = keyof typeof fragmentShaders - -export class ShaderDisplacementGenerator { - private canvas: HTMLCanvasElement - private context: CanvasRenderingContext2D - private canvasDPI = 1 - - constructor(private options: ShaderOptions) { - this.canvas = document.createElement("canvas") - this.canvas.width = options.width * this.canvasDPI - this.canvas.height = options.height * this.canvasDPI - this.canvas.style.display = "none" - - const context = this.canvas.getContext("2d") - if (!context) { - throw new Error("Could not get 2D context") - } - this.context = context - } - - updateShader(mousePosition?: Vec2): string { - const w = this.options.width * this.canvasDPI - const h = this.options.height * this.canvasDPI - - let maxScale = 0 - const rawValues: number[] = [] - - // Calculate displacement values - for (let y = 0; y < h; y++) { - for (let x = 0; x < w; x++) { - const uv: Vec2 = { x: x / w, y: y / h } - - const pos = this.options.fragment(uv, mousePosition) - const dx = pos.x * w - x - const dy = pos.y * h - y - - maxScale = Math.max(maxScale, Math.abs(dx), Math.abs(dy)) - rawValues.push(dx, dy) - } - } - - // Improved normalization to prevent artifacts while maintaining intensity - if (maxScale > 0) { - maxScale = Math.max(maxScale, 1) // Ensure minimum scale to prevent over-normalization - } else { - maxScale = 1 - } - - // Create ImageData and fill it - const imageData = this.context.createImageData(w, h) - const data = imageData.data - - // Convert to image data with smoother normalization - let rawIndex = 0 - for (let y = 0; y < h; y++) { - for (let x = 0; x < w; x++) { - const dx = rawValues[rawIndex++] - const dy = rawValues[rawIndex++] - - // Smooth the displacement values at edges to prevent hard transitions - const edgeDistance = Math.min(x, y, w - x - 1, h - y - 1) - const edgeFactor = Math.min(1, edgeDistance / 2) // Smooth within 2 pixels of edge - - const smoothedDx = dx * edgeFactor - const smoothedDy = dy * edgeFactor - - const r = smoothedDx / maxScale + 0.5 - const g = smoothedDy / maxScale + 0.5 - - const pixelIndex = (y * w + x) * 4 - data[pixelIndex] = Math.max(0, Math.min(255, r * 255)) // Red channel (X displacement) - data[pixelIndex + 1] = Math.max(0, Math.min(255, g * 255)) // Green channel (Y displacement) - data[pixelIndex + 2] = Math.max(0, Math.min(255, g * 255)) // Blue channel (Y displacement for SVG filter compatibility) - data[pixelIndex + 3] = 255 // Alpha channel - } - } - - this.context.putImageData(imageData, 0, 0) - return this.canvas.toDataURL() - } - - destroy(): void { - this.canvas.remove() - } - - getScale(): number { - return this.canvasDPI - } -} diff --git a/plugins/liquid-glass/utils.ts b/plugins/liquid-glass/utils.ts deleted file mode 100644 index ce6b8ff..0000000 --- a/plugins/liquid-glass/utils.ts +++ /dev/null @@ -1,8 +0,0 @@ -export const displacementMap = - "" - -export const polarDisplacementMap = - "" - -export const prominentDisplacementMap = - ""
View and manage your video projects
- {searchTerm - ? 'Try adjusting your search terms' - : 'Create your first video project to see it here' - } -
Click to upload a video file