"use client"; import { useAtom } from "jotai"; import { ChevronDown, Dot, File as FileIcon, FolderOpen, Upload, X } from "lucide-react"; import { useTranslations } from "next-intl"; import { type ChangeEvent, useCallback, useEffect, useMemo, useRef, useState } from "react"; import { useDropzone } from "react-dropzone"; import { toast } from "sonner"; import { uploadDocumentMutationAtom } from "@/atoms/documents/document-mutation.atoms"; import { Accordion, AccordionContent, AccordionItem, AccordionTrigger, } from "@/components/ui/accordion"; import { Badge } from "@/components/ui/badge"; import { Button } from "@/components/ui/button"; import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger, } from "@/components/ui/dropdown-menu"; import { Progress } from "@/components/ui/progress"; import { Spinner } from "@/components/ui/spinner"; import { Switch } from "@/components/ui/switch"; import { useElectronAPI } from "@/hooks/use-platform"; import { trackDocumentUploadFailure, trackDocumentUploadStarted, trackDocumentUploadSuccess, } from "@/lib/posthog/events"; import { getAcceptedFileTypes, getSupportedExtensions, getSupportedExtensionsSet, } from "@/lib/supported-extensions"; interface DocumentUploadTabProps { searchSpaceId: string; onSuccess?: () => void; onAccordionStateChange?: (isExpanded: boolean) => void; } interface FileWithId { id: string; file: File; } const MAX_FILE_SIZE_MB = 500; const MAX_FILE_SIZE_BYTES = MAX_FILE_SIZE_MB * 1024 * 1024; const toggleRowClass = "flex items-center justify-between rounded-lg bg-slate-400/5 dark:bg-white/5 p-3"; export function DocumentUploadTab({ searchSpaceId, onSuccess, onAccordionStateChange, }: DocumentUploadTabProps) { const t = useTranslations("upload_documents"); const [files, setFiles] = useState([]); const [uploadProgress, setUploadProgress] = useState(0); const [accordionValue, setAccordionValue] = useState(""); const [shouldSummarize, setShouldSummarize] = useState(false); const [uploadDocumentMutation] = useAtom(uploadDocumentMutationAtom); const { mutate: uploadDocuments, isPending: isUploading } = uploadDocumentMutation; const fileInputRef = useRef(null); const folderInputRef = useRef(null); const progressIntervalRef = useRef | null>(null); useEffect(() => { return () => { if (progressIntervalRef.current) { clearInterval(progressIntervalRef.current); } }; }, []); const electronAPI = useElectronAPI(); const isElectron = !!electronAPI?.browseFiles; const acceptedFileTypes = useMemo(() => getAcceptedFileTypes(), []); const supportedExtensions = useMemo( () => getSupportedExtensions(acceptedFileTypes), [acceptedFileTypes] ); const supportedExtensionsSet = useMemo( () => getSupportedExtensionsSet(acceptedFileTypes), [acceptedFileTypes] ); const addFiles = useCallback( (incoming: File[]) => { const oversized = incoming.filter((f) => f.size > MAX_FILE_SIZE_BYTES); if (oversized.length > 0) { toast.error(t("file_too_large"), { description: t("file_too_large_desc", { name: oversized[0].name, maxMB: MAX_FILE_SIZE_MB, }), }); } const valid = incoming.filter((f) => f.size <= MAX_FILE_SIZE_BYTES); if (valid.length === 0) return; setFiles((prev) => { const newEntries = valid.map((f) => ({ id: crypto.randomUUID?.() ?? `file-${Date.now()}-${Math.random().toString(36)}`, file: f, })); return [...prev, ...newEntries]; }); }, [t] ); const onDrop = useCallback( (acceptedFiles: File[]) => { addFiles(acceptedFiles); }, [addFiles] ); const { getRootProps, getInputProps, isDragActive } = useDropzone({ onDrop, accept: acceptedFileTypes, maxSize: MAX_FILE_SIZE_BYTES, noClick: isElectron, }); const handleFileInputClick = useCallback((e: React.MouseEvent) => { e.stopPropagation(); }, []); const handleBrowseFiles = useCallback(async () => { if (!electronAPI?.browseFiles) return; const paths = await electronAPI.browseFiles(); if (!paths || paths.length === 0) return; const fileDataList = await electronAPI.readLocalFiles(paths); const filtered = fileDataList.filter( (fd: { name: string; data: ArrayBuffer; mimeType: string }) => { const ext = fd.name.includes(".") ? `.${fd.name.split(".").pop()?.toLowerCase()}` : ""; return ext !== "" && supportedExtensionsSet.has(ext); } ); if (filtered.length === 0) { toast.error(t("no_supported_files_in_folder")); return; } const newFiles: FileWithId[] = filtered.map( (fd: { name: string; data: ArrayBuffer; mimeType: string }) => ({ id: crypto.randomUUID?.() ?? `file-${Date.now()}-${Math.random().toString(36)}`, file: new File([fd.data], fd.name, { type: fd.mimeType }), }) ); setFiles((prev) => [...prev, ...newFiles]); }, [electronAPI, supportedExtensionsSet, t]); const handleFolderChange = useCallback( (e: ChangeEvent) => { const fileList = e.target.files; if (!fileList || fileList.length === 0) return; const folderFiles = Array.from(fileList).filter((f) => { const ext = f.name.includes(".") ? `.${f.name.split(".").pop()?.toLowerCase()}` : ""; return ext !== "" && supportedExtensionsSet.has(ext); }); if (folderFiles.length === 0) { toast.error(t("no_supported_files_in_folder")); e.target.value = ""; return; } addFiles(folderFiles); e.target.value = ""; }, [addFiles, supportedExtensionsSet, t] ); const formatFileSize = (bytes: number) => { if (bytes === 0) return "0 Bytes"; const k = 1024; const sizes = ["Bytes", "KB", "MB", "GB", "TB"]; const i = Math.floor(Math.log(bytes) / Math.log(k)); return `${parseFloat((bytes / k ** i).toFixed(2))} ${sizes[i]}`; }; const totalFileSize = files.reduce((total, entry) => total + entry.file.size, 0); const hasContent = files.length > 0; const handleAccordionChange = useCallback( (value: string) => { setAccordionValue(value); onAccordionStateChange?.(value === "supported-file-types"); }, [onAccordionStateChange] ); const handleUpload = async () => { setUploadProgress(0); trackDocumentUploadStarted(Number(searchSpaceId), files.length, totalFileSize); progressIntervalRef.current = setInterval(() => { setUploadProgress((prev) => (prev >= 90 ? prev : prev + Math.random() * 10)); }, 200); const rawFiles = files.map((entry) => entry.file); uploadDocuments( { files: rawFiles, search_space_id: Number(searchSpaceId), should_summarize: shouldSummarize, }, { onSuccess: () => { if (progressIntervalRef.current) clearInterval(progressIntervalRef.current); setUploadProgress(100); trackDocumentUploadSuccess(Number(searchSpaceId), files.length); toast(t("upload_initiated"), { description: t("upload_initiated_desc") }); onSuccess?.(); }, onError: (error: unknown) => { if (progressIntervalRef.current) clearInterval(progressIntervalRef.current); setUploadProgress(0); const message = error instanceof Error ? error.message : "Upload failed"; trackDocumentUploadFailure(Number(searchSpaceId), message); toast(t("upload_error"), { description: `${t("upload_error_desc")}: ${message}`, }); }, } ); }; const renderBrowseButton = (options?: { compact?: boolean; fullWidth?: boolean }) => { const { compact, fullWidth } = options ?? {}; const sizeClass = compact ? "h-7" : "h-8"; const widthClass = fullWidth ? "w-full" : ""; if (isElectron) { return ( e.stopPropagation()}> e.stopPropagation()} > Files folderInputRef.current?.click()}> Folder ); } return ( e.stopPropagation()}> e.stopPropagation()} > fileInputRef.current?.click()}> {t("browse_files")} folderInputRef.current?.click()}> {t("browse_folder")} ); }; return (
{/* Hidden file input */} {/* Hidden folder input for web folder browsing */} )} /> {/* MOBILE DROP ZONE */}
{hasContent ? ( isElectron ? (
{renderBrowseButton({ compact: true, fullWidth: true })}
) : ( ) ) : (
{ if (!isElectron) fileInputRef.current?.click(); }} onKeyDown={(e) => { if (e.key === "Enter" || e.key === " ") { e.preventDefault(); if (!isElectron) fileInputRef.current?.click(); } }} >

{isElectron ? "Select files or folder" : "Tap to select files or folder"}

{t("file_size_limit")}

e.stopPropagation()} onKeyDown={(e) => e.stopPropagation()} role="group" > {renderBrowseButton({ fullWidth: true })}
)}
{/* DESKTOP DROP ZONE */}
{hasContent ? (
{isDragActive ? t("drop_files") : t("drag_drop_more")} {renderBrowseButton({ compact: true })}
) : (
{isDragActive && (

{t("drop_files")}

)}

{t("drag_drop")}

{t("file_size_limit")}

{renderBrowseButton()}
)}
{/* FILES SELECTED */} {files.length > 0 && (

{t("selected_files", { count: files.length })} {formatFileSize(totalFileSize)}

{files.map((entry) => (
{entry.file.name.split(".").pop() || "?"} {entry.file.name} {formatFileSize(entry.file.size)}
))}
{isUploading && (
{t("uploading_files")} {Math.round(uploadProgress)}%
)}

Enable AI Summary

Improves search quality but adds latency

)} {/* SUPPORTED FORMATS */} {t("supported_file_types")}
{supportedExtensions.map((ext) => ( {ext} ))}
); }