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 { useTranslation } from 'react-i18next'; import { FolderOpen, Loader2, Plus, Upload, Trash2, Download, MoreVertical, ChevronRight, FolderPlus, FileBox, Clock, HardDrive, File, MoveRight, CheckSquare, Square, LayoutGrid, List, Search, SortAsc, SortDesc, AlertTriangle, Filter, X, CheckCircle, XCircle, Link2, Unlink, Archive as ArchiveIcon, Briefcase, Printer, Pencil, Play, Image, User, Box, } from 'lucide-react'; import { api } from '../api/client'; import type { LibraryFolderTree, LibraryFileListItem, LibraryFolderCreate, LibraryFolderUpdate, AppSettings, Archive, Permission, } from '../api/client'; import { Button } from '../components/Button'; import { ConfirmModal } from '../components/ConfirmModal'; import { PrintModal } from '../components/PrintModal'; import { ModelViewerModal } from '../components/ModelViewerModal'; import { useToast } from '../contexts/ToastContext'; import { useIsMobile } from '../hooks/useIsMobile'; import { useAuth } from '../contexts/AuthContext'; type SortField = 'name' | 'date' | 'size' | 'type' | 'prints'; type SortDirection = 'asc' | 'desc'; type TFunction = (key: string, options?: Record) => string; // 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; t: TFunction; } function NewFolderModal({ parentId, onClose, onSave, isLoading, t }: NewFolderModalProps) { const [name, setName] = useState(''); const handleSubmit = (e: React.FormEvent) => { e.preventDefault(); onSave({ name: name.trim(), parent_id: parentId }); }; return (

{t('fileManager.newFolder')}

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={t('fileManager.folderNamePlaceholder')} autoFocus required />
); } // Rename Modal interface RenameModalProps { type: 'file' | 'folder'; currentName: string; onClose: () => void; onSave: (newName: string) => void; isLoading: boolean; t: TFunction; } function RenameModal({ type, currentName, onClose, onSave, isLoading, t }: RenameModalProps) { const [name, setName] = useState(currentName); const handleSubmit = (e: React.FormEvent) => { e.preventDefault(); if (name.trim() && name.trim() !== currentName) { onSave(name.trim()); } }; return (

{type === 'file' ? t('fileManager.renameFile') : t('fileManager.renameFolder')}

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" autoFocus required />
); } // Move Files Modal interface MoveFilesModalProps { folders: LibraryFolderTree[]; selectedFiles: number[]; currentFolderId: number | null; onClose: () => void; onMove: (folderId: number | null) => void; isLoading: boolean; t: TFunction; } function MoveFilesModal({ folders, selectedFiles, currentFolderId, onClose, onMove, isLoading, t }: 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: t('fileManager.rootNoFolder'), depth: 0 }, ...flattenFolders(folders)]; return (

{t('fileManager.moveFiles', { count: selectedFiles.length })}

{flatFolders.map((folder) => ( ))}
); } // Link Folder Modal interface LinkFolderModalProps { folder: LibraryFolderTree; onClose: () => void; onLink: (update: LibraryFolderUpdate) => void; isLoading: boolean; t: TFunction; } function LinkFolderModal({ folder, onClose, onLink, isLoading, t }: 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 (

{t('fileManager.linkFolder')}

{t('fileManager.linkFolderDescription', { name: folder.name })}

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

{t('fileManager.noProjectsFound')}

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

{t('fileManager.noArchivesFound')}

) )}
{isLinked && ( )}
); } // Upload Modal with Drag & Drop interface UploadModalProps { folderId: number | null; onClose: () => void; onUploadComplete: () => void; t: TFunction; } interface UploadFile { file: File; status: 'pending' | 'uploading' | 'success' | 'error'; error?: string; isZip?: boolean; is3mf?: boolean; extractedCount?: number; } function UploadModal({ folderId, onClose, onUploadComplete, t }: UploadModalProps) { const [files, setFiles] = useState([]); const [isDragging, setIsDragging] = useState(false); const [isUploading, setIsUploading] = useState(false); const [preserveZipStructure, setPreserveZipStructure] = useState(true); const [createFolderFromZip, setCreateFolderFromZip] = useState(false); const [generateStlThumbnails, setGenerateStlThumbnails] = useState(true); 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', isZip: file.name.toLowerCase().endsWith('.zip'), is3mf: file.name.toLowerCase().endsWith('.3mf'), })); setFiles((prev) => [...prev, ...uploadFiles]); }; const removeFile = (index: number) => { setFiles((prev) => prev.filter((_, i) => i !== index)); }; const hasZipFiles = files.some((f) => f.isZip && f.status === 'pending'); const hasStlFiles = files.some((f) => f.file.name.toLowerCase().endsWith('.stl') && f.status === 'pending'); const has3mfFiles = files.some((f) => f.is3mf && f.status === 'pending'); const handleUpload = async () => { if (files.length === 0) return; setIsUploading(true); // Handle all files with library upload (ZIP and regular files including .3mf) 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 { if (files[i].isZip) { // Extract ZIP file const result = await api.extractZipFile(files[i].file, folderId, preserveZipStructure, createFolderFromZip, generateStlThumbnails); setFiles((prev) => prev.map((f, idx) => idx === i ? { ...f, status: result.errors.length > 0 && result.extracted === 0 ? 'error' : 'success', extractedCount: result.extracted, error: result.errors.length > 0 ? `${result.errors.length} files failed` : undefined, } : f ) ); } else { // Regular file upload (STL, .3mf, etc.) - .3mf files automatically get metadata extracted await api.uploadLibraryFile(files[i].file, folderId, generateStlThumbnails); 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 (

{t('fileManager.uploadFiles')}

{/* 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 ? t('fileManager.dropFilesHere') : t('fileManager.dragDropFiles')}

{t('fileManager.orClickToBrowse')}

{t('fileManager.allFileTypesSupported')}

{/* ZIP Options */} {hasZipFiles && (

{t('fileManager.zipFilesDetected')}

{t('fileManager.zipExtractOptions')}

)} {/* 3MF File Info - Advanced Extraction */} {has3mfFiles && (

{t('fileManager.threemfDetected')}

{t('fileManager.threemfExtractionInfo')}

)} {/* STL Thumbnail Options - show for STL files or ZIP files (which may contain STLs) */} {(hasStlFiles || hasZipFiles) && (

{t('fileManager.stlThumbnailGeneration')}

{hasZipFiles && !hasStlFiles ? t('fileManager.zipMayContainStl') : t('fileManager.thumbnailsCanBeGenerated')}

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

{uploadFile.file.name}

{(uploadFile.file.size / 1024 / 1024).toFixed(2)} MB {uploadFile.isZip && uploadFile.status === 'pending' && ( • {t('fileManager.willBeExtracted')} )} {uploadFile.extractedCount !== undefined && ( • {t('fileManager.filesExtracted', { count: uploadFile.extractedCount })} )}

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

{t('fileManager.uploadComplete', { succeeded: successCount })} {errorCount > 0 && , {t('fileManager.uploadFailed', { count: errorCount })}}

)}
{!allDone && ( )}
); } // Folder Tree Item interface FolderTreeItemProps { folder: LibraryFolderTree; selectedFolderId: number | null; onSelect: (id: number | null) => void; onDelete: (id: number) => void; onLink: (folder: LibraryFolderTree) => void; onRename: (folder: LibraryFolderTree) => void; depth?: number; wrapNames?: boolean; hasPermission: (permission: Permission) => boolean; t: TFunction; } function FolderTreeItem({ folder, selectedFolderId, onSelect, onDelete, onLink, onRename, depth = 0, wrapNames = false, hasPermission, t }: 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; isMobile: boolean; onSelect: (id: number) => void; onDelete: (id: number) => void; onDownload: (id: number) => void; onAddToQueue?: (id: number) => void; onPrint?: (file: LibraryFileListItem) => void; onPreview3d?: (file: LibraryFileListItem) => void; onRename?: (file: LibraryFileListItem) => void; onGenerateThumbnail?: (file: LibraryFileListItem) => void; thumbnailVersion?: number; hasPermission: (permission: Permission) => boolean; canModify: (resource: 'queue' | 'archives' | 'library', action: 'update' | 'delete' | 'reprint', createdById: number | null | undefined) => boolean; authEnabled: boolean; t: TFunction; } function FileCard({ file, isSelected, isMobile, onSelect, onDelete, onDownload, onAddToQueue, onPrint, onPreview3d, onRename, onGenerateThumbnail, thumbnailVersion, hasPermission, canModify, authEnabled, t }: FileCardProps) { const [showActions, setShowActions] = useState(false); return (
onSelect(file.id)} > {/* Thumbnail */}
{file.thumbnail_path ? ( {file.filename} ) : ( )} {/* 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.sliced_for_model && (
{file.sliced_for_model}
)} {file.print_count > 0 && (
{t('fileManager.printedCount', { count: file.print_count })}
)} {authEnabled && file.created_by_username && (
{file.created_by_username}
)}
{/* Actions - always visible on mobile, hover on desktop */}
e.stopPropagation()}> {showActions && ( <>
setShowActions(false)} />
{onPrint && isSlicedFilename(file.filename) && ( )} {onAddToQueue && isSlicedFilename(file.filename) && ( )} {onPreview3d && (file.file_type === '3mf' || file.file_type === 'gcode' || file.file_type === 'stl') && ( )} {onRename && ( )} {onGenerateThumbnail && file.file_type === 'stl' && ( )}
)}
{/* Selection checkbox - always visible on mobile, hover on desktop */}
{isSelected &&
}
); } export function FileManagerPage() { const { t } = useTranslation(); const queryClient = useQueryClient(); const { showToast } = useToast(); const { hasPermission, hasAnyPermission, canModify, authEnabled } = useAuth(); 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 [printFile, setPrintFile] = useState(null); const [printMultiFile, setPrintMultiFile] = useState(null); const [scheduleFile, setScheduleFile] = useState(null); const [renameItem, setRenameItem] = useState<{ type: 'file' | 'folder'; id: number; name: string } | null>(null); const [thumbnailVersions, setThumbnailVersions] = useState>({}); const [viewerFile, setViewerFile] = useState(null); const [viewMode, setViewMode] = useState<'grid' | 'list'>(() => { return (localStorage.getItem('library-view-mode') as 'grid' | 'list') || 'grid'; }); const [wrapFolderNames, setWrapFolderNames] = useState(() => { return localStorage.getItem('library-wrap-folders') === 'true'; }); // Resizable sidebar state const [sidebarWidth, setSidebarWidth] = useState(() => { const saved = localStorage.getItem('library-sidebar-width'); return saved ? parseInt(saved, 10) : 256; // Default w-64 = 256px }); const [isResizing, setIsResizing] = useState(false); const sidebarRef = useRef(null); // Handle sidebar resize useEffect(() => { if (!isResizing) return; // Prevent text selection during resize document.body.style.userSelect = 'none'; document.body.style.cursor = 'col-resize'; const handleMouseMove = (e: MouseEvent) => { if (!sidebarRef.current) return; const containerRect = sidebarRef.current.parentElement?.getBoundingClientRect(); if (!containerRect) return; // Calculate new width based on mouse position relative to container const newWidth = e.clientX - containerRect.left; // Clamp between 200px and 500px const clampedWidth = Math.min(500, Math.max(200, newWidth)); setSidebarWidth(clampedWidth); }; const handleMouseUp = () => { setIsResizing(false); document.body.style.userSelect = ''; document.body.style.cursor = ''; // Save to localStorage localStorage.setItem('library-sidebar-width', String(sidebarWidth)); }; document.addEventListener('mousemove', handleMouseMove); document.addEventListener('mouseup', handleMouseUp); return () => { document.removeEventListener('mousemove', handleMouseMove); document.removeEventListener('mouseup', handleMouseUp); document.body.style.userSelect = ''; document.body.style.cursor = ''; }; }, [isResizing, sidebarWidth]); // Filter and sort state (persist sort preferences to localStorage) const [searchQuery, setSearchQuery] = useState(''); const [filterType, setFilterType] = useState('all'); const [filterUsername, setFilterUsername] = useState(''); const [sortField, setSortField] = useState(() => { const saved = localStorage.getItem('library-sort-field'); return (saved as SortField) || 'name'; }); const [sortDirection, setSortDirection] = useState(() => { const saved = localStorage.getItem('library-sort-direction'); return (saved as SortDirection) || 'asc'; }); // Mobile detection for touch-friendly UI const isMobile = useIsMobile(); // Update selectedFolderId when URL parameter changes (e.g., navigating from Project or Archive page) useEffect(() => { const folderParam = searchParams.get('folder'); if (folderParam) { const newFolderId = parseInt(folderParam, 10); 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 users for the username filter autocomplete const { data: users } = useQuery({ queryKey: ['users'], queryFn: () => api.getUsers(), }); // 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 username filter if (filterUsername.trim()) { const query = filterUsername.toLowerCase(); result = result.filter( (f) => f.created_by_username && f.created_by_username.toLowerCase().includes(query) ); } // 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, filterUsername, 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(t('fileManager.toast.folderCreated'), '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(t('fileManager.toast.folderDeleted'), '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(t('fileManager.toast.fileDeleted'), 'success'); }, onError: (error: Error) => { setDeleteConfirm(null); showToast(error.message, 'error'); }, }); const bulkDeleteMutation = useMutation({ mutationFn: (fileIds: number[]) => api.bulkDeleteLibrary(fileIds, []), onSuccess: (_, fileIds) => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); showToast(t('fileManager.toast.filesDeleted', { count: fileIds.length }), 'success'); setSelectedFiles([]); setDeleteConfirm(null); }, 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(t('fileManager.toast.filesMoved'), '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 ? t('fileManager.toast.folderUnlinked') : t('fileManager.toast.folderLinked'), 'success'); }, onError: (error: Error) => showToast(error.message, 'error'), }); const renameFileMutation = useMutation({ mutationFn: ({ id, filename }: { id: number; filename: string }) => api.updateLibraryFile(id, { filename }), onSuccess: () => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); setRenameItem(null); showToast(t('fileManager.toast.fileRenamed'), 'success'); }, onError: (error: Error) => { setRenameItem(null); showToast(error.message, 'error'); }, }); const renameFolderMutation = useMutation({ mutationFn: ({ id, name }: { id: number; name: string }) => api.updateLibraryFolder(id, { name }), onSuccess: () => { // Invalidate both folders and files - files may display folder info queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-files'] }); setRenameItem(null); showToast(t('fileManager.toast.folderRenamed'), 'success'); }, onError: (error: Error) => { setRenameItem(null); showToast(error.message, 'error'); }, }); const batchThumbnailMutation = useMutation({ mutationFn: () => api.batchGenerateStlThumbnails({ all_missing: true }), onSuccess: (result) => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); // Update thumbnail versions for cache busting if (result.succeeded > 0) { const now = Date.now(); const newVersions: Record = {}; result.results.forEach((r) => { if (r.success) { newVersions[r.file_id] = now; } }); setThumbnailVersions((prev) => ({ ...prev, ...newVersions })); } if (result.succeeded > 0 && result.failed === 0) { showToast(t('fileManager.toast.thumbnailsGenerated', { count: result.succeeded }), 'success'); } else if (result.succeeded > 0 && result.failed > 0) { showToast(t('fileManager.toast.thumbnailsGeneratedPartial', { succeeded: result.succeeded, failed: result.failed }), 'success'); } else if (result.processed === 0) { showToast(t('fileManager.toast.noStlMissingThumbnails'), 'info'); } else { showToast(t('fileManager.toast.failedToGenerateThumbnails', { error: result.results[0]?.error || 'Unknown error' }), 'error'); } }, onError: (error: Error) => showToast(error.message, 'error'), }); const singleThumbnailMutation = useMutation({ mutationFn: (fileId: number) => api.batchGenerateStlThumbnails({ file_ids: [fileId] }), onSuccess: (result) => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); // Update thumbnail version for cache busting if (result.succeeded > 0) { const fileId = result.results[0]?.file_id; if (fileId) { setThumbnailVersions((prev) => ({ ...prev, [fileId]: Date.now() })); } showToast(t('fileManager.toast.thumbnailGenerated'), 'success'); } else { showToast(t('fileManager.toast.failedToGenerateThumbnail', { error: result.results[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) => { api.downloadLibraryFile(id).catch((err) => { console.error('Library file download failed:', err); }); }; 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') { bulkDeleteMutation.mutate(selectedFiles); } }; const isDeleting = deleteFolderMutation.isPending || deleteFileMutation.isPending || bulkDeleteMutation.isPending; const handleViewModeChange = (mode: 'grid' | 'list') => { setViewMode(mode); localStorage.setItem('library-view-mode', mode); }; const isLoading = foldersLoading || filesLoading; return (
{/* Header */}

{t('fileManager.title')}

{t('fileManager.subtitle')}

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

{t('fileManager.lowDiskSpaceWarning')}

{t('fileManager.lowDiskSpaceDetails', { free: formatFileSize(stats.disk_free_bytes), total: formatFileSize(stats.disk_total_bytes), threshold: settings.library_disk_warning_gb })}

)} {/* Stats bar */} {stats && (
{t('fileManager.files')}: {stats.total_files}
{t('fileManager.folders')}: {stats.total_folders}
{t('fileManager.size')}: {formatFileSize(stats.total_size_bytes)}
{t('fileManager.free')}: {formatFileSize(stats.disk_free_bytes)}
)} {/* Main content */}
{/* Mobile folder selector */}
{/* Folder sidebar - resizable, hidden on mobile */}
{/* Resize handle - drag to resize, double-click to reset */}
{ e.preventDefault(); setIsResizing(true); }} onDoubleClick={() => { setSidebarWidth(256); // Reset to default w-64 localStorage.setItem('library-sidebar-width', '256'); }} title={t('fileManager.dragToResizeTooltip')} > {/* Grip dots */}

{t('fileManager.folders')}

{/* All Files (root) */}
setSelectedFolderId(null)} > {t('fileManager.allFiles')}
{/* Folder tree */} {folders?.map((folder) => ( setDeleteConfirm({ type: 'folder', id })} onLink={setLinkFolder} onRename={(f) => setRenameItem({ type: 'folder', id: f.id, name: f.name })} wrapNames={wrapFolderNames} hasPermission={hasPermission} t={t} /> ))}
{/* Files area */}
{/* Search, Filter, Sort toolbar - sticky on mobile for easier access */} {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 */}
{/* Username filter with autocomplete - only show when auth is enabled */} {authEnabled && (
setFilterUsername(e.target.value)} list="usernames-list" className={`w-32 sm:w-40 px-2 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 ${filterUsername ? 'pr-7' : ''}`} style={filterUsername ? { WebkitAppearance: 'none', MozAppearance: 'textfield' } : undefined} /> {filterUsername && ( )} {users?.map((user) => (
)} {/* Sort */}
{/* Results count */} {(searchQuery || filterType !== 'all' || filterUsername) && ( {t('fileManager.resultsCount', { showing: filteredAndSortedFiles.length, total: files.length })} )}
)} {/* Selection toolbar - sticky on mobile below search bar */} {filteredAndSortedFiles.length > 0 && (
{/* Select all / Deselect all */} {selectedFiles.length === filteredAndSortedFiles.length && selectedFiles.length > 0 ? ( ) : ( )} {selectedFiles.length > 0 && ( <> {t('fileManager.selected', { count: selectedFiles.length })}
{selectedSlicedFiles.length === 1 && ( )} {selectedSlicedFiles.length === 1 && ( )}
)}
)} {/* File grid/list */} {isLoading ? (

{t('fileManager.loadingFiles')}

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

{selectedFolderId !== null ? t('fileManager.folderIsEmpty') : t('fileManager.noFilesYet')}

{selectedFolderId !== null ? t('fileManager.folderEmptyDescription') : t('fileManager.noFilesDescription')}

) : filteredAndSortedFiles.length === 0 ? (

{t('fileManager.noMatchingFiles')}

{t('fileManager.noMatchingFilesDescription')}

) : viewMode === 'grid' ? (
{filteredAndSortedFiles.map((file) => ( setDeleteConfirm({ type: 'file', id })} onDownload={handleDownload} onAddToQueue={(id) => { const file = files?.find(f => f.id === id); if (file) setScheduleFile(file); }} onPrint={setPrintFile} onPreview3d={setViewerFile} onRename={(f) => setRenameItem({ type: 'file', id: f.id, name: f.filename })} onGenerateThumbnail={(f) => singleThumbnailMutation.mutate(f.id)} thumbnailVersion={thumbnailVersions[file.id]} hasPermission={hasPermission} canModify={canModify} authEnabled={authEnabled} /> ))}
) : (
{/* List header - hidden on mobile, show simplified on small screens */}
{t('common.name')}
{authEnabled &&
{t('fileManager.uploadedBy', { defaultValue: 'Uploaded By' })}
}
{t('common.type')}
{t('fileManager.size')}
{t('fileManager.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}
{/* Uploaded By - only show when auth is enabled */} {authEnabled && (
{file.created_by_username ? ( <> {file.created_by_username} ) : ( '-' )}
)} {/* 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) && ( <> )} {(file.file_type === '3mf' || file.file_type === 'gcode' || file.file_type === 'stl') && ( )} {file.file_type === 'stl' && ( )}
))}
)}
{/* Modals */} {showNewFolderModal && ( setShowNewFolderModal(false)} onSave={(data) => createFolderMutation.mutate(data)} isLoading={createFolderMutation.isPending} t={t} /> )} {showMoveModal && folders && ( setShowMoveModal(false)} onMove={(folderId) => moveFilesMutation.mutate({ fileIds: selectedFiles, folderId })} isLoading={moveFilesMutation.isPending} t={t} /> )} {showUploadModal && ( setShowUploadModal(false)} onUploadComplete={handleUploadComplete} t={t} /> )} {linkFolder && ( setLinkFolder(null)} onLink={(data) => updateFolderMutation.mutate({ id: linkFolder.id, data })} isLoading={updateFolderMutation.isPending} t={t} /> )} {deleteConfirm && ( setDeleteConfirm(null)} /> )} {printFile && ( setPrintFile(null)} onSuccess={() => { setPrintFile(null); queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['archives'] }); }} /> )} {printMultiFile && ( setPrintMultiFile(null)} onSuccess={() => { setPrintMultiFile(null); setSelectedFiles([]); queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['archives'] }); }} /> )} {scheduleFile && ( setScheduleFile(null)} onSuccess={() => { setScheduleFile(null); setSelectedFiles([]); queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['queue'] }); queryClient.invalidateQueries({ queryKey: ['archives'] }); }} /> )} {viewerFile && ( setViewerFile(null)} /> )} {renameItem && ( setRenameItem(null)} onSave={(newName) => { if (renameItem.type === 'file') { renameFileMutation.mutate({ id: renameItem.id, filename: newName }); } else { renameFolderMutation.mutate({ id: renameItem.id, name: newName }); } }} isLoading={renameFileMutation.isPending || renameFolderMutation.isPending} t={t} /> )}
); }