"use client"; import { useDragDrop } from "@/hooks/use-drag-drop"; import { processMediaFiles } from "@/lib/media-processing"; import { useMediaStore, type MediaItem } from "@/stores/media-store"; import { Image, Music, Plus, Upload, Video } from "lucide-react"; import { useEffect, useRef, useState } from "react"; import { toast } from "sonner"; import { Button } from "@/components/ui/button"; import { DragOverlay } from "@/components/ui/drag-overlay"; import { ContextMenu, ContextMenuContent, ContextMenuItem, ContextMenuTrigger, } from "@/components/ui/context-menu"; import { Input } from "@/components/ui/input"; import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue, } from "@/components/ui/select"; import { DraggableMediaItem } from "@/components/ui/draggable-item"; import { useProjectStore } from "@/stores/project-store"; export function MediaView() { const { mediaItems, addMediaItem, removeMediaItem } = useMediaStore(); const { activeProject } = useProjectStore(); const fileInputRef = useRef(null); const [isProcessing, setIsProcessing] = useState(false); const [progress, setProgress] = useState(0); const [searchQuery, setSearchQuery] = useState(""); const [mediaFilter, setMediaFilter] = useState("all"); const processFiles = async (files: FileList | File[]) => { if (!files || files.length === 0) return; if (!activeProject) { toast.error("No active project"); return; } setIsProcessing(true); setProgress(0); try { // Process files (extract metadata, generate thumbnails, etc.) const processedItems = await processMediaFiles(files, (p) => setProgress(p) ); // Add each processed media item to the store for (const item of processedItems) { await addMediaItem(activeProject.id, item); } } catch (error) { // Show error toast if processing fails console.error("Error processing files:", error); toast.error("Failed to process files"); } finally { setIsProcessing(false); setProgress(0); } }; const { isDragOver, dragProps } = useDragDrop({ // When files are dropped, process them onDrop: processFiles, }); const handleFileSelect = () => fileInputRef.current?.click(); // Open file picker const handleFileChange = (e: React.ChangeEvent) => { // When files are selected via file picker, process them if (e.target.files) processFiles(e.target.files); e.target.value = ""; // Reset input }; const handleRemove = async (e: React.MouseEvent, id: string) => { // Remove a media item from the store e.stopPropagation(); if (!activeProject) { toast.error("No active project"); return; } // Media store now handles cascade deletion automatically await removeMediaItem(activeProject.id, id); }; const formatDuration = (duration: number) => { // Format seconds as mm:ss const min = Math.floor(duration / 60); const sec = Math.floor(duration % 60); return `${min}:${sec.toString().padStart(2, "0")}`; }; const [filteredMediaItems, setFilteredMediaItems] = useState(mediaItems); useEffect(() => { const filtered = mediaItems.filter((item) => { if (mediaFilter && mediaFilter !== "all" && item.type !== mediaFilter) { return false; } if ( searchQuery && !item.name.toLowerCase().includes(searchQuery.toLowerCase()) ) { return false; } return true; }); setFilteredMediaItems(filtered); }, [mediaItems, mediaFilter, searchQuery]); const renderPreview = (item: MediaItem) => { // Render a preview for each media type (image, video, audio, unknown) if (item.type === "image") { return (
{item.name}
); } if (item.type === "video") { if (item.thumbnailUrl) { return (
{item.name}
{item.duration && (
{formatDuration(item.duration)}
)}
); } return (
); } if (item.type === "audio") { return (
Audio {item.duration && ( {formatDuration(item.duration)} )}
); } return (
Unknown
); }; return ( <> {/* Hidden file input for uploading media */}
{/* Show overlay when dragging files over the panel */}
{/* Button to add/upload media */}
{/* Search and filter controls */} setSearchQuery(e.target.value)} /> {/* Add media button */}
{/* Show message if no media, otherwise show media grid */} {filteredMediaItems.length === 0 ? (

No media in project

Drag files here or use the button above

) : (
{/* Render each media item as a draggable button */} {filteredMediaItems.map((item) => ( Export clips handleRemove(e, item.id)} > Delete ))}
)}
); }