FileManagerPage.tsx 62 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600
  1. import { useState, useRef, useCallback, useMemo, useEffect, type DragEvent } from 'react';
  2. import { useSearchParams } from 'react-router-dom';
  3. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  4. import {
  5. FolderOpen,
  6. Loader2,
  7. Plus,
  8. Upload,
  9. Trash2,
  10. Download,
  11. MoreVertical,
  12. ChevronRight,
  13. FolderPlus,
  14. FileBox,
  15. Clock,
  16. HardDrive,
  17. Copy,
  18. File,
  19. MoveRight,
  20. CheckSquare,
  21. Square,
  22. LayoutGrid,
  23. List,
  24. Search,
  25. SortAsc,
  26. SortDesc,
  27. AlertTriangle,
  28. Filter,
  29. X,
  30. CheckCircle,
  31. XCircle,
  32. Link2,
  33. Unlink,
  34. Archive as ArchiveIcon,
  35. Briefcase,
  36. Printer,
  37. } from 'lucide-react';
  38. import { api } from '../api/client';
  39. import type {
  40. LibraryFolderTree,
  41. LibraryFileListItem,
  42. LibraryFolderCreate,
  43. LibraryFolderUpdate,
  44. AppSettings,
  45. Archive,
  46. } from '../api/client';
  47. import { Button } from '../components/Button';
  48. import { ConfirmModal } from '../components/ConfirmModal';
  49. import { useToast } from '../contexts/ToastContext';
  50. type SortField = 'name' | 'date' | 'size' | 'type' | 'prints';
  51. type SortDirection = 'asc' | 'desc';
  52. // Utility to format file size
  53. function formatFileSize(bytes: number): string {
  54. if (bytes < 1024) return `${bytes} B`;
  55. if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
  56. if (bytes < 1024 * 1024 * 1024) return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
  57. return `${(bytes / (1024 * 1024 * 1024)).toFixed(2)} GB`;
  58. }
  59. // Utility to format duration
  60. function formatDuration(seconds: number | null): string {
  61. if (!seconds) return '-';
  62. const hours = Math.floor(seconds / 3600);
  63. const mins = Math.floor((seconds % 3600) / 60);
  64. if (hours > 0) return `${hours}h ${mins}m`;
  65. return `${mins}m`;
  66. }
  67. // New Folder Modal
  68. interface NewFolderModalProps {
  69. parentId: number | null;
  70. onClose: () => void;
  71. onSave: (data: LibraryFolderCreate) => void;
  72. isLoading: boolean;
  73. }
  74. function NewFolderModal({ parentId, onClose, onSave, isLoading }: NewFolderModalProps) {
  75. const [name, setName] = useState('');
  76. const handleSubmit = (e: React.FormEvent) => {
  77. e.preventDefault();
  78. onSave({ name: name.trim(), parent_id: parentId });
  79. };
  80. return (
  81. <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
  82. <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-sm border border-bambu-dark-tertiary">
  83. <div className="p-4 border-b border-bambu-dark-tertiary">
  84. <h2 className="text-lg font-semibold text-white">New Folder</h2>
  85. </div>
  86. <form onSubmit={handleSubmit} className="p-4 space-y-4">
  87. <div>
  88. <label className="block text-sm font-medium text-white mb-1">
  89. Folder Name
  90. </label>
  91. <input
  92. type="text"
  93. value={name}
  94. onChange={(e) => setName(e.target.value)}
  95. 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"
  96. placeholder="e.g., Functional Parts"
  97. autoFocus
  98. required
  99. />
  100. </div>
  101. <div className="flex justify-end gap-2 pt-2">
  102. <Button type="button" variant="secondary" onClick={onClose}>
  103. Cancel
  104. </Button>
  105. <Button type="submit" disabled={!name.trim() || isLoading}>
  106. {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : 'Create'}
  107. </Button>
  108. </div>
  109. </form>
  110. </div>
  111. </div>
  112. );
  113. }
  114. // Move Files Modal
  115. interface MoveFilesModalProps {
  116. folders: LibraryFolderTree[];
  117. selectedFiles: number[];
  118. currentFolderId: number | null;
  119. onClose: () => void;
  120. onMove: (folderId: number | null) => void;
  121. isLoading: boolean;
  122. }
  123. function MoveFilesModal({ folders, selectedFiles, currentFolderId, onClose, onMove, isLoading }: MoveFilesModalProps) {
  124. const [targetFolder, setTargetFolder] = useState<number | null>(null);
  125. const flattenFolders = (items: LibraryFolderTree[], depth = 0): { id: number | null; name: string; depth: number }[] => {
  126. const result: { id: number | null; name: string; depth: number }[] = [];
  127. for (const item of items) {
  128. result.push({ id: item.id, name: item.name, depth });
  129. if (item.children.length > 0) {
  130. result.push(...flattenFolders(item.children, depth + 1));
  131. }
  132. }
  133. return result;
  134. };
  135. const flatFolders = [{ id: null, name: 'Root (No Folder)', depth: 0 }, ...flattenFolders(folders)];
  136. return (
  137. <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
  138. <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-sm border border-bambu-dark-tertiary">
  139. <div className="p-4 border-b border-bambu-dark-tertiary">
  140. <h2 className="text-lg font-semibold text-white">Move {selectedFiles.length} File(s)</h2>
  141. </div>
  142. <div className="p-4 space-y-4">
  143. <div className="max-h-64 overflow-y-auto space-y-1">
  144. {flatFolders.map((folder) => (
  145. <button
  146. key={folder.id ?? 'root'}
  147. onClick={() => setTargetFolder(folder.id)}
  148. disabled={folder.id === currentFolderId}
  149. className={`w-full text-left px-3 py-2 rounded transition-colors flex items-center gap-2 ${
  150. targetFolder === folder.id
  151. ? 'bg-bambu-green/20 text-bambu-green'
  152. : folder.id === currentFolderId
  153. ? 'opacity-50 cursor-not-allowed text-bambu-gray'
  154. : 'hover:bg-bambu-dark text-white'
  155. }`}
  156. style={{ paddingLeft: `${12 + folder.depth * 16}px` }}
  157. >
  158. <FolderOpen className="w-4 h-4" />
  159. {folder.name}
  160. {folder.id === currentFolderId && <span className="text-xs text-bambu-gray ml-auto">(current)</span>}
  161. </button>
  162. ))}
  163. </div>
  164. <div className="flex justify-end gap-2 pt-2">
  165. <Button type="button" variant="secondary" onClick={onClose}>
  166. Cancel
  167. </Button>
  168. <Button onClick={() => onMove(targetFolder)} disabled={isLoading}>
  169. {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : 'Move'}
  170. </Button>
  171. </div>
  172. </div>
  173. </div>
  174. </div>
  175. );
  176. }
  177. // Link Folder Modal
  178. interface LinkFolderModalProps {
  179. folder: LibraryFolderTree;
  180. onClose: () => void;
  181. onLink: (update: LibraryFolderUpdate) => void;
  182. isLoading: boolean;
  183. }
  184. function LinkFolderModal({ folder, onClose, onLink, isLoading }: LinkFolderModalProps) {
  185. const [linkType, setLinkType] = useState<'project' | 'archive'>('project');
  186. const [selectedId, setSelectedId] = useState<number | null>(
  187. folder.project_id || folder.archive_id || null
  188. );
  189. // Initialize linkType based on existing link
  190. useState(() => {
  191. if (folder.archive_id) setLinkType('archive');
  192. });
  193. const { data: projects } = useQuery({
  194. queryKey: ['projects'],
  195. queryFn: () => api.getProjects(),
  196. });
  197. const { data: archives } = useQuery({
  198. queryKey: ['archives-for-link'],
  199. queryFn: () => api.getArchives(undefined, undefined, 100),
  200. });
  201. const handleSave = () => {
  202. if (linkType === 'project') {
  203. onLink({
  204. project_id: selectedId,
  205. archive_id: 0, // Unlink archive
  206. });
  207. } else {
  208. onLink({
  209. project_id: 0, // Unlink project
  210. archive_id: selectedId,
  211. });
  212. }
  213. };
  214. const handleUnlink = () => {
  215. onLink({
  216. project_id: 0,
  217. archive_id: 0,
  218. });
  219. };
  220. const isLinked = folder.project_id || folder.archive_id;
  221. return (
  222. <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
  223. <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-md border border-bambu-dark-tertiary">
  224. <div className="p-4 border-b border-bambu-dark-tertiary flex items-center justify-between">
  225. <h2 className="text-lg font-semibold text-white flex items-center gap-2">
  226. <Link2 className="w-5 h-5 text-bambu-green" />
  227. Link Folder
  228. </h2>
  229. <button onClick={onClose} className="p-1 hover:bg-bambu-dark rounded">
  230. <X className="w-5 h-5 text-bambu-gray" />
  231. </button>
  232. </div>
  233. <div className="p-4 space-y-4">
  234. <p className="text-sm text-bambu-gray">
  235. Link "<span className="text-white">{folder.name}</span>" to a project or archive for quick access.
  236. </p>
  237. {/* Link type selector */}
  238. <div className="flex gap-2">
  239. <button
  240. onClick={() => { setLinkType('project'); setSelectedId(null); }}
  241. className={`flex-1 flex items-center justify-center gap-2 px-3 py-2 rounded-lg border transition-colors ${
  242. linkType === 'project'
  243. ? 'border-bambu-green bg-bambu-green/10 text-bambu-green'
  244. : 'border-bambu-dark-tertiary text-bambu-gray hover:text-white'
  245. }`}
  246. >
  247. <Briefcase className="w-4 h-4" />
  248. Project
  249. </button>
  250. <button
  251. onClick={() => { setLinkType('archive'); setSelectedId(null); }}
  252. className={`flex-1 flex items-center justify-center gap-2 px-3 py-2 rounded-lg border transition-colors ${
  253. linkType === 'archive'
  254. ? 'border-bambu-green bg-bambu-green/10 text-bambu-green'
  255. : 'border-bambu-dark-tertiary text-bambu-gray hover:text-white'
  256. }`}
  257. >
  258. <ArchiveIcon className="w-4 h-4" />
  259. Archive
  260. </button>
  261. </div>
  262. {/* Selection list */}
  263. <div className="max-h-64 overflow-y-auto space-y-1 bg-bambu-dark rounded-lg p-2">
  264. {linkType === 'project' ? (
  265. projects && projects.length > 0 ? (
  266. projects.map((project) => (
  267. <button
  268. key={project.id}
  269. onClick={() => setSelectedId(project.id)}
  270. className={`w-full text-left px-3 py-2 rounded transition-colors flex items-center gap-2 ${
  271. selectedId === project.id
  272. ? 'bg-bambu-green/20 text-bambu-green'
  273. : 'hover:bg-bambu-dark-tertiary text-white'
  274. }`}
  275. >
  276. <div
  277. className="w-3 h-3 rounded-full flex-shrink-0"
  278. style={{ backgroundColor: project.color || '#00ae42' }}
  279. />
  280. <span className="truncate">{project.name}</span>
  281. </button>
  282. ))
  283. ) : (
  284. <p className="text-sm text-bambu-gray text-center py-4">No projects found</p>
  285. )
  286. ) : (
  287. archives && archives.length > 0 ? (
  288. archives.map((archive: Archive) => (
  289. <button
  290. key={archive.id}
  291. onClick={() => setSelectedId(archive.id)}
  292. className={`w-full text-left px-3 py-2 rounded transition-colors flex items-center gap-2 ${
  293. selectedId === archive.id
  294. ? 'bg-bambu-green/20 text-bambu-green'
  295. : 'hover:bg-bambu-dark-tertiary text-white'
  296. }`}
  297. >
  298. <FileBox className="w-4 h-4 text-bambu-gray flex-shrink-0" />
  299. <span className="truncate">{archive.print_name || archive.filename}</span>
  300. </button>
  301. ))
  302. ) : (
  303. <p className="text-sm text-bambu-gray text-center py-4">No archives found</p>
  304. )
  305. )}
  306. </div>
  307. </div>
  308. <div className="p-4 border-t border-bambu-dark-tertiary flex justify-between">
  309. {isLinked && (
  310. <Button variant="danger" onClick={handleUnlink} disabled={isLoading}>
  311. <Unlink className="w-4 h-4 mr-2" />
  312. Unlink
  313. </Button>
  314. )}
  315. <div className={`flex gap-2 ${!isLinked ? 'ml-auto' : ''}`}>
  316. <Button variant="secondary" onClick={onClose}>
  317. Cancel
  318. </Button>
  319. <Button onClick={handleSave} disabled={!selectedId || isLoading}>
  320. {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : 'Link'}
  321. </Button>
  322. </div>
  323. </div>
  324. </div>
  325. </div>
  326. );
  327. }
  328. // Upload Modal with Drag & Drop
  329. interface UploadModalProps {
  330. folderId: number | null;
  331. onClose: () => void;
  332. onUploadComplete: () => void;
  333. }
  334. interface UploadFile {
  335. file: File;
  336. status: 'pending' | 'uploading' | 'success' | 'error';
  337. error?: string;
  338. }
  339. function UploadModal({ folderId, onClose, onUploadComplete }: UploadModalProps) {
  340. const [files, setFiles] = useState<UploadFile[]>([]);
  341. const [isDragging, setIsDragging] = useState(false);
  342. const [isUploading, setIsUploading] = useState(false);
  343. const fileInputRef = useRef<HTMLInputElement>(null);
  344. const handleDragOver = (e: DragEvent<HTMLDivElement>) => {
  345. e.preventDefault();
  346. setIsDragging(true);
  347. };
  348. const handleDragLeave = (e: DragEvent<HTMLDivElement>) => {
  349. e.preventDefault();
  350. setIsDragging(false);
  351. };
  352. const handleDrop = (e: DragEvent<HTMLDivElement>) => {
  353. e.preventDefault();
  354. setIsDragging(false);
  355. const droppedFiles = Array.from(e.dataTransfer.files);
  356. addFiles(droppedFiles);
  357. };
  358. const handleFileSelect = (e: React.ChangeEvent<HTMLInputElement>) => {
  359. if (e.target.files) {
  360. addFiles(Array.from(e.target.files));
  361. }
  362. };
  363. const addFiles = (newFiles: File[]) => {
  364. const uploadFiles: UploadFile[] = newFiles.map((file) => ({
  365. file,
  366. status: 'pending',
  367. }));
  368. setFiles((prev) => [...prev, ...uploadFiles]);
  369. };
  370. const removeFile = (index: number) => {
  371. setFiles((prev) => prev.filter((_, i) => i !== index));
  372. };
  373. const handleUpload = async () => {
  374. if (files.length === 0) return;
  375. setIsUploading(true);
  376. for (let i = 0; i < files.length; i++) {
  377. if (files[i].status !== 'pending') continue;
  378. setFiles((prev) =>
  379. prev.map((f, idx) => (idx === i ? { ...f, status: 'uploading' } : f))
  380. );
  381. try {
  382. await api.uploadLibraryFile(files[i].file, folderId);
  383. setFiles((prev) =>
  384. prev.map((f, idx) => (idx === i ? { ...f, status: 'success' } : f))
  385. );
  386. } catch (err) {
  387. setFiles((prev) =>
  388. prev.map((f, idx) =>
  389. idx === i
  390. ? { ...f, status: 'error', error: err instanceof Error ? err.message : 'Upload failed' }
  391. : f
  392. )
  393. );
  394. }
  395. }
  396. setIsUploading(false);
  397. onUploadComplete();
  398. // Auto-close modal after upload completes
  399. onClose();
  400. };
  401. const pendingCount = files.filter((f) => f.status === 'pending').length;
  402. const successCount = files.filter((f) => f.status === 'success').length;
  403. const errorCount = files.filter((f) => f.status === 'error').length;
  404. const allDone = files.length > 0 && pendingCount === 0 && !isUploading;
  405. return (
  406. <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
  407. <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-lg border border-bambu-dark-tertiary">
  408. <div className="p-4 border-b border-bambu-dark-tertiary flex items-center justify-between">
  409. <h2 className="text-lg font-semibold text-white">Upload Files</h2>
  410. <button onClick={onClose} className="p-1 hover:bg-bambu-dark rounded">
  411. <X className="w-5 h-5 text-bambu-gray" />
  412. </button>
  413. </div>
  414. <div className="p-4 space-y-4">
  415. {/* Drop Zone */}
  416. <div
  417. onDragOver={handleDragOver}
  418. onDragLeave={handleDragLeave}
  419. onDrop={handleDrop}
  420. onClick={() => fileInputRef.current?.click()}
  421. className={`border-2 border-dashed rounded-lg p-8 text-center cursor-pointer transition-colors ${
  422. isDragging
  423. ? 'border-bambu-green bg-bambu-green/10'
  424. : 'border-bambu-dark-tertiary hover:border-bambu-green/50'
  425. }`}
  426. >
  427. <Upload className={`w-10 h-10 mx-auto mb-3 ${isDragging ? 'text-bambu-green' : 'text-bambu-gray'}`} />
  428. <p className="text-white font-medium">
  429. {isDragging ? 'Drop files here' : 'Drag & drop files here'}
  430. </p>
  431. <p className="text-sm text-bambu-gray mt-1">or click to browse</p>
  432. </div>
  433. <input
  434. ref={fileInputRef}
  435. type="file"
  436. multiple
  437. className="hidden"
  438. onChange={handleFileSelect}
  439. />
  440. {/* File List */}
  441. {files.length > 0 && (
  442. <div className="max-h-48 overflow-y-auto space-y-2">
  443. {files.map((uploadFile, index) => (
  444. <div
  445. key={index}
  446. className="flex items-center gap-3 p-2 bg-bambu-dark rounded-lg"
  447. >
  448. <File className="w-4 h-4 text-bambu-gray flex-shrink-0" />
  449. <div className="flex-1 min-w-0">
  450. <p className="text-sm text-white truncate">{uploadFile.file.name}</p>
  451. <p className="text-xs text-bambu-gray">
  452. {(uploadFile.file.size / 1024 / 1024).toFixed(2)} MB
  453. </p>
  454. </div>
  455. {uploadFile.status === 'pending' && (
  456. <button
  457. onClick={() => removeFile(index)}
  458. className="p-1 hover:bg-bambu-dark-tertiary rounded"
  459. >
  460. <X className="w-4 h-4 text-bambu-gray" />
  461. </button>
  462. )}
  463. {uploadFile.status === 'uploading' && (
  464. <Loader2 className="w-4 h-4 text-bambu-green animate-spin" />
  465. )}
  466. {uploadFile.status === 'success' && (
  467. <CheckCircle className="w-4 h-4 text-green-500" />
  468. )}
  469. {uploadFile.status === 'error' && (
  470. <span title={uploadFile.error}>
  471. <XCircle className="w-4 h-4 text-red-500" />
  472. </span>
  473. )}
  474. </div>
  475. ))}
  476. </div>
  477. )}
  478. {/* Summary */}
  479. {allDone && (
  480. <div className="p-3 bg-bambu-dark rounded-lg">
  481. <p className="text-sm text-white">
  482. Upload complete: {successCount} succeeded
  483. {errorCount > 0 && <span className="text-red-400">, {errorCount} failed</span>}
  484. </p>
  485. </div>
  486. )}
  487. </div>
  488. <div className="p-4 border-t border-bambu-dark-tertiary flex justify-end gap-2">
  489. <Button variant="secondary" onClick={onClose}>
  490. {allDone ? 'Close' : 'Cancel'}
  491. </Button>
  492. {!allDone && (
  493. <Button
  494. onClick={handleUpload}
  495. disabled={pendingCount === 0 || isUploading}
  496. >
  497. {isUploading ? (
  498. <>
  499. <Loader2 className="w-4 h-4 mr-2 animate-spin" />
  500. Uploading...
  501. </>
  502. ) : (
  503. <>
  504. <Upload className="w-4 h-4 mr-2" />
  505. Upload {pendingCount > 0 ? `(${pendingCount})` : ''}
  506. </>
  507. )}
  508. </Button>
  509. )}
  510. </div>
  511. </div>
  512. </div>
  513. );
  514. }
  515. // Folder Tree Item
  516. interface FolderTreeItemProps {
  517. folder: LibraryFolderTree;
  518. selectedFolderId: number | null;
  519. onSelect: (id: number | null) => void;
  520. onDelete: (id: number) => void;
  521. onLink: (folder: LibraryFolderTree) => void;
  522. depth?: number;
  523. }
  524. function FolderTreeItem({ folder, selectedFolderId, onSelect, onDelete, onLink, depth = 0 }: FolderTreeItemProps) {
  525. const [expanded, setExpanded] = useState(true);
  526. const [showActions, setShowActions] = useState(false);
  527. const hasChildren = folder.children.length > 0;
  528. const isLinked = folder.project_id || folder.archive_id;
  529. return (
  530. <div>
  531. <div
  532. className={`group flex items-center gap-1 px-2 py-1.5 rounded cursor-pointer transition-colors ${
  533. selectedFolderId === folder.id
  534. ? 'bg-bambu-green/20 text-bambu-green'
  535. : 'hover:bg-bambu-dark text-white'
  536. }`}
  537. style={{ paddingLeft: `${8 + depth * 12}px` }}
  538. onClick={() => onSelect(folder.id)}
  539. >
  540. {hasChildren ? (
  541. <button
  542. onClick={(e) => {
  543. e.stopPropagation();
  544. setExpanded(!expanded);
  545. }}
  546. className="p-0.5 hover:bg-bambu-dark-tertiary rounded"
  547. >
  548. <ChevronRight className={`w-3.5 h-3.5 transition-transform ${expanded ? 'rotate-90' : ''}`} />
  549. </button>
  550. ) : (
  551. <div className="w-4.5" />
  552. )}
  553. <FolderOpen className="w-4 h-4 text-bambu-green flex-shrink-0" />
  554. <span className="text-sm truncate flex-1">{folder.name}</span>
  555. {/* Link indicator - clickable to change link */}
  556. {isLinked && (
  557. <button
  558. onClick={(e) => { e.stopPropagation(); onLink(folder); }}
  559. className="flex items-center gap-1 text-xs px-1.5 py-0.5 rounded bg-blue-500/20 text-blue-400 hover:bg-blue-500/30 transition-colors"
  560. title={`${folder.project_name ? `Project: ${folder.project_name}` : `Archive: ${folder.archive_name}`} (click to change)`}
  561. >
  562. <Link2 className="w-3 h-3" />
  563. {folder.project_name ? (
  564. <Briefcase className="w-3 h-3" />
  565. ) : (
  566. <ArchiveIcon className="w-3 h-3" />
  567. )}
  568. </button>
  569. )}
  570. {folder.file_count > 0 && (
  571. <span className="text-xs text-bambu-gray">{folder.file_count}</span>
  572. )}
  573. {/* Quick link button - always visible for unlinked folders */}
  574. {!isLinked && (
  575. <button
  576. onClick={(e) => { e.stopPropagation(); onLink(folder); }}
  577. className="p-1 rounded hover:bg-bambu-dark-tertiary"
  578. title="Link to project or archive"
  579. >
  580. <Link2 className="w-3.5 h-3.5 text-bambu-gray hover:text-bambu-green" />
  581. </button>
  582. )}
  583. <div className="flex items-center gap-0.5 opacity-0 group-hover:opacity-100 transition-opacity" onClick={(e) => e.stopPropagation()}>
  584. <div className="relative">
  585. <button
  586. onClick={() => setShowActions(!showActions)}
  587. className="p-1 rounded hover:bg-bambu-dark-tertiary"
  588. >
  589. <MoreVertical className="w-3.5 h-3.5 text-bambu-gray" />
  590. </button>
  591. {showActions && (
  592. <>
  593. <div className="fixed inset-0 z-10" onClick={() => setShowActions(false)} />
  594. <div className="absolute right-0 top-full mt-1 z-20 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1 min-w-[120px]">
  595. <button
  596. className="w-full px-3 py-1.5 text-left text-sm text-white hover:bg-bambu-dark flex items-center gap-2"
  597. onClick={() => { onLink(folder); setShowActions(false); }}
  598. >
  599. <Link2 className="w-3.5 h-3.5" />
  600. {isLinked ? 'Change Link...' : 'Link to...'}
  601. </button>
  602. <button
  603. className="w-full px-3 py-1.5 text-left text-sm text-red-400 hover:bg-bambu-dark flex items-center gap-2"
  604. onClick={() => { onDelete(folder.id); setShowActions(false); }}
  605. >
  606. <Trash2 className="w-3.5 h-3.5" />
  607. Delete
  608. </button>
  609. </div>
  610. </>
  611. )}
  612. </div>
  613. </div>
  614. </div>
  615. {hasChildren && expanded && (
  616. <div>
  617. {folder.children.map((child) => (
  618. <FolderTreeItem
  619. key={child.id}
  620. folder={child}
  621. selectedFolderId={selectedFolderId}
  622. onSelect={onSelect}
  623. onDelete={onDelete}
  624. onLink={onLink}
  625. depth={depth + 1}
  626. />
  627. ))}
  628. </div>
  629. )}
  630. </div>
  631. );
  632. }
  633. // Helper to check if a file is sliced (printable)
  634. function isSlicedFilename(filename: string): boolean {
  635. const lower = filename.toLowerCase();
  636. return lower.endsWith('.gcode') || lower.includes('.gcode.');
  637. }
  638. // File Card
  639. interface FileCardProps {
  640. file: LibraryFileListItem;
  641. isSelected: boolean;
  642. onSelect: (id: number) => void;
  643. onDelete: (id: number) => void;
  644. onDownload: (id: number) => void;
  645. onAddToQueue?: (id: number) => void;
  646. }
  647. function FileCard({ file, isSelected, onSelect, onDelete, onDownload, onAddToQueue }: FileCardProps) {
  648. const [showActions, setShowActions] = useState(false);
  649. return (
  650. <div
  651. className={`group relative bg-bambu-card rounded-lg border transition-all cursor-pointer overflow-hidden ${
  652. isSelected
  653. ? 'border-bambu-green ring-1 ring-bambu-green'
  654. : 'border-bambu-dark-tertiary hover:border-bambu-green/50'
  655. }`}
  656. onClick={() => onSelect(file.id)}
  657. >
  658. {/* Thumbnail */}
  659. <div className="aspect-square bg-bambu-dark flex items-center justify-center overflow-hidden">
  660. {file.thumbnail_path ? (
  661. <img
  662. src={api.getLibraryFileThumbnailUrl(file.id)}
  663. alt={file.filename}
  664. className="w-full h-full object-cover"
  665. />
  666. ) : (
  667. <FileBox className="w-12 h-12 text-bambu-gray/30" />
  668. )}
  669. {/* Duplicate badge */}
  670. {file.duplicate_count > 0 && (
  671. <div className="absolute top-2 left-2 flex items-center gap-1 bg-amber-500/90 text-white text-xs px-1.5 py-0.5 rounded">
  672. <Copy className="w-3 h-3" />
  673. {file.duplicate_count}
  674. </div>
  675. )}
  676. {/* File type badge */}
  677. <div className={`absolute top-2 right-2 text-xs px-1.5 py-0.5 rounded font-medium ${
  678. file.file_type === '3mf' ? 'bg-bambu-green/90 text-white'
  679. : file.file_type === 'gcode' ? 'bg-blue-500/90 text-white'
  680. : file.file_type === 'stl' ? 'bg-purple-500/90 text-white'
  681. : 'bg-bambu-gray/90 text-white'
  682. }`}>
  683. {file.file_type.toUpperCase()}
  684. </div>
  685. </div>
  686. {/* Info */}
  687. <div className="p-3">
  688. <h3 className="text-sm font-medium text-white truncate" title={file.print_name || file.filename}>
  689. {file.print_name || file.filename}
  690. </h3>
  691. <div className="flex items-center gap-3 mt-1 text-xs text-bambu-gray">
  692. <span>{formatFileSize(file.file_size)}</span>
  693. {file.print_time_seconds && (
  694. <span className="flex items-center gap-1">
  695. <Clock className="w-3 h-3" />
  696. {formatDuration(file.print_time_seconds)}
  697. </span>
  698. )}
  699. </div>
  700. {file.print_count > 0 && (
  701. <div className="mt-1 text-xs text-bambu-green">
  702. Printed {file.print_count}x
  703. </div>
  704. )}
  705. </div>
  706. {/* Actions */}
  707. <div className="absolute bottom-2 right-2 opacity-0 group-hover:opacity-100 transition-opacity" onClick={(e) => e.stopPropagation()}>
  708. <button
  709. onClick={() => setShowActions(!showActions)}
  710. className="p-1.5 rounded bg-bambu-dark-secondary/90 hover:bg-bambu-dark-tertiary"
  711. >
  712. <MoreVertical className="w-4 h-4 text-bambu-gray" />
  713. </button>
  714. {showActions && (
  715. <>
  716. <div className="fixed inset-0 z-10" onClick={() => setShowActions(false)} />
  717. <div className="absolute right-0 bottom-8 z-20 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg shadow-xl py-1 min-w-[140px]">
  718. {onAddToQueue && isSlicedFilename(file.filename) && (
  719. <button
  720. className="w-full px-3 py-1.5 text-left text-sm text-bambu-green hover:bg-bambu-dark flex items-center gap-2"
  721. onClick={() => { onAddToQueue(file.id); setShowActions(false); }}
  722. >
  723. <Printer className="w-3.5 h-3.5" />
  724. Add to Queue
  725. </button>
  726. )}
  727. <button
  728. className="w-full px-3 py-1.5 text-left text-sm text-white hover:bg-bambu-dark flex items-center gap-2"
  729. onClick={() => { onDownload(file.id); setShowActions(false); }}
  730. >
  731. <Download className="w-3.5 h-3.5" />
  732. Download
  733. </button>
  734. <button
  735. className="w-full px-3 py-1.5 text-left text-sm text-red-400 hover:bg-bambu-dark flex items-center gap-2"
  736. onClick={() => { onDelete(file.id); setShowActions(false); }}
  737. >
  738. <Trash2 className="w-3.5 h-3.5" />
  739. Delete
  740. </button>
  741. </div>
  742. </>
  743. )}
  744. </div>
  745. {/* Selection checkbox */}
  746. <div className={`absolute top-2 left-2 w-5 h-5 rounded border-2 flex items-center justify-center transition-all ${
  747. isSelected
  748. ? 'bg-bambu-green border-bambu-green'
  749. : 'border-white/30 bg-black/30 opacity-0 group-hover:opacity-100'
  750. }`}>
  751. {isSelected && <div className="w-2 h-2 bg-white rounded-sm" />}
  752. </div>
  753. </div>
  754. );
  755. }
  756. export function FileManagerPage() {
  757. const queryClient = useQueryClient();
  758. const { showToast } = useToast();
  759. const [searchParams] = useSearchParams();
  760. // Read folder ID from URL query parameter
  761. const folderIdFromUrl = searchParams.get('folder');
  762. const initialFolderId = folderIdFromUrl ? parseInt(folderIdFromUrl, 10) : null;
  763. // State
  764. const [selectedFolderId, setSelectedFolderId] = useState<number | null>(initialFolderId);
  765. const [selectedFiles, setSelectedFiles] = useState<number[]>([]);
  766. const [showNewFolderModal, setShowNewFolderModal] = useState(false);
  767. const [showMoveModal, setShowMoveModal] = useState(false);
  768. const [showUploadModal, setShowUploadModal] = useState(false);
  769. const [linkFolder, setLinkFolder] = useState<LibraryFolderTree | null>(null);
  770. const [deleteConfirm, setDeleteConfirm] = useState<{ type: 'file' | 'folder' | 'bulk'; id: number; count?: number } | null>(null);
  771. const [viewMode, setViewMode] = useState<'grid' | 'list'>(() => {
  772. return (localStorage.getItem('library-view-mode') as 'grid' | 'list') || 'grid';
  773. });
  774. // Filter and sort state
  775. const [searchQuery, setSearchQuery] = useState('');
  776. const [filterType, setFilterType] = useState<string>('all');
  777. const [sortField, setSortField] = useState<SortField>('date');
  778. const [sortDirection, setSortDirection] = useState<SortDirection>('desc');
  779. // Update selectedFolderId when URL parameter changes (e.g., navigating from Project or Archive page)
  780. useEffect(() => {
  781. const folderParam = searchParams.get('folder');
  782. const newFolderId = folderParam ? parseInt(folderParam, 10) : null;
  783. if (newFolderId !== selectedFolderId) {
  784. setSelectedFolderId(newFolderId);
  785. }
  786. }, [searchParams]);
  787. // Queries
  788. const { data: settings } = useQuery({
  789. queryKey: ['settings'],
  790. queryFn: () => api.getSettings() as Promise<AppSettings>,
  791. });
  792. const { data: folders, isLoading: foldersLoading } = useQuery({
  793. queryKey: ['library-folders'],
  794. queryFn: () => api.getLibraryFolders(),
  795. });
  796. const { data: files, isLoading: filesLoading } = useQuery({
  797. queryKey: ['library-files', selectedFolderId],
  798. queryFn: () => api.getLibraryFiles(selectedFolderId, selectedFolderId === null),
  799. });
  800. const { data: stats } = useQuery({
  801. queryKey: ['library-stats'],
  802. queryFn: () => api.getLibraryStats(),
  803. });
  804. // Get unique file types for filter dropdown
  805. const fileTypes = useMemo(() => {
  806. if (!files) return [];
  807. const types = new Set(files.map((f) => f.file_type));
  808. return Array.from(types).sort();
  809. }, [files]);
  810. // Filter and sort files
  811. const filteredAndSortedFiles = useMemo(() => {
  812. if (!files) return [];
  813. let result = [...files];
  814. // Apply search filter
  815. if (searchQuery.trim()) {
  816. const query = searchQuery.toLowerCase();
  817. result = result.filter(
  818. (f) =>
  819. f.filename.toLowerCase().includes(query) ||
  820. (f.print_name && f.print_name.toLowerCase().includes(query))
  821. );
  822. }
  823. // Apply type filter
  824. if (filterType !== 'all') {
  825. result = result.filter((f) => f.file_type === filterType);
  826. }
  827. // Apply sorting
  828. result.sort((a, b) => {
  829. let comparison = 0;
  830. switch (sortField) {
  831. case 'name':
  832. comparison = (a.print_name || a.filename).localeCompare(b.print_name || b.filename);
  833. break;
  834. case 'date':
  835. comparison = new Date(a.created_at).getTime() - new Date(b.created_at).getTime();
  836. break;
  837. case 'size':
  838. comparison = a.file_size - b.file_size;
  839. break;
  840. case 'type':
  841. comparison = a.file_type.localeCompare(b.file_type);
  842. break;
  843. case 'prints':
  844. comparison = a.print_count - b.print_count;
  845. break;
  846. }
  847. return sortDirection === 'asc' ? comparison : -comparison;
  848. });
  849. return result;
  850. }, [files, searchQuery, filterType, sortField, sortDirection]);
  851. // Check if disk space is low
  852. const isDiskSpaceLow = useMemo(() => {
  853. if (!stats || !settings) return false;
  854. const thresholdBytes = (settings.library_disk_warning_gb || 5) * 1024 * 1024 * 1024;
  855. return stats.disk_free_bytes < thresholdBytes;
  856. }, [stats, settings]);
  857. // Mutations
  858. const createFolderMutation = useMutation({
  859. mutationFn: (data: LibraryFolderCreate) => api.createLibraryFolder(data),
  860. onSuccess: () => {
  861. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  862. setShowNewFolderModal(false);
  863. showToast('Folder created', 'success');
  864. },
  865. onError: (error: Error) => showToast(error.message, 'error'),
  866. });
  867. const deleteFolderMutation = useMutation({
  868. mutationFn: (id: number) => api.deleteLibraryFolder(id),
  869. onSuccess: () => {
  870. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  871. queryClient.invalidateQueries({ queryKey: ['library-files'] });
  872. queryClient.invalidateQueries({ queryKey: ['library-stats'] });
  873. if (selectedFolderId === deleteConfirm?.id) {
  874. setSelectedFolderId(null);
  875. }
  876. setDeleteConfirm(null);
  877. showToast('Folder deleted', 'success');
  878. },
  879. onError: (error: Error) => {
  880. setDeleteConfirm(null);
  881. showToast(error.message, 'error');
  882. },
  883. });
  884. const deleteFileMutation = useMutation({
  885. mutationFn: (id: number) => api.deleteLibraryFile(id),
  886. onSuccess: () => {
  887. queryClient.invalidateQueries({ queryKey: ['library-files'] });
  888. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  889. queryClient.invalidateQueries({ queryKey: ['library-stats'] });
  890. setSelectedFiles((prev) => prev.filter((id) => id !== deleteConfirm?.id));
  891. setDeleteConfirm(null);
  892. showToast('File deleted', 'success');
  893. },
  894. onError: (error: Error) => {
  895. setDeleteConfirm(null);
  896. showToast(error.message, 'error');
  897. },
  898. });
  899. const moveFilesMutation = useMutation({
  900. mutationFn: ({ fileIds, folderId }: { fileIds: number[]; folderId: number | null }) =>
  901. api.moveLibraryFiles(fileIds, folderId),
  902. onSuccess: () => {
  903. queryClient.invalidateQueries({ queryKey: ['library-files'] });
  904. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  905. setSelectedFiles([]);
  906. setShowMoveModal(false);
  907. showToast('Files moved', 'success');
  908. },
  909. onError: (error: Error) => showToast(error.message, 'error'),
  910. });
  911. const updateFolderMutation = useMutation({
  912. mutationFn: ({ id, data }: { id: number; data: LibraryFolderUpdate }) =>
  913. api.updateLibraryFolder(id, data),
  914. onSuccess: (_, variables) => {
  915. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  916. // Invalidate project/archive folder queries so other pages see the update
  917. queryClient.invalidateQueries({ queryKey: ['project-folders'] });
  918. queryClient.invalidateQueries({ queryKey: ['archive-folders'] });
  919. setLinkFolder(null);
  920. const isUnlink = variables.data.project_id === 0 && variables.data.archive_id === 0;
  921. showToast(isUnlink ? 'Folder unlinked' : 'Folder linked', 'success');
  922. },
  923. onError: (error: Error) => showToast(error.message, 'error'),
  924. });
  925. const addToQueueMutation = useMutation({
  926. mutationFn: (fileIds: number[]) => api.addLibraryFilesToQueue(fileIds),
  927. onSuccess: (result) => {
  928. queryClient.invalidateQueries({ queryKey: ['library-files'] });
  929. queryClient.invalidateQueries({ queryKey: ['queue'] });
  930. setSelectedFiles([]);
  931. if (result.added.length > 0 && result.errors.length === 0) {
  932. showToast(
  933. `Added ${result.added.length} file${result.added.length > 1 ? 's' : ''} to queue`,
  934. 'success'
  935. );
  936. } else if (result.added.length > 0 && result.errors.length > 0) {
  937. showToast(
  938. `Added ${result.added.length} file${result.added.length > 1 ? 's' : ''}, ${result.errors.length} failed`,
  939. 'success'
  940. );
  941. } else {
  942. showToast(`Failed to add files: ${result.errors[0]?.error || 'Unknown error'}`, 'error');
  943. }
  944. },
  945. onError: (error: Error) => showToast(error.message, 'error'),
  946. });
  947. // Helper to check if a file is sliced (printable)
  948. const isSlicedFile = useCallback((filename: string) => {
  949. const lower = filename.toLowerCase();
  950. return lower.endsWith('.gcode') || lower.includes('.gcode.');
  951. }, []);
  952. // Get sliced files from selection
  953. const selectedSlicedFiles = useMemo(() => {
  954. if (!files) return [];
  955. return files.filter(f => selectedFiles.includes(f.id) && isSlicedFile(f.filename));
  956. }, [files, selectedFiles, isSlicedFile]);
  957. // Handlers
  958. const handleFileSelect = useCallback((id: number) => {
  959. // Always toggle selection (multi-select by default)
  960. setSelectedFiles((prev) => {
  961. return prev.includes(id) ? prev.filter((x) => x !== id) : [...prev, id];
  962. });
  963. }, []);
  964. const handleSelectAll = useCallback(() => {
  965. if (filteredAndSortedFiles.length > 0) {
  966. setSelectedFiles(filteredAndSortedFiles.map((f) => f.id));
  967. }
  968. }, [filteredAndSortedFiles]);
  969. const handleDeselectAll = useCallback(() => {
  970. setSelectedFiles([]);
  971. }, []);
  972. const handleUploadComplete = () => {
  973. queryClient.invalidateQueries({ queryKey: ['library-files'] });
  974. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  975. queryClient.invalidateQueries({ queryKey: ['library-stats'] });
  976. };
  977. const handleDownload = (id: number) => {
  978. window.open(api.getLibraryFileDownloadUrl(id), '_blank');
  979. };
  980. const handleDeleteConfirm = () => {
  981. if (!deleteConfirm) return;
  982. if (deleteConfirm.type === 'file') {
  983. deleteFileMutation.mutate(deleteConfirm.id);
  984. } else if (deleteConfirm.type === 'folder') {
  985. deleteFolderMutation.mutate(deleteConfirm.id);
  986. } else if (deleteConfirm.type === 'bulk') {
  987. // Bulk delete selected files
  988. api.bulkDeleteLibrary(selectedFiles, []).then(() => {
  989. queryClient.invalidateQueries({ queryKey: ['library-files'] });
  990. queryClient.invalidateQueries({ queryKey: ['library-folders'] });
  991. queryClient.invalidateQueries({ queryKey: ['library-stats'] });
  992. showToast(`Deleted ${selectedFiles.length} files`, 'success');
  993. setSelectedFiles([]);
  994. setDeleteConfirm(null);
  995. }).catch((err) => {
  996. showToast(err.message, 'error');
  997. setDeleteConfirm(null);
  998. });
  999. }
  1000. };
  1001. const handleViewModeChange = (mode: 'grid' | 'list') => {
  1002. setViewMode(mode);
  1003. localStorage.setItem('library-view-mode', mode);
  1004. };
  1005. const isLoading = foldersLoading || filesLoading;
  1006. return (
  1007. <div className="p-4 md:p-8 h-[calc(100vh-64px)] flex flex-col">
  1008. {/* Header */}
  1009. <div className="flex flex-col sm:flex-row sm:items-center justify-between gap-4 mb-6">
  1010. <div>
  1011. <h1 className="text-2xl font-bold text-white flex items-center gap-3">
  1012. <div className="p-2.5 bg-bambu-green/10 rounded-xl">
  1013. <FolderOpen className="w-6 h-6 text-bambu-green" />
  1014. </div>
  1015. File Manager
  1016. </h1>
  1017. <p className="text-sm text-bambu-gray mt-2 ml-14">
  1018. Organize and manage your print files
  1019. </p>
  1020. </div>
  1021. <div className="flex items-center gap-2">
  1022. {/* View mode toggle */}
  1023. <div className="flex items-center bg-bambu-dark rounded-lg p-1">
  1024. <button
  1025. onClick={() => handleViewModeChange('grid')}
  1026. className={`p-1.5 rounded transition-colors ${
  1027. viewMode === 'grid' ? 'bg-bambu-card text-white' : 'text-bambu-gray hover:text-white'
  1028. }`}
  1029. title="Grid view"
  1030. >
  1031. <LayoutGrid className="w-4 h-4" />
  1032. </button>
  1033. <button
  1034. onClick={() => handleViewModeChange('list')}
  1035. className={`p-1.5 rounded transition-colors ${
  1036. viewMode === 'list' ? 'bg-bambu-card text-white' : 'text-bambu-gray hover:text-white'
  1037. }`}
  1038. title="List view"
  1039. >
  1040. <List className="w-4 h-4" />
  1041. </button>
  1042. </div>
  1043. <Button variant="secondary" onClick={() => setShowNewFolderModal(true)}>
  1044. <FolderPlus className="w-4 h-4 mr-2" />
  1045. New Folder
  1046. </Button>
  1047. <Button onClick={() => setShowUploadModal(true)}>
  1048. <Upload className="w-4 h-4 mr-2" />
  1049. Upload
  1050. </Button>
  1051. </div>
  1052. </div>
  1053. {/* Disk space warning */}
  1054. {isDiskSpaceLow && stats && settings && (
  1055. <div className="flex items-center gap-3 mb-4 p-3 bg-amber-500/10 border border-amber-500/30 rounded-lg">
  1056. <AlertTriangle className="w-5 h-5 text-amber-500 flex-shrink-0" />
  1057. <div className="flex-1">
  1058. <p className="text-sm text-amber-500 font-medium">Low disk space warning</p>
  1059. <p className="text-xs text-amber-500/80">
  1060. Only {formatFileSize(stats.disk_free_bytes)} free of {formatFileSize(stats.disk_total_bytes)} total.
  1061. Threshold is set to {settings.library_disk_warning_gb} GB in settings.
  1062. </p>
  1063. </div>
  1064. </div>
  1065. )}
  1066. {/* Stats bar */}
  1067. {stats && (
  1068. <div className="flex items-center gap-6 mb-6 p-3 bg-bambu-card rounded-lg border border-bambu-dark-tertiary">
  1069. <div className="flex items-center gap-2 text-sm">
  1070. <File className="w-4 h-4 text-bambu-green" />
  1071. <span className="text-bambu-gray">Files:</span>
  1072. <span className="text-white font-medium">{stats.total_files}</span>
  1073. </div>
  1074. <div className="flex items-center gap-2 text-sm">
  1075. <FolderOpen className="w-4 h-4 text-blue-400" />
  1076. <span className="text-bambu-gray">Folders:</span>
  1077. <span className="text-white font-medium">{stats.total_folders}</span>
  1078. </div>
  1079. <div className="flex items-center gap-2 text-sm">
  1080. <HardDrive className="w-4 h-4 text-amber-400" />
  1081. <span className="text-bambu-gray">Size:</span>
  1082. <span className="text-white font-medium">{formatFileSize(stats.total_size_bytes)}</span>
  1083. </div>
  1084. <div className="flex items-center gap-2 text-sm ml-auto">
  1085. <span className="text-bambu-gray">Free:</span>
  1086. <span className={`font-medium ${isDiskSpaceLow ? 'text-amber-500' : 'text-white'}`}>
  1087. {formatFileSize(stats.disk_free_bytes)}
  1088. </span>
  1089. </div>
  1090. </div>
  1091. )}
  1092. {/* Main content */}
  1093. <div className="flex-1 flex gap-6 min-h-0">
  1094. {/* Folder sidebar */}
  1095. <div className="w-64 flex-shrink-0 bg-bambu-card rounded-lg border border-bambu-dark-tertiary overflow-hidden flex flex-col">
  1096. <div className="p-3 border-b border-bambu-dark-tertiary">
  1097. <h2 className="text-sm font-medium text-white">Folders</h2>
  1098. </div>
  1099. <div className="flex-1 overflow-y-auto p-2">
  1100. {/* All Files (root) */}
  1101. <div
  1102. className={`flex items-center gap-2 px-2 py-1.5 rounded cursor-pointer transition-colors ${
  1103. selectedFolderId === null
  1104. ? 'bg-bambu-green/20 text-bambu-green'
  1105. : 'hover:bg-bambu-dark text-white'
  1106. }`}
  1107. onClick={() => setSelectedFolderId(null)}
  1108. >
  1109. <FileBox className="w-4 h-4" />
  1110. <span className="text-sm">All Files</span>
  1111. </div>
  1112. {/* Folder tree */}
  1113. {folders?.map((folder) => (
  1114. <FolderTreeItem
  1115. key={folder.id}
  1116. folder={folder}
  1117. selectedFolderId={selectedFolderId}
  1118. onSelect={setSelectedFolderId}
  1119. onDelete={(id) => setDeleteConfirm({ type: 'folder', id })}
  1120. onLink={setLinkFolder}
  1121. />
  1122. ))}
  1123. </div>
  1124. </div>
  1125. {/* Files area */}
  1126. <div className="flex-1 flex flex-col min-w-0">
  1127. {/* Search, Filter, Sort toolbar */}
  1128. {files && files.length > 0 && (
  1129. <div className="flex items-center gap-3 mb-4 p-3 bg-bambu-card rounded-lg border border-bambu-dark-tertiary">
  1130. {/* Search */}
  1131. <div className="relative flex-1 max-w-xs">
  1132. <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray" />
  1133. <input
  1134. type="text"
  1135. placeholder="Search files..."
  1136. value={searchQuery}
  1137. onChange={(e) => setSearchQuery(e.target.value)}
  1138. 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"
  1139. />
  1140. </div>
  1141. {/* Type filter */}
  1142. <div className="flex items-center gap-2">
  1143. <Filter className="w-4 h-4 text-bambu-gray" />
  1144. <select
  1145. value={filterType}
  1146. onChange={(e) => setFilterType(e.target.value)}
  1147. className="bg-bambu-dark border border-bambu-dark-tertiary rounded px-2 py-1.5 text-sm text-white focus:outline-none focus:border-bambu-green"
  1148. >
  1149. <option value="all">All types</option>
  1150. {fileTypes.map((type) => (
  1151. <option key={type} value={type}>
  1152. {type.toUpperCase()}
  1153. </option>
  1154. ))}
  1155. </select>
  1156. </div>
  1157. {/* Sort */}
  1158. <div className="flex items-center gap-2">
  1159. <select
  1160. value={sortField}
  1161. onChange={(e) => setSortField(e.target.value as SortField)}
  1162. className="bg-bambu-dark border border-bambu-dark-tertiary rounded px-2 py-1.5 text-sm text-white focus:outline-none focus:border-bambu-green"
  1163. >
  1164. <option value="date">Date</option>
  1165. <option value="name">Name</option>
  1166. <option value="size">Size</option>
  1167. <option value="type">Type</option>
  1168. <option value="prints">Prints</option>
  1169. </select>
  1170. <button
  1171. onClick={() => setSortDirection((d) => (d === 'asc' ? 'desc' : 'asc'))}
  1172. className="p-1.5 rounded bg-bambu-dark border border-bambu-dark-tertiary hover:border-bambu-green transition-colors"
  1173. title={sortDirection === 'asc' ? 'Ascending' : 'Descending'}
  1174. >
  1175. {sortDirection === 'asc' ? (
  1176. <SortAsc className="w-4 h-4 text-white" />
  1177. ) : (
  1178. <SortDesc className="w-4 h-4 text-white" />
  1179. )}
  1180. </button>
  1181. </div>
  1182. {/* Results count */}
  1183. {(searchQuery || filterType !== 'all') && (
  1184. <span className="text-sm text-bambu-gray">
  1185. {filteredAndSortedFiles.length} of {files.length} files
  1186. </span>
  1187. )}
  1188. </div>
  1189. )}
  1190. {/* Selection toolbar */}
  1191. {filteredAndSortedFiles.length > 0 && (
  1192. <div className="flex items-center gap-2 mb-4 p-2 bg-bambu-card rounded-lg border border-bambu-dark-tertiary">
  1193. {/* Select all / Deselect all */}
  1194. {selectedFiles.length === filteredAndSortedFiles.length && selectedFiles.length > 0 ? (
  1195. <Button
  1196. variant="secondary"
  1197. size="sm"
  1198. onClick={handleDeselectAll}
  1199. >
  1200. <Square className="w-4 h-4 mr-1" />
  1201. Deselect All
  1202. </Button>
  1203. ) : (
  1204. <Button
  1205. variant="secondary"
  1206. size="sm"
  1207. onClick={handleSelectAll}
  1208. >
  1209. <CheckSquare className="w-4 h-4 mr-1" />
  1210. Select All
  1211. </Button>
  1212. )}
  1213. {selectedFiles.length > 0 && (
  1214. <>
  1215. <span className="text-sm text-bambu-gray ml-2">
  1216. {selectedFiles.length} selected
  1217. </span>
  1218. <div className="flex-1" />
  1219. {selectedSlicedFiles.length > 0 && (
  1220. <Button
  1221. variant="primary"
  1222. size="sm"
  1223. onClick={() => addToQueueMutation.mutate(selectedSlicedFiles.map(f => f.id))}
  1224. disabled={addToQueueMutation.isPending}
  1225. >
  1226. <Printer className="w-4 h-4 mr-1" />
  1227. {addToQueueMutation.isPending ? 'Adding...' : `Add to Queue${selectedSlicedFiles.length < selectedFiles.length ? ` (${selectedSlicedFiles.length})` : ''}`}
  1228. </Button>
  1229. )}
  1230. <Button
  1231. variant="secondary"
  1232. size="sm"
  1233. onClick={() => setShowMoveModal(true)}
  1234. >
  1235. <MoveRight className="w-4 h-4 mr-1" />
  1236. Move
  1237. </Button>
  1238. <Button
  1239. variant="danger"
  1240. size="sm"
  1241. onClick={() => {
  1242. if (selectedFiles.length === 1) {
  1243. setDeleteConfirm({ type: 'file', id: selectedFiles[0] });
  1244. } else {
  1245. setDeleteConfirm({ type: 'bulk', id: 0, count: selectedFiles.length });
  1246. }
  1247. }}
  1248. >
  1249. <Trash2 className="w-4 h-4 mr-1" />
  1250. Delete
  1251. </Button>
  1252. <Button
  1253. variant="secondary"
  1254. size="sm"
  1255. onClick={handleDeselectAll}
  1256. >
  1257. Clear
  1258. </Button>
  1259. </>
  1260. )}
  1261. </div>
  1262. )}
  1263. {/* File grid/list */}
  1264. {isLoading ? (
  1265. <div className="flex-1 flex items-center justify-center">
  1266. <div className="flex flex-col items-center gap-3">
  1267. <Loader2 className="w-8 h-8 animate-spin text-bambu-green" />
  1268. <p className="text-sm text-bambu-gray">Loading files...</p>
  1269. </div>
  1270. </div>
  1271. ) : files?.length === 0 ? (
  1272. <div className="flex-1 flex flex-col items-center justify-center">
  1273. <div className="p-4 bg-bambu-dark rounded-2xl mb-4">
  1274. <FileBox className="w-12 h-12 text-bambu-gray/50" />
  1275. </div>
  1276. <h3 className="text-lg font-medium text-white mb-2">
  1277. {selectedFolderId !== null ? 'Folder is empty' : 'No files yet'}
  1278. </h3>
  1279. <p className="text-bambu-gray text-center max-w-md mb-6">
  1280. {selectedFolderId !== null
  1281. ? 'Upload files or move files into this folder to get started.'
  1282. : 'Upload files to start organizing your print-related files.'}
  1283. </p>
  1284. <Button onClick={() => setShowUploadModal(true)}>
  1285. <Plus className="w-4 h-4 mr-2" />
  1286. Upload Files
  1287. </Button>
  1288. </div>
  1289. ) : filteredAndSortedFiles.length === 0 ? (
  1290. <div className="flex-1 flex flex-col items-center justify-center">
  1291. <div className="p-4 bg-bambu-dark rounded-2xl mb-4">
  1292. <Search className="w-12 h-12 text-bambu-gray/50" />
  1293. </div>
  1294. <h3 className="text-lg font-medium text-white mb-2">No matching files</h3>
  1295. <p className="text-bambu-gray text-center max-w-md mb-6">
  1296. No files match your current search or filter criteria.
  1297. </p>
  1298. <Button variant="secondary" onClick={() => { setSearchQuery(''); setFilterType('all'); }}>
  1299. Clear filters
  1300. </Button>
  1301. </div>
  1302. ) : viewMode === 'grid' ? (
  1303. <div className="flex-1 overflow-y-auto">
  1304. <div className="grid grid-cols-2 sm:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5 2xl:grid-cols-6 gap-4">
  1305. {filteredAndSortedFiles.map((file) => (
  1306. <FileCard
  1307. key={file.id}
  1308. file={file}
  1309. isSelected={selectedFiles.includes(file.id)}
  1310. onSelect={handleFileSelect}
  1311. onDelete={(id) => setDeleteConfirm({ type: 'file', id })}
  1312. onDownload={handleDownload}
  1313. onAddToQueue={(id) => addToQueueMutation.mutate([id])}
  1314. />
  1315. ))}
  1316. </div>
  1317. </div>
  1318. ) : (
  1319. <div className="flex-1 overflow-y-auto">
  1320. <div className="bg-bambu-card rounded-lg border border-bambu-dark-tertiary overflow-hidden">
  1321. {/* List header */}
  1322. <div className="grid grid-cols-[auto_1fr_100px_100px_100px_80px] gap-4 px-4 py-2 bg-bambu-dark-secondary border-b border-bambu-dark-tertiary text-xs text-bambu-gray font-medium">
  1323. <div className="w-6" />
  1324. <div>Name</div>
  1325. <div>Type</div>
  1326. <div>Size</div>
  1327. <div>Prints</div>
  1328. <div />
  1329. </div>
  1330. {/* List rows */}
  1331. {filteredAndSortedFiles.map((file) => (
  1332. <div
  1333. key={file.id}
  1334. className={`grid grid-cols-[auto_1fr_100px_100px_100px_80px] gap-4 px-4 py-3 items-center border-b border-bambu-dark-tertiary last:border-b-0 cursor-pointer hover:bg-bambu-dark/50 transition-colors ${
  1335. selectedFiles.includes(file.id) ? 'bg-bambu-green/10' : ''
  1336. }`}
  1337. onClick={() => handleFileSelect(file.id)}
  1338. >
  1339. {/* Checkbox */}
  1340. <div className={`w-5 h-5 rounded border-2 flex items-center justify-center ${
  1341. selectedFiles.includes(file.id)
  1342. ? 'bg-bambu-green border-bambu-green'
  1343. : 'border-bambu-gray/50'
  1344. }`}>
  1345. {selectedFiles.includes(file.id) && <div className="w-2 h-2 bg-white rounded-sm" />}
  1346. </div>
  1347. {/* Name with thumbnail */}
  1348. <div className="flex items-center gap-3 min-w-0">
  1349. <div className="relative group/thumb">
  1350. <div className="w-10 h-10 rounded bg-bambu-dark flex-shrink-0 overflow-hidden">
  1351. {file.thumbnail_path ? (
  1352. <img
  1353. src={api.getLibraryFileThumbnailUrl(file.id)}
  1354. alt=""
  1355. className="w-full h-full object-cover"
  1356. />
  1357. ) : (
  1358. <div className="w-full h-full flex items-center justify-center">
  1359. <FileBox className="w-5 h-5 text-bambu-gray/50" />
  1360. </div>
  1361. )}
  1362. </div>
  1363. {/* Hover preview */}
  1364. {file.thumbnail_path && (
  1365. <div className="absolute left-0 top-full mt-2 z-50 hidden group-hover/thumb:block">
  1366. <div className="w-48 h-48 rounded-lg bg-bambu-dark-secondary border border-bambu-dark-tertiary shadow-xl overflow-hidden">
  1367. <img
  1368. src={api.getLibraryFileThumbnailUrl(file.id)}
  1369. alt={file.filename}
  1370. className="w-full h-full object-contain"
  1371. />
  1372. </div>
  1373. </div>
  1374. )}
  1375. </div>
  1376. <div className="min-w-0">
  1377. <div className="text-sm text-white truncate">{file.print_name || file.filename}</div>
  1378. {file.duplicate_count > 0 && (
  1379. <div className="flex items-center gap-1 text-xs text-amber-400">
  1380. <Copy className="w-3 h-3" />
  1381. {file.duplicate_count} duplicate(s)
  1382. </div>
  1383. )}
  1384. </div>
  1385. </div>
  1386. {/* Type */}
  1387. <div>
  1388. <span className={`text-xs px-1.5 py-0.5 rounded font-medium ${
  1389. file.file_type === '3mf' ? 'bg-bambu-green/20 text-bambu-green'
  1390. : file.file_type === 'gcode' ? 'bg-blue-500/20 text-blue-400'
  1391. : file.file_type === 'stl' ? 'bg-purple-500/20 text-purple-400'
  1392. : 'bg-bambu-gray/20 text-bambu-gray'
  1393. }`}>
  1394. {file.file_type.toUpperCase()}
  1395. </span>
  1396. </div>
  1397. {/* Size */}
  1398. <div className="text-sm text-bambu-gray">{formatFileSize(file.file_size)}</div>
  1399. {/* Prints */}
  1400. <div className="text-sm text-bambu-gray">{file.print_count > 0 ? `${file.print_count}x` : '-'}</div>
  1401. {/* Actions */}
  1402. <div className="flex items-center gap-1" onClick={(e) => e.stopPropagation()}>
  1403. {isSlicedFilename(file.filename) && (
  1404. <button
  1405. onClick={() => addToQueueMutation.mutate([file.id])}
  1406. className="p-1.5 rounded hover:bg-bambu-dark text-bambu-gray hover:text-bambu-green transition-colors"
  1407. title="Add to Queue"
  1408. disabled={addToQueueMutation.isPending}
  1409. >
  1410. <Printer className="w-4 h-4" />
  1411. </button>
  1412. )}
  1413. <button
  1414. onClick={() => handleDownload(file.id)}
  1415. className="p-1.5 rounded hover:bg-bambu-dark text-bambu-gray hover:text-white transition-colors"
  1416. title="Download"
  1417. >
  1418. <Download className="w-4 h-4" />
  1419. </button>
  1420. <button
  1421. onClick={() => setDeleteConfirm({ type: 'file', id: file.id })}
  1422. className="p-1.5 rounded hover:bg-bambu-dark text-bambu-gray hover:text-red-400 transition-colors"
  1423. title="Delete"
  1424. >
  1425. <Trash2 className="w-4 h-4" />
  1426. </button>
  1427. </div>
  1428. </div>
  1429. ))}
  1430. </div>
  1431. </div>
  1432. )}
  1433. </div>
  1434. </div>
  1435. {/* Modals */}
  1436. {showNewFolderModal && (
  1437. <NewFolderModal
  1438. parentId={selectedFolderId}
  1439. onClose={() => setShowNewFolderModal(false)}
  1440. onSave={(data) => createFolderMutation.mutate(data)}
  1441. isLoading={createFolderMutation.isPending}
  1442. />
  1443. )}
  1444. {showMoveModal && folders && (
  1445. <MoveFilesModal
  1446. folders={folders}
  1447. selectedFiles={selectedFiles}
  1448. currentFolderId={selectedFolderId}
  1449. onClose={() => setShowMoveModal(false)}
  1450. onMove={(folderId) => moveFilesMutation.mutate({ fileIds: selectedFiles, folderId })}
  1451. isLoading={moveFilesMutation.isPending}
  1452. />
  1453. )}
  1454. {showUploadModal && (
  1455. <UploadModal
  1456. folderId={selectedFolderId}
  1457. onClose={() => setShowUploadModal(false)}
  1458. onUploadComplete={handleUploadComplete}
  1459. />
  1460. )}
  1461. {linkFolder && (
  1462. <LinkFolderModal
  1463. folder={linkFolder}
  1464. onClose={() => setLinkFolder(null)}
  1465. onLink={(data) => updateFolderMutation.mutate({ id: linkFolder.id, data })}
  1466. isLoading={updateFolderMutation.isPending}
  1467. />
  1468. )}
  1469. {deleteConfirm && (
  1470. <ConfirmModal
  1471. title={
  1472. deleteConfirm.type === 'folder'
  1473. ? 'Delete Folder'
  1474. : deleteConfirm.type === 'bulk'
  1475. ? `Delete ${deleteConfirm.count} Files`
  1476. : 'Delete File'
  1477. }
  1478. message={
  1479. deleteConfirm.type === 'folder'
  1480. ? 'Are you sure you want to delete this folder? All files inside will also be deleted.'
  1481. : deleteConfirm.type === 'bulk'
  1482. ? `Are you sure you want to delete ${deleteConfirm.count} selected files? This action cannot be undone.`
  1483. : 'Are you sure you want to delete this file?'
  1484. }
  1485. confirmText="Delete"
  1486. variant="danger"
  1487. onConfirm={handleDeleteConfirm}
  1488. onCancel={() => setDeleteConfirm(null)}
  1489. />
  1490. )}
  1491. </div>
  1492. );
  1493. }