import { useState, useRef, useCallback, useMemo, useEffect } 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, Link2, Unlink, Archive as ArchiveIcon, Briefcase, Printer, Pencil, Play, Image, User, Box, RefreshCw, Lock, FolderSymlink, } from 'lucide-react'; import { api } from '../api/client'; import type { LibraryFolderTree, LibraryFileListItem, LibraryFolderCreate, LibraryFolderUpdate, ExternalFolderCreate, 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 { FileUploadModal } from '../components/FileUploadModal'; import { useToast } from '../contexts/ToastContext'; import { useIsMobile } from '../hooks/useIsMobile'; import { useAuth } from '../contexts/AuthContext'; import { formatDuration, parseUTCDate } from '../utils/date'; import { formatFileSize } from '../utils/file'; type SortField = 'name' | 'date' | 'size' | 'type' | 'prints'; type SortDirection = 'asc' | 'desc'; type TFunction = (key: string, options?: Record) => string; // 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 />
); } // External Folder Modal interface ExternalFolderModalProps { onClose: () => void; onSave: (data: ExternalFolderCreate) => void; isLoading: boolean; t: TFunction; } function ExternalFolderModal({ onClose, onSave, isLoading, t }: ExternalFolderModalProps) { const [name, setName] = useState(''); const [path, setPath] = useState(''); const [readonly, setReadonly] = useState(true); const [showHidden, setShowHidden] = useState(false); const handleSubmit = (e: React.FormEvent) => { e.preventDefault(); onSave({ name: name.trim(), external_path: path.trim(), readonly, show_hidden: showHidden, }); }; return (

{t('fileManager.linkExternalFolder')}

{t('fileManager.linkExternalFolderDescription')}

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.externalFolderNamePlaceholder')} autoFocus required />
setPath(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 font-mono text-sm" placeholder="/mnt/nas/3d-prints" required />

{t('fileManager.externalPathHelp')}

); } // 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) { // For files, separate the extension so users can only edit the base name // Handle compound extensions like .gcode.3mf const fileExtension = type === 'file' ? (currentName.match(/(\.gcode\.3mf|\.3mf|\.gcode)$/i)?.[1] ?? '') : ''; const baseName = type === 'file' && fileExtension ? currentName.slice(0, -fileExtension.length) : currentName; const [name, setName] = useState(baseName); const handleSubmit = (e: React.FormEvent) => { e.preventDefault(); const fullName = type === 'file' ? name.trim() + fileExtension : name.trim(); if (name.trim() && fullName !== currentName) { onSave(fullName); } }; return (

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

setName(e.target.value)} className="flex-1 bg-transparent px-3 py-2 text-white placeholder-bambu-gray focus:outline-none min-w-0" autoFocus required /> {fileExtension && ( {fileExtension} )}
); } // 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 && ( )}
); } // 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; const isExternal = folder.is_external; return (
onSelect(folder.id)} > {hasChildren ? ( ) : (
)} {isExternal ? ( ) : ( )} {folder.name} {/* Link indicator - clickable to change link */} {isLinked && ( )} {/* Read-only indicator for external folders */} {isExternal && folder.external_readonly && ( )} {folder.file_count > 0 && ( {folder.file_count} )} {/* Quick link button - always visible for unlinked folders */} {!isLinked && !isExternal && ( )}
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 [showExternalFolderModal, setShowExternalFolderModal] = 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 = (parseUTCDate(a.created_at)?.getTime() ?? 0) - (parseUTCDate(b.created_at)?.getTime() ?? 0); 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 createExternalFolderMutation = useMutation({ mutationFn: async (data: ExternalFolderCreate) => { const folder = await api.createExternalFolder(data); // Auto-scan after creation await api.scanExternalFolder(folder.id); return folder; }, onSuccess: (folder) => { queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); setShowExternalFolderModal(false); setSelectedFolderId(folder.id); showToast(t('fileManager.toast.externalFolderLinked'), 'success'); }, onError: (error: Error) => showToast(error.message, 'error'), }); const scanExternalFolderMutation = useMutation({ mutationFn: (folderId: number) => api.scanExternalFolder(folderId), onSuccess: (result) => { queryClient.invalidateQueries({ queryKey: ['library-files'] }); queryClient.invalidateQueries({ queryKey: ['library-folders'] }); queryClient.invalidateQueries({ queryKey: ['library-stats'] }); showToast(t('fileManager.toast.folderScanned', { added: result.added, removed: result.removed }), '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; // Find the selected folder in the tree to check external status const selectedFolder = useMemo(() => { if (!selectedFolderId || !folders) return null; const findFolder = (items: LibraryFolderTree[]): LibraryFolderTree | null => { for (const item of items) { if (item.id === selectedFolderId) return item; const found = findFolder(item.children); if (found) return found; } return null; }; return findFolder(folders); }, [selectedFolderId, folders]); 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 */}
{/* External folder info bar */} {selectedFolder?.is_external && (
{t('fileManager.externalFolder')} {selectedFolder.external_readonly && ( {t('fileManager.readOnly')} )}

{selectedFolder.external_path}

)} {/* 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} /> )} {showExternalFolderModal && ( setShowExternalFolderModal(false)} onSave={(data) => createExternalFolderMutation.mutate(data)} isLoading={createExternalFolderMutation.isPending} t={t} /> )} {showMoveModal && folders && ( setShowMoveModal(false)} onMove={(folderId) => moveFilesMutation.mutate({ fileIds: selectedFiles, folderId })} isLoading={moveFilesMutation.isPending} t={t} /> )} {showUploadModal && ( setShowUploadModal(false)} onUploadComplete={handleUploadComplete} /> )} {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} /> )}
); }