| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262 |
- 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, unknown>) => 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 (
- <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
- <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-sm border border-bambu-dark-tertiary">
- <div className="p-4 border-b border-bambu-dark-tertiary">
- <h2 className="text-lg font-semibold text-white">{t('fileManager.newFolder')}</h2>
- </div>
- <form onSubmit={handleSubmit} className="p-4 space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-1">
- {t('fileManager.folderName')}
- </label>
- <input
- type="text"
- value={name}
- onChange={(e) => 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
- />
- </div>
- <div className="flex justify-end gap-2 pt-2">
- <Button type="button" variant="secondary" onClick={onClose}>
- {t('common.cancel')}
- </Button>
- <Button type="submit" disabled={!name.trim() || isLoading}>
- {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : t('common.create')}
- </Button>
- </div>
- </form>
- </div>
- </div>
- );
- }
- // 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 (
- <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
- <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-md border border-bambu-dark-tertiary">
- <div className="p-4 border-b border-bambu-dark-tertiary">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <FolderSymlink className="w-5 h-5 text-bambu-green" />
- {t('fileManager.linkExternalFolder')}
- </h2>
- <p className="text-sm text-bambu-gray mt-1">{t('fileManager.linkExternalFolderDescription')}</p>
- </div>
- <form onSubmit={handleSubmit} className="p-4 space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-1">
- {t('fileManager.folderName')}
- </label>
- <input
- type="text"
- value={name}
- onChange={(e) => 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
- />
- </div>
- <div>
- <label className="block text-sm font-medium text-white mb-1">
- {t('fileManager.externalPath')}
- </label>
- <input
- type="text"
- value={path}
- onChange={(e) => 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
- />
- <p className="text-xs text-bambu-gray mt-1">{t('fileManager.externalPathHelp')}</p>
- </div>
- <div className="space-y-2">
- <label className="flex items-center gap-2 cursor-pointer">
- <input
- type="checkbox"
- checked={readonly}
- onChange={(e) => setReadonly(e.target.checked)}
- className="rounded border-bambu-dark-tertiary bg-bambu-dark text-bambu-green focus:ring-bambu-green"
- />
- <span className="text-sm text-white">{t('fileManager.readOnly')}</span>
- <span className="text-xs text-bambu-gray">({t('fileManager.readOnlyHelp')})</span>
- </label>
- <label className="flex items-center gap-2 cursor-pointer">
- <input
- type="checkbox"
- checked={showHidden}
- onChange={(e) => setShowHidden(e.target.checked)}
- className="rounded border-bambu-dark-tertiary bg-bambu-dark text-bambu-green focus:ring-bambu-green"
- />
- <span className="text-sm text-white">{t('fileManager.showHiddenFiles')}</span>
- </label>
- </div>
- <div className="flex justify-end gap-2 pt-2">
- <Button type="button" variant="secondary" onClick={onClose}>
- {t('common.cancel')}
- </Button>
- <Button type="submit" disabled={!name.trim() || !path.trim() || isLoading}>
- {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : t('fileManager.linkFolder')}
- </Button>
- </div>
- </form>
- </div>
- </div>
- );
- }
- // 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 (
- <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
- <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-sm border border-bambu-dark-tertiary">
- <div className="p-4 border-b border-bambu-dark-tertiary">
- <h2 className="text-lg font-semibold text-white">{type === 'file' ? t('fileManager.renameFile') : t('fileManager.renameFolder')}</h2>
- </div>
- <form onSubmit={handleSubmit} className="p-4 space-y-4">
- <div>
- <label className="block text-sm font-medium text-white mb-1">
- {t('common.name')}
- </label>
- <div className="flex items-center bg-bambu-dark border border-bambu-dark-tertiary rounded focus-within:border-bambu-green">
- <input
- type="text"
- value={name}
- onChange={(e) => 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 && (
- <span className="pr-3 text-bambu-gray text-sm select-none whitespace-nowrap">{fileExtension}</span>
- )}
- </div>
- </div>
- <div className="flex justify-end gap-2 pt-2">
- <Button type="button" variant="secondary" onClick={onClose}>
- {t('common.cancel')}
- </Button>
- <Button type="submit" disabled={!name.trim() || name.trim() === baseName || isLoading}>
- {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : t('common.rename')}
- </Button>
- </div>
- </form>
- </div>
- </div>
- );
- }
- // 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<number | null>(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 (
- <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
- <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-sm border border-bambu-dark-tertiary">
- <div className="p-4 border-b border-bambu-dark-tertiary">
- <h2 className="text-lg font-semibold text-white">{t('fileManager.moveFiles', { count: selectedFiles.length })}</h2>
- </div>
- <div className="p-4 space-y-4">
- <div className="max-h-64 overflow-y-auto space-y-1">
- {flatFolders.map((folder) => (
- <button
- key={folder.id ?? 'root'}
- onClick={() => setTargetFolder(folder.id)}
- disabled={folder.id === currentFolderId}
- className={`w-full text-left px-3 py-2 rounded transition-colors flex items-center gap-2 ${
- targetFolder === folder.id
- ? 'bg-bambu-green/20 text-bambu-green'
- : folder.id === currentFolderId
- ? 'opacity-50 cursor-not-allowed text-bambu-gray'
- : 'hover:bg-bambu-dark text-white'
- }`}
- style={{ paddingLeft: `${12 + folder.depth * 16}px` }}
- >
- <FolderOpen className="w-4 h-4" />
- {folder.name}
- {folder.id === currentFolderId && <span className="text-xs text-bambu-gray ml-auto">({t('fileManager.current')})</span>}
- </button>
- ))}
- </div>
- <div className="flex justify-end gap-2 pt-2">
- <Button type="button" variant="secondary" onClick={onClose}>
- {t('common.cancel')}
- </Button>
- <Button onClick={() => onMove(targetFolder)} disabled={isLoading}>
- {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : t('common.move')}
- </Button>
- </div>
- </div>
- </div>
- </div>
- );
- }
- // 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<number | null>(
- 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 (
- <div className="fixed inset-0 bg-black/70 flex items-center justify-center z-50 p-4">
- <div className="bg-bambu-dark-secondary rounded-lg w-full max-w-md border border-bambu-dark-tertiary">
- <div className="p-4 border-b border-bambu-dark-tertiary flex items-center justify-between">
- <h2 className="text-lg font-semibold text-white flex items-center gap-2">
- <Link2 className="w-5 h-5 text-bambu-green" />
- {t('fileManager.linkFolder')}
- </h2>
- <button onClick={onClose} className="p-1 hover:bg-bambu-dark rounded">
- <X className="w-5 h-5 text-bambu-gray" />
- </button>
- </div>
- <div className="p-4 space-y-4">
- <p className="text-sm text-bambu-gray">
- {t('fileManager.linkFolderDescription', { name: folder.name })}
- </p>
- {/* Link type selector */}
- <div className="flex gap-2">
- <button
- onClick={() => { setLinkType('project'); setSelectedId(null); }}
- className={`flex-1 flex items-center justify-center gap-2 px-3 py-2 rounded-lg border transition-colors ${
- linkType === 'project'
- ? 'border-bambu-green bg-bambu-green/10 text-bambu-green'
- : 'border-bambu-dark-tertiary text-bambu-gray hover:text-white'
- }`}
- >
- <Briefcase className="w-4 h-4" />
- {t('fileManager.project')}
- </button>
- <button
- onClick={() => { setLinkType('archive'); setSelectedId(null); }}
- className={`flex-1 flex items-center justify-center gap-2 px-3 py-2 rounded-lg border transition-colors ${
- linkType === 'archive'
- ? 'border-bambu-green bg-bambu-green/10 text-bambu-green'
- : 'border-bambu-dark-tertiary text-bambu-gray hover:text-white'
- }`}
- >
- <ArchiveIcon className="w-4 h-4" />
- {t('fileManager.archive')}
- </button>
- </div>
- {/* Selection list */}
- <div className="max-h-64 overflow-y-auto space-y-1 bg-bambu-dark rounded-lg p-2">
- {linkType === 'project' ? (
- projects && projects.length > 0 ? (
- projects.map((project) => (
- <button
- key={project.id}
- onClick={() => setSelectedId(project.id)}
- className={`w-full text-left px-3 py-2 rounded transition-colors flex items-center gap-2 ${
- selectedId === project.id
- ? 'bg-bambu-green/20 text-bambu-green'
- : 'hover:bg-bambu-dark-tertiary text-white'
- }`}
- >
- <div
- className="w-3 h-3 rounded-full flex-shrink-0"
- style={{ backgroundColor: project.color || '#00ae42' }}
- />
- <span className="truncate">{project.name}</span>
- </button>
- ))
- ) : (
- <p className="text-sm text-bambu-gray text-center py-4">{t('fileManager.noProjectsFound')}</p>
- )
- ) : (
- archives && archives.length > 0 ? (
- archives.map((archive: Archive) => (
- <button
- key={archive.id}
- onClick={() => setSelectedId(archive.id)}
- className={`w-full text-left px-3 py-2 rounded transition-colors flex items-center gap-2 ${
- selectedId === archive.id
- ? 'bg-bambu-green/20 text-bambu-green'
- : 'hover:bg-bambu-dark-tertiary text-white'
- }`}
- >
- <FileBox className="w-4 h-4 text-bambu-gray flex-shrink-0" />
- <span className="truncate">{archive.print_name || archive.filename}</span>
- </button>
- ))
- ) : (
- <p className="text-sm text-bambu-gray text-center py-4">{t('fileManager.noArchivesFound')}</p>
- )
- )}
- </div>
- </div>
- <div className="p-4 border-t border-bambu-dark-tertiary flex justify-between">
- {isLinked && (
- <Button variant="danger" onClick={handleUnlink} disabled={isLoading}>
- <Unlink className="w-4 h-4 mr-2" />
- {t('fileManager.unlink')}
- </Button>
- )}
- <div className={`flex gap-2 ${!isLinked ? 'ml-auto' : ''}`}>
- <Button variant="secondary" onClick={onClose}>
- {t('common.cancel')}
- </Button>
- <Button onClick={handleSave} disabled={!selectedId || isLoading}>
- {isLoading ? <Loader2 className="w-4 h-4 animate-spin" /> : t('fileManager.link')}
- </Button>
- </div>
- </div>
- </div>
- </div>
- );
- }
- // 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;
- defaultExpanded?: boolean;
- hasPermission: (permission: Permission) => boolean;
- t: TFunction;
- }
- function FolderTreeItem({ folder, selectedFolderId, onSelect, onDelete, onLink, onRename, depth = 0, wrapNames = false, defaultExpanded = true, hasPermission, t }: FolderTreeItemProps) {
- const [expanded, setExpanded] = useState(defaultExpanded);
- 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 (
- <div>
- <div
- className={`group flex items-center gap-1 px-2 py-1.5 rounded cursor-pointer transition-colors ${
- selectedFolderId === folder.id
- ? 'bg-bambu-green/20 text-bambu-green'
- : 'hover:bg-bambu-dark text-white'
- }`}
- style={{ paddingLeft: `${8 + depth * 12}px` }}
- onClick={() => onSelect(folder.id)}
- >
- {hasChildren ? (
- <button
- onClick={(e) => {
- e.stopPropagation();
- setExpanded(!expanded);
- }}
- className="p-0.5 hover:bg-bambu-dark-tertiary rounded"
- >
- <ChevronRight className={`w-3.5 h-3.5 transition-transform ${expanded ? 'rotate-90' : ''}`} />
- </button>
- ) : (
- <div className="w-4.5" />
- )}
- {isExternal ? (
- <FolderSymlink className="w-4 h-4 text-purple-400 flex-shrink-0" />
- ) : (
- <FolderOpen className="w-4 h-4 text-bambu-green flex-shrink-0" />
- )}
- <span className={`text-sm flex-1 min-w-0 ${wrapNames ? 'break-all' : 'truncate'}`} title={folder.name}>{folder.name}</span>
- {/* Link indicator - clickable to change link */}
- {isLinked && (
- <button
- onClick={(e) => { e.stopPropagation(); onLink(folder); }}
- className="flex-shrink-0 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"
- title={`${folder.project_name ? `Project: ${folder.project_name}` : `Archive: ${folder.archive_name}`} (click to change)`}
- >
- <Link2 className="w-3 h-3" />
- {folder.project_name ? (
- <Briefcase className="w-3 h-3" />
- ) : (
- <ArchiveIcon className="w-3 h-3" />
- )}
- </button>
- )}
- {/* Read-only indicator for external folders */}
- {isExternal && folder.external_readonly && (
- <span title={t('fileManager.readOnly')}>
- <Lock className="w-3 h-3 text-amber-400 flex-shrink-0" />
- </span>
- )}
- {folder.file_count > 0 && (
- <span className="flex-shrink-0 text-xs text-bambu-gray">{folder.file_count}</span>
- )}
- {/* Quick link button - always visible for unlinked folders */}
- {!isLinked && !isExternal && (
- <button
- onClick={(e) => { e.stopPropagation(); onLink(folder); }}
- className="flex-shrink-0 p-1 rounded hover:bg-bambu-dark-tertiary"
- title={t('fileManager.linkToProjectOrArchive')}
- >
- <Link2 className="w-3.5 h-3.5 text-bambu-gray hover:text-bambu-green" />
- </button>
- )}
- <div className={`flex-shrink-0 flex items-center gap-0.5 transition-opacity ${wrapNames ? '' : 'opacity-0 group-hover:opacity-100'}`} onClick={(e) => e.stopPropagation()}>
- <div className="relative">
- <button
- onClick={() => setShowActions(!showActions)}
- className="p-1 rounded hover:bg-bambu-dark-tertiary"
- >
- <MoreVertical className="w-3.5 h-3.5 text-bambu-gray" />
- </button>
- {showActions && (
- <>
- <div className="fixed inset-0 z-10" onClick={() => setShowActions(false)} />
- <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]">
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('library:update_all') ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('library:update_all')) { onRename(folder); setShowActions(false); } }}
- disabled={!hasPermission('library:update_all')}
- title={!hasPermission('library:update_all') ? t('fileManager.noPermissionRenameFolder') : undefined}
- >
- <Pencil className="w-3.5 h-3.5" />
- {t('common.rename')}
- </button>
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('library:update_all') ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('library:update_all')) { onLink(folder); setShowActions(false); } }}
- disabled={!hasPermission('library:update_all')}
- title={!hasPermission('library:update_all') ? t('fileManager.noPermissionLinkFolder') : undefined}
- >
- <Link2 className="w-3.5 h-3.5" />
- {isLinked ? t('fileManager.changeLink') : t('fileManager.linkTo')}
- </button>
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('library:delete_all') ? 'text-red-400 hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('library:delete_all')) { onDelete(folder.id); setShowActions(false); } }}
- disabled={!hasPermission('library:delete_all')}
- title={!hasPermission('library:delete_all') ? t('fileManager.noPermissionDeleteFolder') : undefined}
- >
- <Trash2 className="w-3.5 h-3.5" />
- {t('common.delete')}
- </button>
- </div>
- </>
- )}
- </div>
- </div>
- </div>
- {hasChildren && expanded && (
- <div>
- {folder.children.map((child) => (
- <FolderTreeItem
- key={child.id}
- folder={child}
- selectedFolderId={selectedFolderId}
- onSelect={onSelect}
- onDelete={onDelete}
- onLink={onLink}
- onRename={onRename}
- depth={depth + 1}
- wrapNames={wrapNames}
- defaultExpanded={defaultExpanded}
- hasPermission={hasPermission}
- t={t}
- />
- ))}
- </div>
- )}
- </div>
- );
- }
- // Helper to check if a file is sliced (printable)
- function isSlicedFilename(filename: string): boolean {
- const lower = filename.toLowerCase();
- return lower.endsWith('.gcode') || lower.endsWith('.gcode.3mf');
- }
- // 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 (
- <div
- className={`group relative bg-bambu-dark-secondary rounded-lg border transition-all cursor-pointer overflow-hidden ${
- isSelected
- ? 'border-bambu-green ring-1 ring-bambu-green'
- : 'border-bambu-dark-tertiary hover:border-bambu-green/50'
- }`}
- onClick={() => onSelect(file.id)}
- >
- {/* Thumbnail */}
- <div className="aspect-square bg-bambu-dark flex items-center justify-center overflow-hidden">
- {file.thumbnail_path ? (
- <img
- src={`${api.getLibraryFileThumbnailUrl(file.id)}${thumbnailVersion ? ((api.getLibraryFileThumbnailUrl(file.id).includes('?') ? '&' : '?') + `v=${thumbnailVersion}`) : ''}`}
- alt={file.filename}
- className="w-full h-full object-cover"
- />
- ) : (
- <FileBox className="w-12 h-12 text-bambu-gray/30" />
- )}
- {/* File type badge */}
- <div className={`absolute top-2 right-2 text-xs px-1.5 py-0.5 rounded font-medium ${
- file.file_type === '3mf' ? 'bg-bambu-green/90 text-white'
- : file.file_type === 'gcode' ? 'bg-blue-500/90 text-white'
- : file.file_type === 'stl' ? 'bg-purple-500/90 text-white'
- : 'bg-bambu-gray/90 text-white'
- }`}>
- {file.file_type.toUpperCase()}
- </div>
- </div>
- {/* Info */}
- <div className="p-3">
- <h3 className="text-sm font-medium text-white truncate" title={file.print_name || file.filename}>
- {file.print_name || file.filename}
- </h3>
- <div className="flex items-center gap-3 mt-1 text-xs text-bambu-gray">
- <span>{formatFileSize(file.file_size)}</span>
- {file.print_time_seconds && (
- <span className="flex items-center gap-1">
- <Clock className="w-3 h-3" />
- {formatDuration(file.print_time_seconds)}
- </span>
- )}
- </div>
- {file.sliced_for_model && (
- <div className="mt-1 text-xs text-bambu-gray flex items-center gap-1">
- <Printer className="w-3 h-3" />
- {file.sliced_for_model}
- </div>
- )}
- {file.print_count > 0 && (
- <div className="mt-1 text-xs text-bambu-green">
- {t('fileManager.printedCount', { count: file.print_count })}
- </div>
- )}
- {authEnabled && file.created_by_username && (
- <div className="mt-1 text-xs text-bambu-gray flex items-center gap-1">
- <User className="w-3 h-3" />
- {file.created_by_username}
- </div>
- )}
- </div>
- {/* Actions - always visible on mobile, hover on desktop */}
- <div className={`absolute bottom-2 right-2 transition-opacity ${isMobile ? 'opacity-100' : 'opacity-0 group-hover:opacity-100'}`} onClick={(e) => e.stopPropagation()}>
- <button
- onClick={() => setShowActions(!showActions)}
- className="p-1.5 rounded bg-bambu-dark-secondary/90 hover:bg-bambu-dark-tertiary"
- >
- <MoreVertical className="w-4 h-4 text-bambu-gray" />
- </button>
- {showActions && (
- <>
- <div className="fixed inset-0 z-10" onClick={() => setShowActions(false)} />
- <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]">
- {onPrint && isSlicedFilename(file.filename) && (
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('printers:control') ? 'text-bambu-green hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('printers:control')) { onPrint(file); setShowActions(false); } }}
- disabled={!hasPermission('printers:control')}
- title={!hasPermission('printers:control') ? t('fileManager.noPermissionPrint') : undefined}
- >
- <Printer className="w-3.5 h-3.5" />
- {t('common.print')}
- </button>
- )}
- {onAddToQueue && isSlicedFilename(file.filename) && (
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('queue:create') ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('queue:create')) { onAddToQueue(file.id); setShowActions(false); } }}
- disabled={!hasPermission('queue:create')}
- title={!hasPermission('queue:create') ? t('fileManager.noPermissionAddToQueue') : undefined}
- >
- <Clock className="w-3.5 h-3.5" />
- {t('fileManager.schedulePrint')}
- </button>
- )}
- {onPreview3d && (file.file_type === '3mf' || file.file_type === 'gcode' || file.file_type === 'stl') && (
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('library:read') ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('library:read')) { onPreview3d(file); setShowActions(false); } }}
- disabled={!hasPermission('library:read')}
- title={!hasPermission('library:read') ? 'You do not have permission to preview files' : undefined}
- >
- <Box className="w-3.5 h-3.5" />
- 3D Preview
- </button>
- )}
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- hasPermission('library:read') ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (hasPermission('library:read')) { onDownload(file.id); setShowActions(false); } }}
- disabled={!hasPermission('library:read')}
- title={!hasPermission('library:read') ? t('fileManager.noPermissionDownload') : undefined}
- >
- <Download className="w-3.5 h-3.5" />
- {t('common.download')}
- </button>
- {onRename && (
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- canModify('library', 'update', file.created_by_id) ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (canModify('library', 'update', file.created_by_id)) { onRename(file); setShowActions(false); } }}
- disabled={!canModify('library', 'update', file.created_by_id)}
- title={!canModify('library', 'update', file.created_by_id) ? t('fileManager.noPermissionRenameFile') : undefined}
- >
- <Pencil className="w-3.5 h-3.5" />
- {t('common.rename')}
- </button>
- )}
- {onGenerateThumbnail && file.file_type === 'stl' && (
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- canModify('library', 'update', file.created_by_id) ? 'text-white hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (canModify('library', 'update', file.created_by_id)) { onGenerateThumbnail(file); setShowActions(false); } }}
- disabled={!canModify('library', 'update', file.created_by_id)}
- title={!canModify('library', 'update', file.created_by_id) ? t('fileManager.noPermissionGenerateThumbnail') : undefined}
- >
- <Image className="w-3.5 h-3.5" />
- {t('fileManager.generateThumbnail')}
- </button>
- )}
- <button
- className={`w-full px-3 py-1.5 text-left text-sm flex items-center gap-2 ${
- canModify('library', 'delete', file.created_by_id) ? 'text-red-400 hover:bg-bambu-dark' : 'text-bambu-gray cursor-not-allowed'
- }`}
- onClick={() => { if (canModify('library', 'delete', file.created_by_id)) { onDelete(file.id); setShowActions(false); } }}
- disabled={!canModify('library', 'delete', file.created_by_id)}
- title={!canModify('library', 'delete', file.created_by_id) ? t('fileManager.noPermissionDeleteFile') : undefined}
- >
- <Trash2 className="w-3.5 h-3.5" />
- {t('common.delete')}
- </button>
- </div>
- </>
- )}
- </div>
- {/* Selection checkbox - always visible on mobile, hover on desktop */}
- <div className={`absolute top-2 left-2 w-5 h-5 rounded border-2 flex items-center justify-center transition-all ${
- isSelected
- ? 'bg-bambu-green border-bambu-green'
- : `border-white/30 bg-black/30 ${isMobile ? 'opacity-100' : 'opacity-0 group-hover:opacity-100'}`
- }`}>
- {isSelected && <div className="w-2 h-2 bg-white rounded-sm" />}
- </div>
- </div>
- );
- }
- 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<number | null>(initialFolderId);
- const [selectedFiles, setSelectedFiles] = useState<number[]>([]);
- const [showNewFolderModal, setShowNewFolderModal] = useState(false);
- const [showExternalFolderModal, setShowExternalFolderModal] = useState(false);
- const [showMoveModal, setShowMoveModal] = useState(false);
- const [showUploadModal, setShowUploadModal] = useState(false);
- const [linkFolder, setLinkFolder] = useState<LibraryFolderTree | null>(null);
- const [deleteConfirm, setDeleteConfirm] = useState<{ type: 'file' | 'folder' | 'bulk'; id: number; count?: number } | null>(null);
- const [printFile, setPrintFile] = useState<LibraryFileListItem | null>(null);
- const [printMultiFile, setPrintMultiFile] = useState<LibraryFileListItem | null>(null);
- const [scheduleFile, setScheduleFile] = useState<LibraryFileListItem | null>(null);
- const [renameItem, setRenameItem] = useState<{ type: 'file' | 'folder'; id: number; name: string } | null>(null);
- const [thumbnailVersions, setThumbnailVersions] = useState<Record<number, number>>({});
- const [viewerFile, setViewerFile] = useState<LibraryFileListItem | null>(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';
- });
- const [collapseFoldersByDefault, setCollapseFoldersByDefault] = useState(() => {
- return localStorage.getItem('library-collapse-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<HTMLDivElement>(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<string>('all');
- const [filterUsername, setFilterUsername] = useState('');
- const [sortField, setSortField] = useState<SortField>(() => {
- const saved = localStorage.getItem('library-sort-field');
- return (saved as SortField) || 'name';
- });
- const [sortDirection, setSortDirection] = useState<SortDirection>(() => {
- 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<AppSettings>,
- });
- 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<number, number> = {};
- 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 (
- <div className="p-4 md:p-8 min-h-[calc(100vh-64px)] lg:h-[calc(100vh-64px)] flex flex-col">
- {/* Header */}
- <div className="flex flex-col sm:flex-row sm:items-center justify-between gap-4 mb-6">
- <div>
- <h1 className="text-2xl font-bold text-white flex items-center gap-3">
- <div className="p-2.5 bg-bambu-green/10 rounded-xl">
- <FolderOpen className="w-6 h-6 text-bambu-green" />
- </div>
- {t('fileManager.title')}
- </h1>
- <p className="text-sm text-bambu-gray mt-2 ml-14">
- {t('fileManager.subtitle')}
- </p>
- </div>
- <div className="flex items-center gap-2">
- {/* View mode toggle */}
- <div className="flex items-center bg-bambu-dark rounded-lg p-1">
- <button
- onClick={() => handleViewModeChange('grid')}
- className={`p-1.5 rounded transition-colors ${
- viewMode === 'grid' ? 'bg-bambu-dark-secondary text-white' : 'text-bambu-gray hover:text-white'
- }`}
- title={t('fileManager.gridView')}
- >
- <LayoutGrid className="w-4 h-4" />
- </button>
- <button
- onClick={() => handleViewModeChange('list')}
- className={`p-1.5 rounded transition-colors ${
- viewMode === 'list' ? 'bg-bambu-dark-secondary text-white' : 'text-bambu-gray hover:text-white'
- }`}
- title={t('fileManager.listView')}
- >
- <List className="w-4 h-4" />
- </button>
- </div>
- <Button
- variant="secondary"
- onClick={() => batchThumbnailMutation.mutate()}
- disabled={batchThumbnailMutation.isPending || !hasAnyPermission('library:update_own', 'library:update_all')}
- title={!hasAnyPermission('library:update_own', 'library:update_all') ? t('fileManager.noPermissionGenerateThumbnail') : t('fileManager.generateThumbnailsForMissing')}
- >
- {batchThumbnailMutation.isPending ? (
- <Loader2 className="w-4 h-4 mr-2 animate-spin" />
- ) : (
- <Image className="w-4 h-4 mr-2" />
- )}
- {t('fileManager.generateThumbnails')}
- </Button>
- <Button
- variant="secondary"
- onClick={() => setShowExternalFolderModal(true)}
- disabled={!hasPermission('library:upload')}
- title={!hasPermission('library:upload') ? t('fileManager.noPermissionCreateFolder') : t('fileManager.linkExternalFolder')}
- >
- <FolderSymlink className="w-4 h-4 mr-2" />
- {t('fileManager.linkExternal')}
- </Button>
- <Button
- variant="secondary"
- onClick={() => setShowNewFolderModal(true)}
- disabled={!hasPermission('library:upload')}
- title={!hasPermission('library:upload') ? t('fileManager.noPermissionCreateFolder') : undefined}
- >
- <FolderPlus className="w-4 h-4 mr-2" />
- {t('fileManager.newFolder')}
- </Button>
- <Button
- onClick={() => setShowUploadModal(true)}
- disabled={!hasPermission('library:upload')}
- title={!hasPermission('library:upload') ? t('fileManager.noPermissionUpload') : undefined}
- >
- <Upload className="w-4 h-4 mr-2" />
- {t('common.upload')}
- </Button>
- </div>
- </div>
- {/* Disk space warning */}
- {isDiskSpaceLow && stats && settings && (
- <div className="flex items-center gap-3 mb-4 p-3 bg-amber-500/10 border border-amber-500/30 rounded-lg">
- <AlertTriangle className="w-5 h-5 text-amber-500 flex-shrink-0" />
- <div className="flex-1">
- <p className="text-sm text-amber-500 font-medium">{t('fileManager.lowDiskSpaceWarning')}</p>
- <p className="text-xs text-amber-500/80">
- {t('fileManager.lowDiskSpaceDetails', { free: formatFileSize(stats.disk_free_bytes), total: formatFileSize(stats.disk_total_bytes), threshold: settings.library_disk_warning_gb })}
- </p>
- </div>
- </div>
- )}
- {/* Stats bar */}
- {stats && (
- <div className="flex flex-wrap items-center gap-3 sm:gap-6 mb-6 p-3 bg-bambu-dark-secondary rounded-lg border border-bambu-dark-tertiary">
- <div className="flex items-center gap-2 text-sm">
- <File className="w-4 h-4 text-bambu-green" />
- <span className="text-bambu-gray">{t('fileManager.files')}:</span>
- <span className="text-white font-medium">{stats.total_files}</span>
- </div>
- <div className="flex items-center gap-2 text-sm">
- <FolderOpen className="w-4 h-4 text-blue-400" />
- <span className="text-bambu-gray">{t('fileManager.folders')}:</span>
- <span className="text-white font-medium">{stats.total_folders}</span>
- </div>
- <div className="flex items-center gap-2 text-sm">
- <HardDrive className="w-4 h-4 text-amber-400" />
- <span className="text-bambu-gray">{t('fileManager.size')}:</span>
- <span className="text-white font-medium">{formatFileSize(stats.total_size_bytes)}</span>
- </div>
- <div className="flex items-center gap-2 text-sm sm:ml-auto">
- <span className="text-bambu-gray">{t('fileManager.free')}:</span>
- <span className={`font-medium ${isDiskSpaceLow ? 'text-amber-500' : 'text-white'}`}>
- {formatFileSize(stats.disk_free_bytes)}
- </span>
- </div>
- </div>
- )}
- {/* Main content */}
- <div className="flex-1 flex flex-col lg:flex-row gap-4 lg:gap-6 min-h-0">
- {/* Mobile folder selector */}
- <div className="lg:hidden">
- <select
- value={selectedFolderId ?? ''}
- onChange={(e) => setSelectedFolderId(e.target.value ? parseInt(e.target.value, 10) : null)}
- className="w-full bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg px-3 py-2.5 text-white focus:outline-none focus:border-bambu-green"
- >
- <option value="">📁 {t('fileManager.allFiles')}</option>
- {folders && (() => {
- // Flatten folder tree for mobile selector
- const flattenFolders = (items: LibraryFolderTree[], depth = 0): { id: number; name: string; fileCount: number; depth: number }[] => {
- const result: { id: number; name: string; fileCount: number; depth: number }[] = [];
- for (const item of items) {
- result.push({ id: item.id, name: item.name, fileCount: item.file_count, depth });
- if (item.children.length > 0) {
- result.push(...flattenFolders(item.children, depth + 1));
- }
- }
- return result;
- };
- return flattenFolders(folders).map((folder) => (
- <option key={folder.id} value={folder.id}>
- {'│ '.repeat(folder.depth)}📂 {folder.name} {folder.fileCount > 0 ? `(${folder.fileCount})` : ''}
- </option>
- ));
- })()}
- </select>
- </div>
- {/* Folder sidebar - resizable, hidden on mobile */}
- <div
- ref={sidebarRef}
- className="hidden lg:flex flex-shrink-0 bg-bambu-dark-secondary rounded-lg border border-bambu-dark-tertiary overflow-hidden flex-col relative"
- style={{ width: `${sidebarWidth}px` }}
- >
- {/* Resize handle - drag to resize, double-click to reset */}
- <div
- className={`absolute right-0 top-0 bottom-0 w-1.5 cursor-col-resize z-10 group/resize flex items-center justify-center transition-colors ${
- isResizing ? 'bg-bambu-green' : 'hover:bg-bambu-green/50'
- }`}
- onMouseDown={(e) => {
- e.preventDefault();
- setIsResizing(true);
- }}
- onDoubleClick={() => {
- setSidebarWidth(256); // Reset to default w-64
- localStorage.setItem('library-sidebar-width', '256');
- }}
- title={t('fileManager.dragToResizeTooltip')}
- >
- {/* Grip dots */}
- <div className={`flex flex-col gap-1 opacity-0 group-hover/resize:opacity-100 transition-opacity ${isResizing ? 'opacity-100' : ''}`}>
- <div className="w-0.5 h-0.5 rounded-full bg-white/70" />
- <div className="w-0.5 h-0.5 rounded-full bg-white/70" />
- <div className="w-0.5 h-0.5 rounded-full bg-white/70" />
- </div>
- </div>
- <div className="p-3 border-b border-bambu-dark-tertiary flex items-center justify-between">
- <h2 className="text-sm font-medium text-white">{t('fileManager.folders')}</h2>
- <div className="flex items-center gap-1">
- <button
- onClick={() => {
- const newValue = !collapseFoldersByDefault;
- setCollapseFoldersByDefault(newValue);
- localStorage.setItem('library-collapse-folders', String(newValue));
- }}
- className={`text-xs px-1.5 py-0.5 rounded transition-colors ${
- collapseFoldersByDefault
- ? 'bg-bambu-green/20 text-bambu-green'
- : 'text-bambu-gray hover:text-white hover:bg-bambu-dark'
- }`}
- title={collapseFoldersByDefault ? t('fileManager.expandFoldersByDefault') : t('fileManager.collapseFoldersByDefault')}
- >
- {t('fileManager.collapse')}
- </button>
- <button
- onClick={() => {
- const newValue = !wrapFolderNames;
- setWrapFolderNames(newValue);
- localStorage.setItem('library-wrap-folders', String(newValue));
- }}
- className={`text-xs px-1.5 py-0.5 rounded transition-colors ${
- wrapFolderNames
- ? 'bg-bambu-green/20 text-bambu-green'
- : 'text-bambu-gray hover:text-white hover:bg-bambu-dark'
- }`}
- title={wrapFolderNames ? t('fileManager.disableTextWrapping') : t('fileManager.enableTextWrapping')}
- >
- {t('fileManager.wrap')}
- </button>
- </div>
- </div>
- <div className="flex-1 overflow-y-auto p-2">
- {/* All Files (root) */}
- <div
- className={`flex items-center gap-2 px-2 py-1.5 rounded cursor-pointer transition-colors ${
- selectedFolderId === null
- ? 'bg-bambu-green/20 text-bambu-green'
- : 'hover:bg-bambu-dark text-white'
- }`}
- onClick={() => setSelectedFolderId(null)}
- >
- <FileBox className="w-4 h-4" />
- <span className="text-sm">{t('fileManager.allFiles')}</span>
- </div>
- {/* Folder tree — re-key on the collapse toggle so flipping it
- remounts every FolderTreeItem, which re-reads defaultExpanded
- and makes the preference take effect immediately. */}
- {folders?.map((folder) => (
- <FolderTreeItem
- key={`${folder.id}-${collapseFoldersByDefault ? 'c' : 'e'}`}
- folder={folder}
- selectedFolderId={selectedFolderId}
- onSelect={setSelectedFolderId}
- onDelete={(id) => setDeleteConfirm({ type: 'folder', id })}
- onLink={setLinkFolder}
- onRename={(f) => setRenameItem({ type: 'folder', id: f.id, name: f.name })}
- wrapNames={wrapFolderNames}
- defaultExpanded={!collapseFoldersByDefault}
- hasPermission={hasPermission}
- t={t}
- />
- ))}
- </div>
- </div>
- {/* Files area */}
- <div className="flex-1 flex flex-col min-w-0 min-h-0">
- {/* External folder info bar */}
- {selectedFolder?.is_external && (
- <div className="flex items-center gap-3 mb-4 p-3 bg-purple-500/10 border border-purple-500/30 rounded-lg">
- <FolderSymlink className="w-5 h-5 text-purple-400 flex-shrink-0" />
- <div className="flex-1 min-w-0">
- <div className="flex items-center gap-2">
- <span className="text-sm font-medium text-purple-300">{t('fileManager.externalFolder')}</span>
- {selectedFolder.external_readonly && (
- <span className="text-xs px-1.5 py-0.5 rounded bg-amber-500/20 text-amber-400 flex items-center gap-1">
- <Lock className="w-3 h-3" />
- {t('fileManager.readOnly')}
- </span>
- )}
- </div>
- <p className="text-xs text-bambu-gray truncate font-mono" title={selectedFolder.external_path || ''}>
- {selectedFolder.external_path}
- </p>
- </div>
- <Button
- variant="secondary"
- size="sm"
- onClick={() => selectedFolderId && scanExternalFolderMutation.mutate(selectedFolderId)}
- disabled={scanExternalFolderMutation.isPending}
- title={t('fileManager.scanFolder')}
- >
- {scanExternalFolderMutation.isPending ? (
- <Loader2 className="w-4 h-4 animate-spin" />
- ) : (
- <RefreshCw className="w-4 h-4" />
- )}
- <span className="ml-1.5">{t('fileManager.scanFolder')}</span>
- </Button>
- </div>
- )}
- {/* Search, Filter, Sort toolbar - sticky on mobile for easier access */}
- {files && files.length > 0 && (
- <div className="flex flex-wrap items-center gap-2 sm:gap-3 mb-4 p-2 sm:p-3 bg-bambu-dark-secondary rounded-lg border border-bambu-dark-tertiary sticky top-0 z-10 lg:static">
- {/* Search */}
- <div className="relative w-full sm:w-auto sm:flex-1 sm:max-w-xs">
- <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray" />
- <input
- type="text"
- placeholder={t('fileManager.searchFiles')}
- value={searchQuery}
- onChange={(e) => 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"
- />
- </div>
- {/* Type filter */}
- <div className="flex items-center gap-2">
- <Filter className="w-4 h-4 text-bambu-gray hidden sm:block" />
- <select
- value={filterType}
- onChange={(e) => setFilterType(e.target.value)}
- 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"
- >
- <option value="all">{t('fileManager.allTypes')}</option>
- {fileTypes.map((type) => (
- <option key={type} value={type}>
- {type.toUpperCase()}
- </option>
- ))}
- </select>
- </div>
- {/* Username filter with autocomplete - only show when auth is enabled */}
- {authEnabled && (
- <div className="relative">
- <input
- type="text"
- placeholder={t('fileManager.filterByUser', { defaultValue: 'Filter by user' })}
- value={filterUsername}
- onChange={(e) => 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 && (
- <button
- onClick={() => setFilterUsername('')}
- className="absolute right-2 top-1/2 -translate-y-1/2 text-bambu-gray hover:text-white z-10"
- >
- <X className="w-3 h-3" />
- </button>
- )}
- <datalist id="usernames-list">
- {users?.map((user) => (
- <option key={user.id} value={user.username} />
- ))}
- </datalist>
- </div>
- )}
- {/* Sort */}
- <div className="flex items-center gap-2">
- <select
- value={sortField}
- onChange={(e) => {
- const newField = e.target.value as SortField;
- setSortField(newField);
- localStorage.setItem('library-sort-field', newField);
- }}
- 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"
- >
- <option value="name">{t('common.name')}</option>
- <option value="date">{t('common.date')}</option>
- <option value="size">{t('fileManager.size')}</option>
- <option value="type">{t('common.type')}</option>
- <option value="prints">{t('fileManager.prints')}</option>
- </select>
- <button
- onClick={() => setSortDirection((d) => {
- const newDir = d === 'asc' ? 'desc' : 'asc';
- localStorage.setItem('library-sort-direction', newDir);
- return newDir;
- })}
- className="p-1.5 rounded bg-bambu-dark border border-bambu-dark-tertiary hover:border-bambu-green transition-colors"
- title={sortDirection === 'asc' ? t('fileManager.ascending') : t('fileManager.descending')}
- >
- {sortDirection === 'asc' ? (
- <SortAsc className="w-4 h-4 text-white" />
- ) : (
- <SortDesc className="w-4 h-4 text-white" />
- )}
- </button>
- </div>
- {/* Results count */}
- {(searchQuery || filterType !== 'all' || filterUsername) && (
- <span className="text-sm text-bambu-gray hidden sm:inline">
- {t('fileManager.resultsCount', { showing: filteredAndSortedFiles.length, total: files.length })}
- </span>
- )}
- </div>
- )}
- {/* Selection toolbar - sticky on mobile below search bar */}
- {filteredAndSortedFiles.length > 0 && (
- <div className="flex flex-wrap items-center gap-2 mb-4 p-2 bg-bambu-dark-secondary rounded-lg border border-bambu-dark-tertiary sticky top-[52px] z-10 lg:static">
- {/* Select all / Deselect all */}
- {selectedFiles.length === filteredAndSortedFiles.length && selectedFiles.length > 0 ? (
- <Button
- variant="secondary"
- size="sm"
- onClick={handleDeselectAll}
- >
- <Square className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('fileManager.deselectAll')}</span>
- </Button>
- ) : (
- <Button
- variant="secondary"
- size="sm"
- onClick={handleSelectAll}
- >
- <CheckSquare className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('fileManager.selectAll')}</span>
- </Button>
- )}
- {selectedFiles.length > 0 && (
- <>
- <span className="text-sm text-bambu-gray ml-2">
- {t('fileManager.selected', { count: selectedFiles.length })}
- </span>
- <div className="hidden sm:block flex-1" />
- <div className="w-full sm:w-auto flex flex-wrap items-center gap-2 mt-2 sm:mt-0">
- {selectedSlicedFiles.length === 1 && (
- <Button
- variant="primary"
- size="sm"
- onClick={() => setPrintMultiFile(selectedSlicedFiles[0])}
- disabled={!hasPermission('printers:control')}
- title={!hasPermission('printers:control') ? t('fileManager.noPermissionPrint') : undefined}
- >
- <Play className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('common.print')}</span>
- </Button>
- )}
- {selectedSlicedFiles.length === 1 && (
- <Button
- variant="secondary"
- size="sm"
- // Note: Schedule dialog (PrintModal) is designed for single file at a time
- // but supports scheduling to multiple printers. This provides more control
- // over scheduling options compared to the previous bulk queue mutation.
- onClick={() => setScheduleFile(selectedSlicedFiles[0])}
- disabled={!hasPermission('queue:create')}
- title={!hasPermission('queue:create') ? t('fileManager.noPermissionAddToQueue') : undefined}
- >
- <Clock className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('fileManager.schedulePrint')}</span>
- </Button>
- )}
- <Button
- variant="secondary"
- size="sm"
- onClick={() => setShowMoveModal(true)}
- disabled={!hasAnyPermission('library:update_own', 'library:update_all')}
- title={!hasAnyPermission('library:update_own', 'library:update_all') ? t('fileManager.noPermissionMoveFiles') : undefined}
- >
- <MoveRight className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('common.move')}</span>
- </Button>
- <Button
- variant="danger"
- size="sm"
- onClick={() => {
- if (selectedFiles.length === 1) {
- setDeleteConfirm({ type: 'file', id: selectedFiles[0] });
- } else {
- setDeleteConfirm({ type: 'bulk', id: 0, count: selectedFiles.length });
- }
- }}
- disabled={!hasAnyPermission('library:delete_own', 'library:delete_all')}
- title={!hasAnyPermission('library:delete_own', 'library:delete_all') ? t('fileManager.noPermissionDeleteFiles') : undefined}
- >
- <Trash2 className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('common.delete')}</span>
- </Button>
- <Button
- variant="secondary"
- size="sm"
- onClick={handleDeselectAll}
- >
- <X className="w-4 h-4 sm:mr-1" />
- <span className="hidden sm:inline">{t('common.clear')}</span>
- </Button>
- </div>
- </>
- )}
- </div>
- )}
- {/* File grid/list */}
- {isLoading ? (
- <div className="flex-1 flex items-center justify-center">
- <div className="flex flex-col items-center gap-3">
- <Loader2 className="w-8 h-8 animate-spin text-bambu-green" />
- <p className="text-sm text-bambu-gray">{t('fileManager.loadingFiles')}</p>
- </div>
- </div>
- ) : files?.length === 0 ? (
- <div className="flex-1 flex flex-col items-center justify-center">
- <div className="p-4 bg-bambu-dark rounded-2xl mb-4">
- <FileBox className="w-12 h-12 text-bambu-gray/50" />
- </div>
- <h3 className="text-lg font-medium text-white mb-2">
- {selectedFolderId !== null ? t('fileManager.folderIsEmpty') : t('fileManager.noFilesYet')}
- </h3>
- <p className="text-bambu-gray text-center max-w-md mb-6">
- {selectedFolderId !== null
- ? t('fileManager.folderEmptyDescription')
- : t('fileManager.noFilesDescription')}
- </p>
- <Button
- onClick={() => setShowUploadModal(true)}
- disabled={!hasPermission('library:upload')}
- title={!hasPermission('library:upload') ? t('fileManager.noPermissionUpload') : undefined}
- >
- <Plus className="w-4 h-4 mr-2" />
- {t('fileManager.uploadFiles')}
- </Button>
- </div>
- ) : filteredAndSortedFiles.length === 0 ? (
- <div className="flex-1 flex flex-col items-center justify-center">
- <div className="p-4 bg-bambu-dark rounded-2xl mb-4">
- <Search className="w-12 h-12 text-bambu-gray/50" />
- </div>
- <h3 className="text-lg font-medium text-white mb-2">{t('fileManager.noMatchingFiles')}</h3>
- <p className="text-bambu-gray text-center max-w-md mb-6">
- {t('fileManager.noMatchingFilesDescription')}
- </p>
- <Button variant="secondary" onClick={() => { setSearchQuery(''); setFilterType('all'); }}>
- {t('fileManager.clearFilters')}
- </Button>
- </div>
- ) : viewMode === 'grid' ? (
- <div className="flex-1 lg:overflow-y-auto">
- <div className="grid grid-cols-2 sm:grid-cols-3 lg:grid-cols-4 xl:grid-cols-5 2xl:grid-cols-6 gap-4">
- {filteredAndSortedFiles.map((file) => (
- <FileCard
- key={file.id}
- file={file}
- isSelected={selectedFiles.includes(file.id)}
- isMobile={isMobile}
- t={t}
- onSelect={handleFileSelect}
- onDelete={(id) => 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}
- />
- ))}
- </div>
- </div>
- ) : (
- <div className="flex-1 lg:overflow-y-auto">
- <div className="bg-bambu-dark-secondary rounded-lg border border-bambu-dark-tertiary overflow-hidden">
- {/* List header - hidden on mobile, show simplified on small screens */}
- <div className={`hidden sm:grid ${authEnabled ? 'grid-cols-[auto_1fr_120px_100px_100px_100px_80px]' : '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`}>
- <div className="w-6" />
- <div>{t('common.name')}</div>
- {authEnabled && <div>{t('fileManager.uploadedBy', { defaultValue: 'Uploaded By' })}</div>}
- <div>{t('common.type')}</div>
- <div>{t('fileManager.size')}</div>
- <div>{t('fileManager.prints')}</div>
- <div />
- </div>
- {/* List rows */}
- {filteredAndSortedFiles.map((file) => (
- <div
- key={file.id}
- className={`grid ${authEnabled ? 'grid-cols-[auto_1fr_120px_100px_100px_100px_80px]' : '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 ${
- selectedFiles.includes(file.id) ? 'bg-bambu-green/10' : ''
- }`}
- onClick={() => handleFileSelect(file.id)}
- >
- {/* Checkbox */}
- <div className={`w-5 h-5 rounded border-2 flex items-center justify-center ${
- selectedFiles.includes(file.id)
- ? 'bg-bambu-green border-bambu-green'
- : 'border-bambu-gray/50'
- }`}>
- {selectedFiles.includes(file.id) && <div className="w-2 h-2 bg-white rounded-sm" />}
- </div>
- {/* Name with thumbnail */}
- <div className="flex items-center gap-3 min-w-0">
- <div className="relative group/thumb">
- <div className="w-10 h-10 rounded bg-bambu-dark flex-shrink-0 overflow-hidden">
- {file.thumbnail_path ? (
- <img
- src={`${api.getLibraryFileThumbnailUrl(file.id)}${thumbnailVersions[file.id] ? ((api.getLibraryFileThumbnailUrl(file.id).includes('?') ? '&' : '?') + `v=${thumbnailVersions[file.id]}`) : ''}`}
- alt=""
- className="w-full h-full object-cover"
- />
- ) : (
- <div className="w-full h-full flex items-center justify-center">
- <FileBox className="w-5 h-5 text-bambu-gray/50" />
- </div>
- )}
- </div>
- {/* Hover preview */}
- {file.thumbnail_path && (
- <div className="absolute left-0 top-full mt-2 z-50 hidden group-hover/thumb:block">
- <div className="w-48 h-48 rounded-lg bg-bambu-dark-secondary border border-bambu-dark-tertiary shadow-xl overflow-hidden">
- <img
- src={`${api.getLibraryFileThumbnailUrl(file.id)}${thumbnailVersions[file.id] ? ((api.getLibraryFileThumbnailUrl(file.id).includes('?') ? '&' : '?') + `v=${thumbnailVersions[file.id]}`) : ''}`}
- alt={file.filename}
- className="w-full h-full object-contain"
- />
- </div>
- </div>
- )}
- </div>
- <div className="min-w-0">
- <div className="text-sm text-white truncate">{file.print_name || file.filename}</div>
- </div>
- </div>
- {/* Uploaded By - only show when auth is enabled */}
- {authEnabled && (
- <div className="text-sm text-bambu-gray flex items-center gap-1">
- {file.created_by_username ? (
- <>
- <User className="w-3 h-3" />
- <span className="truncate">{file.created_by_username}</span>
- </>
- ) : (
- '-'
- )}
- </div>
- )}
- {/* Type */}
- <div>
- <span className={`text-xs px-1.5 py-0.5 rounded font-medium ${
- file.file_type === '3mf' ? 'bg-bambu-green/20 text-bambu-green'
- : file.file_type === 'gcode' ? 'bg-blue-500/20 text-blue-400'
- : file.file_type === 'stl' ? 'bg-purple-500/20 text-purple-400'
- : 'bg-bambu-gray/20 text-bambu-gray'
- }`}>
- {file.file_type.toUpperCase()}
- </span>
- </div>
- {/* Size */}
- <div className="text-sm text-bambu-gray">{formatFileSize(file.file_size)}</div>
- {/* Prints */}
- <div className="text-sm text-bambu-gray">{file.print_count > 0 ? `${file.print_count}x` : '-'}</div>
- {/* Actions */}
- <div className="flex items-center gap-1" onClick={(e) => e.stopPropagation()}>
- {isSlicedFilename(file.filename) && (
- <>
- <button
- onClick={() => hasPermission('printers:control') && setPrintFile(file)}
- className={`p-1.5 rounded transition-colors ${
- hasPermission('printers:control')
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-bambu-green'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={hasPermission('printers:control') ? t('common.print') : t('fileManager.noPermissionPrint')}
- disabled={!hasPermission('printers:control')}
- >
- <Printer className="w-4 h-4" />
- </button>
- <button
- onClick={() => {
- if (hasPermission('queue:create')) {
- setScheduleFile(file);
- }
- }}
- className={`p-1.5 rounded transition-colors ${
- hasPermission('queue:create')
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-white'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={hasPermission('queue:create') ? t('fileManager.schedulePrint') : t('fileManager.noPermissionAddToQueue')}
- disabled={!hasPermission('queue:create')}
- >
- <Clock className="w-4 h-4" />
- </button>
- </>
- )}
- {(file.file_type === '3mf' || file.file_type === 'gcode' || file.file_type === 'stl') && (
- <button
- onClick={() => hasPermission('library:read') && setViewerFile(file)}
- className={`p-1.5 rounded transition-colors ${
- hasPermission('library:read')
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-bambu-green'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={hasPermission('library:read') ? '3D Preview' : 'You do not have permission to preview files'}
- disabled={!hasPermission('library:read')}
- >
- <Box className="w-4 h-4" />
- </button>
- )}
- <button
- onClick={() => hasPermission('library:read') && handleDownload(file.id)}
- className={`p-1.5 rounded transition-colors ${
- hasPermission('library:read')
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-white'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={hasPermission('library:read') ? t('common.download') : t('fileManager.noPermissionDownload')}
- disabled={!hasPermission('library:read')}
- >
- <Download className="w-4 h-4" />
- </button>
- <button
- onClick={() => canModify('library', 'update', file.created_by_id) && setRenameItem({ type: 'file', id: file.id, name: file.filename })}
- className={`p-1.5 rounded transition-colors ${
- canModify('library', 'update', file.created_by_id)
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-white'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={canModify('library', 'update', file.created_by_id) ? t('common.rename') : t('fileManager.noPermissionRenameFile')}
- disabled={!canModify('library', 'update', file.created_by_id)}
- >
- <Pencil className="w-4 h-4" />
- </button>
- {file.file_type === 'stl' && (
- <button
- onClick={() => canModify('library', 'update', file.created_by_id) && singleThumbnailMutation.mutate(file.id)}
- className={`p-1.5 rounded transition-colors ${
- canModify('library', 'update', file.created_by_id)
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-bambu-green'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={canModify('library', 'update', file.created_by_id) ? t('fileManager.generateThumbnail') : t('fileManager.noPermissionGenerateThumbnail')}
- disabled={singleThumbnailMutation.isPending || !canModify('library', 'update', file.created_by_id)}
- >
- <Image className="w-4 h-4" />
- </button>
- )}
- <button
- onClick={() => canModify('library', 'delete', file.created_by_id) && setDeleteConfirm({ type: 'file', id: file.id })}
- className={`p-1.5 rounded transition-colors ${
- canModify('library', 'delete', file.created_by_id)
- ? 'hover:bg-bambu-dark text-bambu-gray hover:text-red-400'
- : 'text-bambu-gray/50 cursor-not-allowed'
- }`}
- title={canModify('library', 'delete', file.created_by_id) ? t('common.delete') : t('fileManager.noPermissionDeleteFile')}
- disabled={!canModify('library', 'delete', file.created_by_id)}
- >
- <Trash2 className="w-4 h-4" />
- </button>
- </div>
- </div>
- ))}
- </div>
- </div>
- )}
- </div>
- </div>
- {/* Modals */}
- {showNewFolderModal && (
- <NewFolderModal
- parentId={selectedFolderId}
- onClose={() => setShowNewFolderModal(false)}
- onSave={(data) => createFolderMutation.mutate(data)}
- isLoading={createFolderMutation.isPending}
- t={t}
- />
- )}
- {showExternalFolderModal && (
- <ExternalFolderModal
- onClose={() => setShowExternalFolderModal(false)}
- onSave={(data) => createExternalFolderMutation.mutate(data)}
- isLoading={createExternalFolderMutation.isPending}
- t={t}
- />
- )}
- {showMoveModal && folders && (
- <MoveFilesModal
- folders={folders}
- selectedFiles={selectedFiles}
- currentFolderId={selectedFolderId}
- onClose={() => setShowMoveModal(false)}
- onMove={(folderId) => moveFilesMutation.mutate({ fileIds: selectedFiles, folderId })}
- isLoading={moveFilesMutation.isPending}
- t={t}
- />
- )}
- {showUploadModal && (
- <FileUploadModal
- folderId={selectedFolderId}
- onClose={() => setShowUploadModal(false)}
- onUploadComplete={handleUploadComplete}
- />
- )}
- {linkFolder && (
- <LinkFolderModal
- folder={linkFolder}
- onClose={() => setLinkFolder(null)}
- onLink={(data) => updateFolderMutation.mutate({ id: linkFolder.id, data })}
- isLoading={updateFolderMutation.isPending}
- t={t}
- />
- )}
- {deleteConfirm && (
- <ConfirmModal
- title={
- deleteConfirm.type === 'folder'
- ? t('fileManager.deleteFolder')
- : deleteConfirm.type === 'bulk'
- ? t('fileManager.deleteFilesCount', { count: deleteConfirm.count })
- : t('fileManager.deleteFile')
- }
- message={
- deleteConfirm.type === 'folder'
- ? t('fileManager.deleteFolderConfirm')
- : deleteConfirm.type === 'bulk'
- ? t('fileManager.deleteFilesConfirm', { count: deleteConfirm.count })
- : t('fileManager.deleteFileConfirm')
- }
- confirmText={t('common.delete')}
- variant="danger"
- isLoading={isDeleting}
- loadingText={t('fileManager.deleting')}
- onConfirm={handleDeleteConfirm}
- onCancel={() => setDeleteConfirm(null)}
- />
- )}
- {printFile && (
- <PrintModal
- mode="reprint"
- libraryFileId={printFile.id}
- archiveName={printFile.print_name || printFile.filename}
- onClose={() => setPrintFile(null)}
- onSuccess={() => {
- setPrintFile(null);
- queryClient.invalidateQueries({ queryKey: ['library-files'] });
- queryClient.invalidateQueries({ queryKey: ['archives'] });
- }}
- />
- )}
- {printMultiFile && (
- <PrintModal
- mode="reprint"
- libraryFileId={printMultiFile.id}
- archiveName={printMultiFile.print_name || printMultiFile.filename}
- onClose={() => setPrintMultiFile(null)}
- onSuccess={() => {
- setPrintMultiFile(null);
- setSelectedFiles([]);
- queryClient.invalidateQueries({ queryKey: ['library-files'] });
- queryClient.invalidateQueries({ queryKey: ['archives'] });
- }}
- />
- )}
- {scheduleFile && (
- <PrintModal
- mode="add-to-queue"
- libraryFileId={scheduleFile.id}
- archiveName={scheduleFile.print_name || scheduleFile.filename}
- onClose={() => setScheduleFile(null)}
- onSuccess={() => {
- setScheduleFile(null);
- setSelectedFiles([]);
- queryClient.invalidateQueries({ queryKey: ['library-files'] });
- queryClient.invalidateQueries({ queryKey: ['queue'] });
- queryClient.invalidateQueries({ queryKey: ['archives'] });
- }}
- />
- )}
- {viewerFile && (
- <ModelViewerModal
- libraryFileId={viewerFile.id}
- title={viewerFile.print_name || viewerFile.filename}
- fileType={viewerFile.file_type}
- onClose={() => setViewerFile(null)}
- />
- )}
- {renameItem && (
- <RenameModal
- type={renameItem.type}
- currentName={renameItem.name}
- onClose={() => 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}
- />
- )}
- </div>
- );
- }
|