'use client'; import { useCallback, useState } from 'react'; import { useDropzone } from 'react-dropzone'; import { Upload, X, FileImage, FileVideo, FileAudio, File } from 'lucide-react'; import { clsx } from 'clsx'; interface FileUploadProps { onUpload: (file: File) => void; accept?: Record; maxSize?: number; label?: string; currentFile?: File | null; onClear?: () => void; } const fileIcons: Record = { image: FileImage, video: FileVideo, audio: FileAudio, }; export default function FileUpload({ onUpload, accept, maxSize = 100 * 1024 * 1024, // 100MB default label = 'Upload a file', currentFile, onClear, }: FileUploadProps) { const [error, setError] = useState(null); const onDrop = useCallback( (acceptedFiles: File[], rejectedFiles: any[]) => { setError(null); if (rejectedFiles.length > 0) { const rejection = rejectedFiles[0]; if (rejection.errors[0]?.code === 'file-too-large') { setError(`File too large. Max size is ${Math.round(maxSize / 1024 / 1024)}MB`); } else if (rejection.errors[0]?.code === 'file-invalid-type') { setError('Invalid file type'); } else { setError('File rejected'); } return; } if (acceptedFiles.length > 0) { onUpload(acceptedFiles[0]); } }, [onUpload, maxSize] ); const { getRootProps, getInputProps, isDragActive } = useDropzone({ onDrop, accept, maxSize, multiple: false, }); const getFileIcon = (file: File) => { const type = file.type.split('/')[0]; const Icon = fileIcons[type] || File; return ; }; const formatFileSize = (bytes: number) => { if (bytes < 1024) return `${bytes} B`; if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`; return `${(bytes / 1024 / 1024).toFixed(1)} MB`; }; if (currentFile) { return (
{getFileIcon(currentFile)}

{currentFile.name}

{formatFileSize(currentFile.size)}

{onClear && ( )}
); } return (

{label}

Drag and drop or browse

{accept && (

Accepted: {Object.keys(accept).join(', ')}

)}
{error && (

{error}

)}
); }