import { useState, useRef, useCallback, useMemo, useEffect, type DragEvent } from 'react'; import { useSearchParams } from 'react-router-dom'; import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query'; import { FolderOpen, Loader2, Plus, Upload, Trash2, Download, MoreVertical, ChevronRight, FolderPlus, FileBox, Clock, HardDrive, Copy, File, MoveRight, CheckSquare, Square, LayoutGrid, List, Search, SortAsc, SortDesc, AlertTriangle, Filter, X, CheckCircle, XCircle, Link2, Unlink, Archive as ArchiveIcon, Briefcase, Printer, } from 'lucide-react'; import { api } from '../api/client'; import type { LibraryFolderTree, LibraryFileListItem, LibraryFolderCreate, LibraryFolderUpdate, AppSettings, Archive, } from '../api/client'; import { Button } from '../components/Button'; import { ConfirmModal } from '../components/ConfirmModal'; import { useToast } from '../contexts/ToastContext'; type SortField = 'name' | 'date' | 'size' | 'type' | 'prints'; type SortDirection = 'asc' | 'desc'; // Utility to format file size function formatFileSize(bytes: number): string { if (bytes < 1024) return `${bytes} B`; if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`; if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`; return `${(bytes / (1024 * 1024 * 1024)).toFixed(2)} GB`; } // Utility to format duration function formatDuration(seconds: number | null): string { if (!seconds) return '-'; const hours = Math.floor(seconds / 3600); const mins = Math.floor((seconds % 3600) / 60); if (hours > 0) return `${hours}h ${mins}m`; return `${mins}m`; } // New Folder Modal interface NewFolderModalProps { parentId: number | null; onClose: () => void; onSave: (data: LibraryFolderCreate) => void; isLoading: boolean; } function NewFolderModal({ parentId, onClose, onSave, isLoading }: NewFolderModalProps) { const [name, setName] = useState(''); const handleSubmit = (e: React.FormEvent) => { e.preventDefault(); onSave({ name: name.trim(), parent_id: parentId }); }; return (

New Folder

setName(e.target.value)} className="w-full bg-bambu-dark border border-bambu-dark-tertiary rounded px-3 py-2 text-white placeholder-bambu-gray focus:outline-none focus:border-bambu-green" placeholder="e.g., Functional Parts" autoFocus required />
); } // Move Files Modal interface MoveFilesModalProps { folders: LibraryFolderTree[]; selectedFiles: number[]; currentFolderId: number | null; onClose: () => void; onMove: (folderId: number | null) => void; isLoading: boolean; } function MoveFilesModal({ folders, selectedFiles, currentFolderId, onClose, onMove, isLoading }: MoveFilesModalProps) { const [targetFolder, setTargetFolder] = useState(null); const flattenFolders = (items: LibraryFolderTree[], depth = 0): { id: number | null; name: string; depth: number }[] => { const result: { id: number | null; name: string; depth: number }[] = []; for (const item of items) { result.push({ id: item.id, name: item.name, depth }); if (item.children.length > 0) { result.push(...flattenFolders(item.children, depth + 1)); } } return result; }; const flatFolders = [{ id: null, name: 'Root (No Folder)', depth: 0 }, ...flattenFolders(folders)]; return (

Move {selectedFiles.length} File(s)

{flatFolders.map((folder) => ( ))}
); } // Link Folder Modal interface LinkFolderModalProps { folder: LibraryFolderTree; onClose: () => void; onLink: (update: LibraryFolderUpdate) => void; isLoading: boolean; } function LinkFolderModal({ folder, onClose, onLink, isLoading }: LinkFolderModalProps) { const [linkType, setLinkType] = useState<'project' | 'archive'>('project'); const [selectedId, setSelectedId] = useState( folder.project_id || folder.archive_id || null ); // Initialize linkType based on existing link useState(() => { if (folder.archive_id) setLinkType('archive'); }); const { data: projects } = useQuery({ queryKey: ['projects'], queryFn: () => api.getProjects(), }); const { data: archives } = useQuery({ queryKey: ['archives-for-link'], queryFn: () => api.getArchives(undefined, undefined, 100), }); const handleSave = () => { if (linkType === 'project') { onLink({ project_id: selectedId, archive_id: 0, // Unlink archive }); } else { onLink({ project_id: 0, // Unlink project archive_id: selectedId, }); } }; const handleUnlink = () => { onLink({ project_id: 0, archive_id: 0, }); }; const isLinked = folder.project_id || folder.archive_id; return (

Link Folder

Link "{folder.name}" to a project or archive for quick access.

{/* Link type selector */}
{/* Selection list */}
{linkType === 'project' ? ( projects && projects.length > 0 ? ( projects.map((project) => ( )) ) : (

No projects found

) ) : ( archives && archives.length > 0 ? ( archives.map((archive: Archive) => ( )) ) : (

No archives found

) )}
{isLinked && ( )}
); } // Upload Modal with Drag & Drop interface UploadModalProps { folderId: number | null; onClose: () => void; onUploadComplete: () => void; } interface UploadFile { file: File; status: 'pending' | 'uploading' | 'success' | 'error'; error?: string; } function UploadModal({ folderId, onClose, onUploadComplete }: UploadModalProps) { const [files, setFiles] = useState([]); const [isDragging, setIsDragging] = useState(false); const [isUploading, setIsUploading] = useState(false); const fileInputRef = useRef(null); const handleDragOver = (e: DragEvent) => { e.preventDefault(); setIsDragging(true); }; const handleDragLeave = (e: DragEvent) => { e.preventDefault(); setIsDragging(false); }; const handleDrop = (e: DragEvent) => { e.preventDefault(); setIsDragging(false); const droppedFiles = Array.from(e.dataTransfer.files); addFiles(droppedFiles); }; const handleFileSelect = (e: React.ChangeEvent) => { if (e.target.files) { addFiles(Array.from(e.target.files)); } }; const addFiles = (newFiles: File[]) => { const uploadFiles: UploadFile[] = newFiles.map((file) => ({ file, status: 'pending', })); setFiles((prev) => [...prev, ...uploadFiles]); }; const removeFile = (index: number) => { setFiles((prev) => prev.filter((_, i) => i !== index)); }; const handleUpload = async () => { if (files.length === 0) return; setIsUploading(true); for (let i = 0; i < files.length; i++) { if (files[i].status !== 'pending') continue; setFiles((prev) => prev.map((f, idx) => (idx === i ? { ...f, status: 'uploading' } : f)) ); try { await api.uploadLibraryFile(files[i].file, folderId); setFiles((prev) => prev.map((f, idx) => (idx === i ? { ...f, status: 'success' } : f)) ); } catch (err) { setFiles((prev) => prev.map((f, idx) => idx === i ? { ...f, status: 'error', error: err instanceof Error ? err.message : 'Upload failed' } : f ) ); } } setIsUploading(false); onUploadComplete(); // Auto-close modal after upload completes onClose(); }; const pendingCount = files.filter((f) => f.status === 'pending').length; const successCount = files.filter((f) => f.status === 'success').length; const errorCount = files.filter((f) => f.status === 'error').length; const allDone = files.length > 0 && pendingCount === 0 && !isUploading; return (

Upload Files

{/* Drop Zone */}
fileInputRef.current?.click()} className={`border-2 border-dashed rounded-lg p-8 text-center cursor-pointer transition-colors ${ isDragging ? 'border-bambu-green bg-bambu-green/10' : 'border-bambu-dark-tertiary hover:border-bambu-green/50' }`} >

{isDragging ? 'Drop files here' : 'Drag & drop files here'}

or click to browse

{/* File List */} {files.length > 0 && (
{files.map((uploadFile, index) => (

{uploadFile.file.name}

{(uploadFile.file.size / 1024 / 1024).toFixed(2)} MB

{uploadFile.status === 'pending' && ( )} {uploadFile.status === 'uploading' && ( )} {uploadFile.status === 'success' && ( )} {uploadFile.status === 'error' && ( )}
))}
)} {/* Summary */} {allDone && (

Upload complete: {successCount} succeeded {errorCount > 0 && , {errorCount} failed}

)}
{!allDone && ( )}
); } // Folder Tree Item interface FolderTreeItemProps { folder: LibraryFolderTree; selectedFolderId: number | null; onSelect: (id: number | null) => void; onDelete: (id: number) => void; onLink: (folder: LibraryFolderTree) => void; depth?: number; } function FolderTreeItem({ folder, selectedFolderId, onSelect, onDelete, onLink, depth = 0 }: FolderTreeItemProps) { const [expanded, setExpanded] = useState(true); const [showActions, setShowActions] = useState(false); const hasChildren = folder.children.length > 0; const isLinked = folder.project_id || folder.archive_id; return (
onSelect(folder.id)} > {hasChildren ? ( ) : (
)} {folder.name} {/* Link indicator - clickable to change link */} {isLinked && ( )} {folder.file_count > 0 && ( {folder.file_count} )} {/* Quick link button - always visible for unlinked folders */} {!isLinked && ( )}
e.stopPropagation()}>
{showActions && ( <>
setShowActions(false)} />
)}
{hasChildren && expanded && (
{folder.children.map((child) => ( ))}
)}
); } // Helper to check if a file is sliced (printable) function isSlicedFilename(filename: string): boolean { const lower = filename.toLowerCase(); return lower.endsWith('.gcode') || lower.includes('.gcode.'); } // File Card interface FileCardProps { file: LibraryFileListItem; isSelected: boolean; onSelect: (id: number) => void; onDelete: (id: number) => void; onDownload: (id: number) => void; onAddToQueue?: (id: number) => void; } function FileCard({ file, isSelected, onSelect, onDelete, onDownload, onAddToQueue }: FileCardProps) { const [showActions, setShowActions] = useState(false); return (
onSelect(file.id)} > {/* Thumbnail */}
{file.thumbnail_path ? ( {file.filename} ) : ( )} {/* Duplicate badge */} {file.duplicate_count > 0 && (
{file.duplicate_count}
)} {/* File type badge */}
{file.file_type.toUpperCase()}
{/* Info */}

{file.print_name || file.filename}

{formatFileSize(file.file_size)} {file.print_time_seconds && ( {formatDuration(file.print_time_seconds)} )}
{file.print_count > 0 && (
Printed {file.print_count}x
)}
{/* Actions */}
e.stopPropagation()}> {showActions && ( <>
setShowActions(false)} />
{onAddToQueue && isSlicedFilename(file.filename) && ( )}
)}
{/* Selection checkbox */}
{isSelected &&
}
); } export function FileManagerPage() { const queryClient = useQueryClient(); const { showToast } = useToast(); const [searchParams] = useSearchParams(); // Read folder ID from URL query parameter const folderIdFromUrl = searchParams.get('folder'); const initialFolderId = folderIdFromUrl ? parseInt(folderIdFromUrl, 10) : null; // State const [selectedFolderId, setSelectedFolderId] = useState(initialFolderId); const [selectedFiles, setSelectedFiles] = useState([]); const [showNewFolderModal, setShowNewFolderModal] = useState(false); const [showMoveModal, setShowMoveModal] = useState(false); const [showUploadModal, setShowUploadModal] = useState(false); const [linkFolder, setLinkFolder] = useState(null); const [deleteConfirm, setDeleteConfirm] = useState<{ type: 'file' | 'folder' | 'bulk'; id: number; count?: number } | null>(null); const [viewMode, setViewMode] = useState<'grid' | 'list'>(() => { return (localStorage.getItem('library-view-mode') as 'grid' | 'list') || 'grid'; }); // Filter and sort state const [searchQuery, setSearchQuery] = useState(''); const [filterType, setFilterType] = useState('all'); const [sortField, setSortField] = useState('date'); const [sortDirection, setSortDirection] = useState('desc'); // Update selectedFolderId when URL parameter changes (e.g., navigating from Project or Archive page) useEffect(() => { const folderParam = searchParams.get('folder'); const newFolderId = folderParam ? parseInt(folderParam, 10) : null; if (newFolderId !== selectedFolderId) { setSelectedFolderId(newFolderId); } }, [searchParams]); // Queries const { data: settings } = useQuery({ queryKey: ['settings'], queryFn: () => api.getSettings() as Promise, }); const { data: folders, isLoading: foldersLoading } = useQuery({ queryKey: ['library-folders'], queryFn: () => api.getLibraryFolders(), }); const { data: files, isLoading: filesLoading } = useQuery({ queryKey: ['library-files', selectedFolderId], queryFn: () => api.getLibraryFiles(selectedFolderId, selectedFolderId === null), }); const { data: stats } = useQuery({ queryKey: ['library-stats'], queryFn: () => api.getLibraryStats(), }); // Get unique file types for filter dropdown const fileTypes = useMemo(() => { if (!files) return []; const types = new Set(files.map((f) => f.file_type)); return Array.from(types).sort(); }, [files]); // Filter and sort files const filteredAndSortedFiles = useMemo(() => { if (!files) return []; let result = [...files]; // Apply search filter if (searchQuery.trim()) { const query = searchQuery.toLowerCase(); result = result.filter( (f) => f.filename.toLowerCase().includes(query) || (f.print_name && f.print_name.toLowerCase().includes(query)) ); } // Apply type filter if (filterType !== 'all') { result = result.filter((f) => f.file_type === filterType); } // Apply sorting result.sort((a, b) => { let comparison = 0; switch (sortField) { case 'name': comparison = (a.print_name || a.filename).localeCompare(b.print_name || b.filename); break; case 'date': comparison = new Date(a.created_at).getTime() - new Date(b.created_at).getTime(); break; case 'size': comparison = a.file_size - b.file_size; break; case 'type': comparison = a.file_type.localeCompare(b.file_type); break; case 'prints': comparison = a.print_count - b.print_count; break; } return sortDirection === 'asc' ? comparison : -comparison; }); return result; }, [files, searchQuery, filterType, sortField, sortDirection]); // Check if disk space is low const isDiskSpaceLow = useMemo(() => { if (!stats || !settings) return false; const thresholdBytes = (settings.library_disk_warning_gb || 5) * 1024 * 1024 * 1024; return stats.disk_free_bytes < thresholdBytes; }, [stats, settings]); // Mutations const createFolderMutation = useMutation({ mutationFn: (data: LibraryFolderCreate) => api.createLibraryFolder(data), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ['library-folders'] }); setShowNewFolderModal(false); showToast('Folder created', 'success'); }, onError: (error: Error) => showToast(error.message, 'error'), }); const deleteFolderMutation = useMutation({ mutationFn: (id: number) => api.deleteLibraryFolder(id), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); if (selectedFolderId === deleteConfirm?.id) { setSelectedFolderId(null); } setDeleteConfirm(null); showToast('Folder deleted', 'success'); }, onError: (error: Error) => { setDeleteConfirm(null); showToast(error.message, 'error'); }, }); const deleteFileMutation = useMutation({ mutationFn: (id: number) => api.deleteLibraryFile(id), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); setSelectedFiles((prev) => prev.filter((id) => id !== deleteConfirm?.id)); setDeleteConfirm(null); showToast('File deleted', 'success'); }, onError: (error: Error) => { setDeleteConfirm(null); showToast(error.message, 'error'); }, }); const moveFilesMutation = useMutation({ mutationFn: ({ fileIds, folderId }: { fileIds: number[]; folderId: number | null }) => api.moveLibraryFiles(fileIds, folderId), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-folders'] }); setSelectedFiles([]); setShowMoveModal(false); showToast('Files moved', 'success'); }, onError: (error: Error) => showToast(error.message, 'error'), }); const updateFolderMutation = useMutation({ mutationFn: ({ id, data }: { id: number; data: LibraryFolderUpdate }) => api.updateLibraryFolder(id, data), onSuccess: (_, variables) => { queryClient.invalidateQueries({ queryKey: ['library-folders'] }); // Invalidate project/archive folder queries so other pages see the update queryClient.invalidateQueries({ queryKey: ['project-folders'] }); queryClient.invalidateQueries({ queryKey: ['archive-folders'] }); setLinkFolder(null); const isUnlink = variables.data.project_id === 0 && variables.data.archive_id === 0; showToast(isUnlink ? 'Folder unlinked' : 'Folder linked', 'success'); }, onError: (error: Error) => showToast(error.message, 'error'), }); const addToQueueMutation = useMutation({ mutationFn: (fileIds: number[]) => api.addLibraryFilesToQueue(fileIds), onSuccess: (result) => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['queue'] }); setSelectedFiles([]); if (result.added.length > 0 && result.errors.length === 0) { showToast( `Added ${result.added.length} file${result.added.length > 1 ? 's' : ''} to queue`, 'success' ); } else if (result.added.length > 0 && result.errors.length > 0) { showToast( `Added ${result.added.length} file${result.added.length > 1 ? 's' : ''}, ${result.errors.length} failed`, 'success' ); } else { showToast(`Failed to add files: ${result.errors[0]?.error || 'Unknown error'}`, 'error'); } }, onError: (error: Error) => showToast(error.message, 'error'), }); // Helper to check if a file is sliced (printable) const isSlicedFile = useCallback((filename: string) => { const lower = filename.toLowerCase(); return lower.endsWith('.gcode') || lower.includes('.gcode.'); }, []); // Get sliced files from selection const selectedSlicedFiles = useMemo(() => { if (!files) return []; return files.filter(f => selectedFiles.includes(f.id) && isSlicedFile(f.filename)); }, [files, selectedFiles, isSlicedFile]); // Handlers const handleFileSelect = useCallback((id: number) => { // Always toggle selection (multi-select by default) setSelectedFiles((prev) => { return prev.includes(id) ? prev.filter((x) => x !== id) : [...prev, id]; }); }, []); const handleSelectAll = useCallback(() => { if (filteredAndSortedFiles.length > 0) { setSelectedFiles(filteredAndSortedFiles.map((f) => f.id)); } }, [filteredAndSortedFiles]); const handleDeselectAll = useCallback(() => { setSelectedFiles([]); }, []); const handleUploadComplete = () => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); }; const handleDownload = (id: number) => { window.open(api.getLibraryFileDownloadUrl(id), '_blank'); }; const handleDeleteConfirm = () => { if (!deleteConfirm) return; if (deleteConfirm.type === 'file') { deleteFileMutation.mutate(deleteConfirm.id); } else if (deleteConfirm.type === 'folder') { deleteFolderMutation.mutate(deleteConfirm.id); } else if (deleteConfirm.type === 'bulk') { // Bulk delete selected files api.bulkDeleteLibrary(selectedFiles, []).then(() => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); showToast(`Deleted ${selectedFiles.length} files`, 'success'); setSelectedFiles([]); setDeleteConfirm(null); }).catch((err) => { showToast(err.message, 'error'); setDeleteConfirm(null); }); } }; const handleViewModeChange = (mode: 'grid' | 'list') => { setViewMode(mode); localStorage.setItem('library-view-mode', mode); }; const isLoading = foldersLoading || filesLoading; return (
{/* Header */}

File Manager

Organize and manage your print files

{/* View mode toggle */}
{/* Disk space warning */} {isDiskSpaceLow && stats && settings && (

Low disk space warning

Only {formatFileSize(stats.disk_free_bytes)} free of {formatFileSize(stats.disk_total_bytes)} total. Threshold is set to {settings.library_disk_warning_gb} GB in settings.

)} {/* Stats bar */} {stats && (
Files: {stats.total_files}
Folders: {stats.total_folders}
Size: {formatFileSize(stats.total_size_bytes)}
Free: {formatFileSize(stats.disk_free_bytes)}
)} {/* Main content */}
{/* Folder sidebar */}

Folders

{/* All Files (root) */}
setSelectedFolderId(null)} > All Files
{/* Folder tree */} {folders?.map((folder) => ( setDeleteConfirm({ type: 'folder', id })} onLink={setLinkFolder} /> ))}
{/* Files area */}
{/* Search, Filter, Sort toolbar */} {files && files.length > 0 && (
{/* Search */}
setSearchQuery(e.target.value)} className="w-full pl-9 pr-3 py-1.5 bg-bambu-dark border border-bambu-dark-tertiary rounded text-sm text-white placeholder-bambu-gray focus:outline-none focus:border-bambu-green" />
{/* Type filter */}
{/* Sort */}
{/* Results count */} {(searchQuery || filterType !== 'all') && ( {filteredAndSortedFiles.length} of {files.length} files )}
)} {/* Selection toolbar */} {filteredAndSortedFiles.length > 0 && (
{/* Select all / Deselect all */} {selectedFiles.length === filteredAndSortedFiles.length && selectedFiles.length > 0 ? ( ) : ( )} {selectedFiles.length > 0 && ( <> {selectedFiles.length} selected
{selectedSlicedFiles.length > 0 && ( )} )}
)} {/* File grid/list */} {isLoading ? (

Loading files...

) : files?.length === 0 ? (

{selectedFolderId !== null ? 'Folder is empty' : 'No files yet'}

{selectedFolderId !== null ? 'Upload files or move files into this folder to get started.' : 'Upload files to start organizing your print-related files.'}

) : filteredAndSortedFiles.length === 0 ? (

No matching files

No files match your current search or filter criteria.

) : viewMode === 'grid' ? (
{filteredAndSortedFiles.map((file) => ( setDeleteConfirm({ type: 'file', id })} onDownload={handleDownload} onAddToQueue={(id) => addToQueueMutation.mutate([id])} /> ))}
) : (
{/* List header */}
Name
Type
Size
Prints
{/* List rows */} {filteredAndSortedFiles.map((file) => (
handleFileSelect(file.id)} > {/* Checkbox */}
{selectedFiles.includes(file.id) &&
}
{/* Name with thumbnail */}
{file.thumbnail_path ? ( ) : (
)}
{/* Hover preview */} {file.thumbnail_path && (
{file.filename}
)}
{file.print_name || file.filename}
{file.duplicate_count > 0 && (
{file.duplicate_count} duplicate(s)
)}
{/* Type */}
{file.file_type.toUpperCase()}
{/* Size */}
{formatFileSize(file.file_size)}
{/* Prints */}
{file.print_count > 0 ? `${file.print_count}x` : '-'}
{/* Actions */}
e.stopPropagation()}> {isSlicedFilename(file.filename) && ( )}
))}
)}
{/* Modals */} {showNewFolderModal && ( setShowNewFolderModal(false)} onSave={(data) => createFolderMutation.mutate(data)} isLoading={createFolderMutation.isPending} /> )} {showMoveModal && folders && ( setShowMoveModal(false)} onMove={(folderId) => moveFilesMutation.mutate({ fileIds: selectedFiles, folderId })} isLoading={moveFilesMutation.isPending} /> )} {showUploadModal && ( setShowUploadModal(false)} onUploadComplete={handleUploadComplete} /> )} {linkFolder && ( setLinkFolder(null)} onLink={(data) => updateFolderMutation.mutate({ id: linkFolder.id, data })} isLoading={updateFolderMutation.isPending} /> )} {deleteConfirm && ( setDeleteConfirm(null)} /> )}
); }