"use client"; import { useTimelineStore, type TimelineClip, type TimelineTrack, } from "@/stores/timeline-store"; import { useMediaStore, type MediaItem } from "@/stores/media-store"; import { usePlaybackStore } from "@/stores/playback-store"; import { VideoPlayer } from "@/components/ui/video-player"; import { Button } from "@/components/ui/button"; import { Play, Pause, Volume2, VolumeX, Plus } from "lucide-react"; import { useState, useRef, useEffect } from "react"; interface ActiveClip { clip: TimelineClip; track: TimelineTrack; mediaItem: MediaItem | null; } export function PreviewPanel() { const { tracks } = useTimelineStore(); const { mediaItems } = useMediaStore(); const { currentTime, muted, toggleMute, volume } = usePlaybackStore(); const [canvasSize, setCanvasSize] = useState({ width: 1920, height: 1080 }); const previewRef = useRef(null); const containerRef = useRef(null); const [previewDimensions, setPreviewDimensions] = useState({ width: 0, height: 0, }); // Calculate optimal preview size that fits in container while maintaining aspect ratio useEffect(() => { const updatePreviewSize = () => { if (!containerRef.current) return; const container = containerRef.current.getBoundingClientRect(); const computedStyle = getComputedStyle(containerRef.current); // Get padding values const paddingTop = parseFloat(computedStyle.paddingTop); const paddingBottom = parseFloat(computedStyle.paddingBottom); const paddingLeft = parseFloat(computedStyle.paddingLeft); const paddingRight = parseFloat(computedStyle.paddingRight); // Get gap value (gap-4 = 1rem = 16px) const gap = parseFloat(computedStyle.gap) || 16; // Get toolbar height if it exists const toolbar = containerRef.current.querySelector("[data-toolbar]"); const toolbarHeight = toolbar ? toolbar.getBoundingClientRect().height : 0; // Calculate available space after accounting for padding, gap, and toolbar const availableWidth = container.width - paddingLeft - paddingRight; const availableHeight = container.height - paddingTop - paddingBottom - toolbarHeight - (toolbarHeight > 0 ? gap : 0); const targetRatio = canvasSize.width / canvasSize.height; const containerRatio = availableWidth / availableHeight; let width, height; if (containerRatio > targetRatio) { // Container is wider - constrain by height height = availableHeight; width = height * targetRatio; } else { // Container is taller - constrain by width width = availableWidth; height = width / targetRatio; } setPreviewDimensions({ width, height }); }; updatePreviewSize(); const resizeObserver = new ResizeObserver(updatePreviewSize); if (containerRef.current) { resizeObserver.observe(containerRef.current); } return () => resizeObserver.disconnect(); }, [canvasSize.width, canvasSize.height]); // Get active clips at current time const getActiveClips = (): ActiveClip[] => { const activeClips: ActiveClip[] = []; tracks.forEach((track) => { track.clips.forEach((clip) => { const clipStart = clip.startTime; const clipEnd = clip.startTime + (clip.duration - clip.trimStart - clip.trimEnd); if (currentTime >= clipStart && currentTime < clipEnd) { const mediaItem = clip.mediaId === "test" ? null // Test clips don't have a real media item : mediaItems.find((item) => item.id === clip.mediaId) || null; activeClips.push({ clip, track, mediaItem }); } }); }); return activeClips; }; const activeClips = getActiveClips(); // Render a clip const renderClip = (clipData: ActiveClip, index: number) => { const { clip, mediaItem } = clipData; // Test clips if (!mediaItem || clip.mediaId === "test") { return (
🎬

{clip.name}

); } // Video clips if (mediaItem.type === "video") { return (
); } // Image clips if (mediaItem.type === "image") { return (
{mediaItem.name}
); } // Audio clips (visual representation) if (mediaItem.type === "audio") { return (
🎵

{mediaItem.name}

); } return null; }; // Canvas presets const canvasPresets = [ { name: "16:9 HD", width: 1920, height: 1080 }, { name: "16:9 4K", width: 3840, height: 2160 }, { name: "9:16 Mobile", width: 1080, height: 1920 }, { name: "1:1 Square", width: 1080, height: 1080 }, { name: "4:3 Standard", width: 1440, height: 1080 }, ]; return (
{/* Controls */}
Canvas:
{/* Preview Area */}
{activeClips.length === 0 ? (
{tracks.length === 0 ? "No media added to timeline" : "No clips at current time"}
) : ( activeClips.map((clipData, index) => renderClip(clipData, index)) )}
); } function PreviewToolbar() { const { isPlaying, toggle } = usePlaybackStore(); return (
); }