QueuePage.tsx 65 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534
  1. import { useState, useMemo, useEffect, useCallback } from 'react';
  2. import { useTranslation } from 'react-i18next';
  3. import { useQuery, useQueries, useMutation, useQueryClient } from '@tanstack/react-query';
  4. import { Link } from 'react-router-dom';
  5. import {
  6. DndContext,
  7. closestCenter,
  8. KeyboardSensor,
  9. PointerSensor,
  10. useSensor,
  11. useSensors,
  12. } from '@dnd-kit/core';
  13. import type { DragEndEvent } from '@dnd-kit/core';
  14. import {
  15. arrayMove,
  16. SortableContext,
  17. sortableKeyboardCoordinates,
  18. useSortable,
  19. verticalListSortingStrategy,
  20. } from '@dnd-kit/sortable';
  21. import { CSS } from '@dnd-kit/utilities';
  22. import {
  23. Clock,
  24. Trash2,
  25. Play,
  26. X,
  27. CheckCircle,
  28. XCircle,
  29. AlertCircle,
  30. Calendar,
  31. Printer,
  32. GripVertical,
  33. SkipForward,
  34. ExternalLink,
  35. Power,
  36. StopCircle,
  37. Pencil,
  38. RefreshCw,
  39. Timer,
  40. ListOrdered,
  41. Layers,
  42. ArrowUp,
  43. ArrowDown,
  44. Hand,
  45. Check,
  46. CheckSquare,
  47. Square,
  48. User,
  49. Pause,
  50. Weight,
  51. ChevronDown,
  52. ChevronRight,
  53. List,
  54. GanttChart,
  55. Code,
  56. Snail,
  57. } from 'lucide-react';
  58. import { api } from '../api/client';
  59. import { type TimeFormat, formatETA, formatDuration, formatRelativeTime, parseUTCDate } from '../utils/date';
  60. import type { PrintQueueItem, PrintQueueBulkUpdate, Permission } from '../api/client';
  61. import { Card } from '../components/Card';
  62. import { Button } from '../components/Button';
  63. import { ConfirmModal } from '../components/ConfirmModal';
  64. import { PrintModal } from '../components/PrintModal';
  65. import { useToast } from '../contexts/ToastContext';
  66. import { useAuth } from '../contexts/AuthContext';
  67. import { QueueStatsBar } from '../components/QueueStatsBar';
  68. import { CompactHistoryRow } from '../components/CompactHistoryRow';
  69. import { QueueTimelineView } from '../components/QueueTimelineView';
  70. function formatWeight(g: number, useKg = false): string {
  71. if (useKg && g >= 1000) return `${(g / 1000).toFixed(1)}kg`;
  72. return `${Math.round(g)}g`;
  73. }
  74. function StatusBadge({ status, waitingReason, printerState, t }: { status: PrintQueueItem['status']; waitingReason?: string | null; printerState?: string | null; t: (key: string) => string }) {
  75. // Special case: pending with waiting_reason shows as "Waiting"
  76. if (status === 'pending' && waitingReason) {
  77. return (
  78. <span className="inline-flex items-center gap-1.5 px-2.5 py-1 rounded-full text-xs font-medium border text-purple-400 bg-purple-400/10 border-purple-400/20">
  79. <Clock className="w-3.5 h-3.5" />
  80. {t('queue.status.waiting')}
  81. </span>
  82. );
  83. }
  84. // Special case: printing but printer is paused
  85. if (status === 'printing' && printerState === 'PAUSE') {
  86. return (
  87. <span className="inline-flex items-center gap-1.5 px-2.5 py-1 rounded-full text-xs font-medium border text-yellow-400 bg-yellow-400/10 border-yellow-400/20">
  88. <Pause className="w-3.5 h-3.5" />
  89. {t('queue.status.paused')}
  90. </span>
  91. );
  92. }
  93. const config = {
  94. pending: { icon: Clock, color: 'text-status-warning bg-status-warning/10 border-status-warning/20', label: t('queue.status.pending') },
  95. printing: { icon: Play, color: 'text-blue-400 bg-blue-400/10 border-blue-400/20', label: t('queue.status.printing') },
  96. completed: { icon: CheckCircle, color: 'text-status-ok bg-status-ok/10 border-status-ok/20', label: t('queue.status.completed') },
  97. failed: { icon: XCircle, color: 'text-status-error bg-status-error/10 border-status-error/20', label: t('queue.status.failed') },
  98. skipped: { icon: SkipForward, color: 'text-orange-400 bg-orange-400/10 border-orange-400/20', label: t('queue.status.skipped') },
  99. cancelled: { icon: X, color: 'text-gray-400 bg-gray-400/10 border-gray-400/20', label: t('queue.status.cancelled') },
  100. };
  101. const { icon: Icon, color, label } = config[status];
  102. return (
  103. <span className={`inline-flex items-center gap-1.5 px-2.5 py-1 rounded-full text-xs font-medium border ${color}`}>
  104. <Icon className="w-3.5 h-3.5" />
  105. {label}
  106. </span>
  107. );
  108. }
  109. // Bulk edit modal for multiple queue items
  110. function BulkEditModal({
  111. selectedCount,
  112. printers,
  113. onSave,
  114. onClose,
  115. isSaving,
  116. canControlPrinter,
  117. t,
  118. }: {
  119. selectedCount: number;
  120. printers: { id: number; name: string }[];
  121. onSave: (data: Partial<PrintQueueBulkUpdate>) => void;
  122. onClose: () => void;
  123. isSaving: boolean;
  124. canControlPrinter: boolean;
  125. t: (key: string, options?: Record<string, unknown>) => string;
  126. }) {
  127. const [printerId, setPrinterId] = useState<number | null | 'unchanged'>('unchanged');
  128. const [manualStart, setManualStart] = useState<boolean | 'unchanged'>('unchanged');
  129. const [autoOffAfter, setAutoOffAfter] = useState<boolean | 'unchanged'>('unchanged');
  130. const [requirePreviousSuccess, setRequirePreviousSuccess] = useState<boolean | 'unchanged'>('unchanged');
  131. const [bedLevelling, setBedLevelling] = useState<boolean | 'unchanged'>('unchanged');
  132. const [flowCali, setFlowCali] = useState<boolean | 'unchanged'>('unchanged');
  133. const [vibrationCali, setVibrationCali] = useState<boolean | 'unchanged'>('unchanged');
  134. const [layerInspect, setLayerInspect] = useState<boolean | 'unchanged'>('unchanged');
  135. const [timelapse, setTimelapse] = useState<boolean | 'unchanged'>('unchanged');
  136. const [useAms, setUseAms] = useState<boolean | 'unchanged'>('unchanged');
  137. const handleSave = () => {
  138. const data: Partial<PrintQueueBulkUpdate> = {};
  139. if (printerId !== 'unchanged') data.printer_id = printerId;
  140. if (manualStart !== 'unchanged') data.manual_start = manualStart;
  141. if (autoOffAfter !== 'unchanged') data.auto_off_after = autoOffAfter;
  142. if (requirePreviousSuccess !== 'unchanged') data.require_previous_success = requirePreviousSuccess;
  143. if (bedLevelling !== 'unchanged') data.bed_levelling = bedLevelling;
  144. if (flowCali !== 'unchanged') data.flow_cali = flowCali;
  145. if (vibrationCali !== 'unchanged') data.vibration_cali = vibrationCali;
  146. if (layerInspect !== 'unchanged') data.layer_inspect = layerInspect;
  147. if (timelapse !== 'unchanged') data.timelapse = timelapse;
  148. if (useAms !== 'unchanged') data.use_ams = useAms;
  149. onSave(data);
  150. };
  151. const hasChanges = printerId !== 'unchanged' || manualStart !== 'unchanged' || autoOffAfter !== 'unchanged' ||
  152. requirePreviousSuccess !== 'unchanged' || bedLevelling !== 'unchanged' || flowCali !== 'unchanged' ||
  153. vibrationCali !== 'unchanged' || layerInspect !== 'unchanged' || timelapse !== 'unchanged' || useAms !== 'unchanged';
  154. return (
  155. <div className="fixed inset-0 z-50 flex items-center justify-center p-4 bg-black/50 backdrop-blur-sm">
  156. <div className="bg-bambu-dark-secondary rounded-xl border border-bambu-dark-tertiary w-full max-w-lg max-h-[90vh] overflow-y-auto">
  157. <div className="flex items-center justify-between p-4 border-b border-bambu-dark-tertiary">
  158. <h2 className="text-lg font-semibold text-white">
  159. {t('queue.bulkEdit.title', { count: selectedCount })}
  160. </h2>
  161. <button onClick={onClose} className="p-1 hover:bg-bambu-dark rounded">
  162. <X className="w-5 h-5 text-bambu-gray" />
  163. </button>
  164. </div>
  165. <div className="p-4 space-y-4">
  166. <p className="text-sm text-bambu-gray">
  167. {t('queue.bulkEdit.description')}
  168. </p>
  169. {/* Printer Assignment */}
  170. <div>
  171. <label className="block text-sm font-medium text-white mb-2">{t('queue.bulkEdit.printer')}</label>
  172. <select
  173. value={printerId === null ? 'null' : printerId === 'unchanged' ? 'unchanged' : String(printerId)}
  174. onChange={(e) => {
  175. const val = e.target.value;
  176. if (val === 'unchanged') setPrinterId('unchanged');
  177. else if (val === 'null') setPrinterId(null);
  178. else setPrinterId(Number(val));
  179. }}
  180. className="w-full px-3 py-2 bg-bambu-dark border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  181. >
  182. <option value="unchanged">{t('queue.bulkEdit.noChange')}</option>
  183. <option value="null">{t('queue.filter.unassigned')}</option>
  184. {printers.map(p => (
  185. <option key={p.id} value={p.id}>{p.name}</option>
  186. ))}
  187. </select>
  188. </div>
  189. {/* Queue Options */}
  190. <div>
  191. <label className="block text-sm font-medium text-white mb-2">{t('queue.bulkEdit.queueOptions')}</label>
  192. <div className="space-y-2">
  193. <TriStateToggle label={t('queue.bulkEdit.staged')} value={manualStart} onChange={setManualStart} t={t} />
  194. <TriStateToggle label={t('queue.bulkEdit.autoPowerOff')} value={autoOffAfter} onChange={setAutoOffAfter} disabled={!canControlPrinter} t={t} />
  195. <TriStateToggle label={t('queue.bulkEdit.requirePrevious')} value={requirePreviousSuccess} onChange={setRequirePreviousSuccess} t={t} />
  196. </div>
  197. </div>
  198. {/* Print Options */}
  199. <div>
  200. <label className="block text-sm font-medium text-white mb-2">{t('queue.bulkEdit.printOptions')}</label>
  201. <div className="space-y-2">
  202. <TriStateToggle label={t('queue.bulkEdit.bedLevelling')} value={bedLevelling} onChange={setBedLevelling} t={t} />
  203. <TriStateToggle label={t('queue.bulkEdit.flowCalibration')} value={flowCali} onChange={setFlowCali} t={t} />
  204. <TriStateToggle label={t('queue.bulkEdit.vibrationCalibration')} value={vibrationCali} onChange={setVibrationCali} t={t} />
  205. <TriStateToggle label={t('queue.bulkEdit.layerInspection')} value={layerInspect} onChange={setLayerInspect} t={t} />
  206. <TriStateToggle label={t('queue.bulkEdit.timelapse')} value={timelapse} onChange={setTimelapse} t={t} />
  207. <TriStateToggle label={t('queue.bulkEdit.useAms')} value={useAms} onChange={setUseAms} t={t} />
  208. </div>
  209. </div>
  210. </div>
  211. <div className="flex justify-end gap-3 p-4 border-t border-bambu-dark-tertiary">
  212. <Button variant="secondary" onClick={onClose}>{t('common.cancel')}</Button>
  213. <Button
  214. onClick={handleSave}
  215. disabled={!hasChanges || isSaving}
  216. >
  217. {isSaving ? t('common.saving') : t('queue.bulkEdit.applyChanges')}
  218. </Button>
  219. </div>
  220. </div>
  221. </div>
  222. );
  223. }
  224. // Tri-state toggle for bulk edit (unchanged / on / off)
  225. function TriStateToggle({
  226. label,
  227. value,
  228. onChange,
  229. disabled,
  230. t,
  231. }: {
  232. label: string;
  233. value: boolean | 'unchanged';
  234. onChange: (val: boolean | 'unchanged') => void;
  235. disabled?: boolean;
  236. t: (key: string) => string;
  237. }) {
  238. return (
  239. <div className={`flex items-center justify-between py-1 ${disabled ? 'opacity-50' : ''}`}>
  240. <span className="text-sm text-bambu-gray">{label}</span>
  241. <div className="flex items-center gap-1 bg-bambu-dark rounded-lg p-0.5">
  242. <button
  243. onClick={() => onChange('unchanged')}
  244. disabled={disabled}
  245. className={`px-2 py-1 text-xs rounded ${value === 'unchanged' ? 'bg-bambu-dark-tertiary text-white' : 'text-bambu-gray hover:text-white'} disabled:cursor-not-allowed`}
  246. >
  247. </button>
  248. <button
  249. onClick={() => onChange(false)}
  250. disabled={disabled}
  251. className={`px-2 py-1 text-xs rounded ${value === false ? 'bg-red-500/20 text-red-400' : 'text-bambu-gray hover:text-white'} disabled:cursor-not-allowed`}
  252. >
  253. {t('common.off')}
  254. </button>
  255. <button
  256. onClick={() => onChange(true)}
  257. disabled={disabled}
  258. className={`px-2 py-1 text-xs rounded ${value === true ? 'bg-bambu-green/20 text-bambu-green' : 'text-bambu-gray hover:text-white'} disabled:cursor-not-allowed`}
  259. >
  260. {t('common.on')}
  261. </button>
  262. </div>
  263. </div>
  264. );
  265. }
  266. // Sortable queue item for drag and drop
  267. function SortableQueueItem({
  268. item,
  269. position,
  270. onEdit,
  271. onCancel,
  272. onRemove,
  273. onStop,
  274. onRequeue,
  275. onStart,
  276. timeFormat = 'system',
  277. isSelected = false,
  278. onToggleSelect,
  279. hasPermission,
  280. canModify,
  281. printerState,
  282. t,
  283. }: {
  284. item: PrintQueueItem;
  285. position?: number;
  286. onEdit: () => void;
  287. onCancel: () => void;
  288. onRemove: () => void;
  289. onStop: () => void;
  290. onRequeue: () => void;
  291. onStart: () => void;
  292. timeFormat?: TimeFormat;
  293. isSelected?: boolean;
  294. onToggleSelect?: () => void;
  295. hasPermission: (permission: Permission) => boolean;
  296. canModify: (resource: 'queue' | 'archives' | 'library', action: 'update' | 'delete' | 'reprint', createdById: number | null | undefined) => boolean;
  297. printerState?: string | null;
  298. t: (key: string, options?: Record<string, unknown>) => string;
  299. }) {
  300. // Fetch printer status every 30 seconds while printing to monitor progress
  301. const { data: status } = useQuery({
  302. queryKey: ['printerStatus', item.printer_id],
  303. queryFn: () => api.getPrinterStatus(item.printer_id!),
  304. refetchInterval: 30000,
  305. enabled: item.printer_id != null && printerState === 'printing',
  306. });
  307. // Determine if we're printing a library file
  308. const isLibraryFile = !!item.library_file_id && !item.archive_id;
  309. // Fetch archive plate details. Skip when the linked archive has been
  310. // soft-deleted (#1348 follow-up): its 3MF is gone from disk so the
  311. // /plates endpoint just 404-storms the queue page.
  312. const { data: archivePlatesData } = useQuery({
  313. queryKey: ['archive-plates', item.archive_id],
  314. queryFn: () => api.getArchivePlates(item.archive_id!),
  315. enabled: !!item.archive_id && !isLibraryFile && !item.archive_deleted,
  316. });
  317. // Fetch library file plate details
  318. const { data: libraryPlatesData } = useQuery({
  319. queryKey: ['library-file-plates', item.library_file_id],
  320. queryFn: () => api.getLibraryFilePlates(item.library_file_id!),
  321. enabled: isLibraryFile && !!item.library_file_id,
  322. });
  323. // Combine plates data from either source
  324. const platesData = isLibraryFile ? libraryPlatesData : archivePlatesData;
  325. const plates = platesData?.plates ?? [];
  326. const canReorder = hasPermission('queue:reorder');
  327. const {
  328. attributes,
  329. listeners,
  330. setNodeRef,
  331. transform,
  332. transition,
  333. isDragging,
  334. } = useSortable({ id: item.id, disabled: item.status !== 'pending' || !canReorder });
  335. const style = {
  336. transform: CSS.Transform.toString(transform),
  337. transition,
  338. };
  339. const isPrinting = item.status === 'printing';
  340. const isPending = item.status === 'pending';
  341. const isHistory = ['completed', 'failed', 'skipped', 'cancelled'].includes(item.status);
  342. const isMobileSelectable = isPending && onToggleSelect;
  343. return (
  344. <div
  345. ref={setNodeRef}
  346. style={style}
  347. className={`
  348. group relative bg-bambu-dark-secondary rounded-xl border transition-all duration-200
  349. border-l-[3px] ${
  350. isPrinting ? 'border-l-blue-500' :
  351. isPending ? 'border-l-yellow-500' :
  352. item.status === 'completed' ? 'border-l-emerald-500' :
  353. item.status === 'failed' ? 'border-l-red-500' :
  354. 'border-l-gray-500'
  355. }
  356. ${isDragging ? 'opacity-50 scale-[1.02] shadow-xl z-50' : ''}
  357. ${isPrinting ? 'border-blue-500/30 bg-gradient-to-r from-blue-500/5 to-transparent' : ''}
  358. ${isSelected && isMobileSelectable ? 'sm:border-bambu-dark-tertiary border-bambu-green/40' : ''}
  359. ${!isSelected && !isPrinting ? 'border-bambu-dark-tertiary hover:border-bambu-dark-tertiary/80' : ''}
  360. ${isMobileSelectable ? 'sm:cursor-default' : ''}
  361. `}
  362. onClick={isMobileSelectable ? () => {
  363. if (window.innerWidth < 640) onToggleSelect();
  364. } : undefined}
  365. >
  366. {/* Mobile selected left accent bar */}
  367. {isMobileSelectable && isSelected && (
  368. <div className="sm:hidden absolute left-0 top-3 bottom-3 w-1 rounded-full bg-bambu-green" />
  369. )}
  370. <div className="flex items-start sm:items-center gap-2 sm:gap-4 p-3 sm:p-4">
  371. {/* Mobile selection indicator — left accent bar only, no tick */}
  372. {/* Selection checkbox for pending items - hidden on mobile, tap card instead */}
  373. {isPending && onToggleSelect && (
  374. <button
  375. onClick={(e) => {
  376. e.stopPropagation();
  377. onToggleSelect();
  378. }}
  379. className={`hidden sm:flex items-center justify-center w-6 h-6 rounded border transition-colors shrink-0 ${
  380. isSelected
  381. ? 'bg-bambu-green border-bambu-green text-white'
  382. : 'border-white/30 bg-black/30 hover:border-bambu-green/50'
  383. }`}
  384. >
  385. {isSelected && <Check className="w-4 h-4" />}
  386. </button>
  387. )}
  388. {/* Drag handle or position number - hidden on mobile */}
  389. {isPending ? (
  390. <div
  391. {...attributes}
  392. {...listeners}
  393. className="hidden sm:flex items-center justify-center w-8 h-8 rounded-lg bg-bambu-dark cursor-grab active:cursor-grabbing hover:bg-bambu-dark-tertiary transition-colors touch-manipulation shrink-0"
  394. >
  395. <GripVertical className="w-4 h-4 text-bambu-gray" />
  396. </div>
  397. ) : position !== undefined ? (
  398. <div className="hidden sm:flex items-center justify-center w-8 h-8 rounded-lg bg-bambu-dark text-bambu-gray text-sm font-medium shrink-0">
  399. #{position}
  400. </div>
  401. ) : (
  402. <div className="hidden sm:block w-8 shrink-0" />
  403. )}
  404. {/* Thumbnail - use plate-specific thumbnail if plate_id is set */}
  405. <div className="w-10 h-10 sm:w-14 sm:h-14 flex-shrink-0 bg-bambu-dark rounded-lg overflow-hidden">
  406. {item.archive_thumbnail ? (
  407. <img
  408. src={
  409. item.plate_id != null
  410. ? api.getArchivePlateThumbnail(item.archive_id!, item.plate_id)
  411. : api.getArchiveThumbnail(item.archive_id!)
  412. }
  413. alt=""
  414. className="w-full h-full object-cover"
  415. />
  416. ) : item.library_file_thumbnail ? (
  417. <img
  418. src={
  419. item.plate_id != null
  420. ? api.getLibraryFilePlateThumbnail(item.library_file_id!, item.plate_id)
  421. : api.getLibraryFileThumbnailUrl(item.library_file_id!)
  422. }
  423. alt=""
  424. className="w-full h-full object-cover"
  425. />
  426. ) : (
  427. <div className="w-full h-full flex items-center justify-center text-bambu-gray">
  428. <Layers className="w-5 h-5 sm:w-6 sm:h-6" />
  429. </div>
  430. )}
  431. </div>
  432. {/* Info */}
  433. <div className="flex-1 min-w-0">
  434. <div className="flex items-center gap-2 mb-1">
  435. <p className="text-sm sm:text-base text-white font-medium truncate">
  436. {item.archive_name || item.library_file_name || `File #${item.archive_id || item.library_file_id}`}
  437. {(platesData?.is_multi_plate ?? false) && item.plate_id !== undefined && item.plate_id !== null && ` • ${plates.find(plate => plate.index === item.plate_id)?.name || t('queue.plateNumber', { index: item.plate_id })}`}
  438. </p>
  439. {item.archive_id ? (
  440. <Link
  441. to={`/archives?highlight=${item.archive_id}`}
  442. className="text-bambu-gray hover:text-bambu-green transition-colors flex-shrink-0"
  443. title={t('queue.viewArchive')}
  444. >
  445. <ExternalLink className="w-3.5 h-3.5" />
  446. </Link>
  447. ) : item.library_file_id ? (
  448. <Link
  449. to={`/library?highlight=${item.library_file_id}`}
  450. className="text-bambu-gray hover:text-bambu-green transition-colors flex-shrink-0"
  451. title={t('queue.viewInFileManager')}
  452. >
  453. <ExternalLink className="w-3.5 h-3.5" />
  454. </Link>
  455. ) : null}
  456. {item.batch_name && (
  457. <span className="flex-shrink-0 px-1.5 py-0.5 text-[10px] sm:text-xs bg-purple-500/20 text-purple-300 rounded border border-purple-500/30">
  458. {item.batch_name}
  459. </span>
  460. )}
  461. </div>
  462. <div className="flex flex-wrap items-center gap-x-3 gap-y-1 text-xs sm:text-sm text-bambu-gray">
  463. <span className={`flex items-center gap-1 sm:gap-1.5 ${item.printer_id === null && !item.target_model ? 'text-orange-400' : ''} ${item.target_model && !item.printer_id ? 'text-blue-400' : ''}`}>
  464. <Printer className="w-3 h-3 sm:w-3.5 sm:h-3.5" />
  465. <span className="truncate max-w-[120px] sm:max-w-none">
  466. {item.target_model && !item.printer_id
  467. ? `${t('queue.filter.any')} ${item.target_model}${item.target_location ? ` @ ${item.target_location}` : ''}${item.required_filament_types?.length ? ` (${item.required_filament_types.join(', ')})` : ''}`
  468. : item.printer_id === null
  469. ? t('queue.filter.unassigned')
  470. : (item.printer_name || `${t('common.printer')} #${item.printer_id}`)}
  471. </span>
  472. </span>
  473. {item.print_time_seconds && (
  474. <span className="flex items-center gap-1 sm:gap-1.5">
  475. <Timer className="w-3 h-3 sm:w-3.5 sm:h-3.5" />
  476. {formatDuration(item.print_time_seconds)}
  477. </span>
  478. )}
  479. {item.filament_used_grams && (
  480. <span className="flex items-center gap-1 sm:gap-1.5">
  481. <Weight className="w-3 h-3 sm:w-3.5 sm:h-3.5" />
  482. {formatWeight(item.filament_used_grams)}
  483. </span>
  484. )}
  485. {item.created_by_username && (
  486. <span className="hidden sm:flex items-center gap-1.5" title={t('queue.addedBy', { name: item.created_by_username })}>
  487. <User className="w-3.5 h-3.5" />
  488. {item.created_by_username}
  489. </span>
  490. )}
  491. {isPending && !item.manual_start && (
  492. <span className="flex items-center gap-1.5">
  493. <Clock className="w-3.5 h-3.5" />
  494. {item.scheduled_time
  495. ? ((parseUTCDate(item.scheduled_time)?.getTime() ?? 0) - Date.now() < -60000
  496. ? t?.('queue.time.overdue') ?? 'Overdue'
  497. : formatRelativeTime(item.scheduled_time, timeFormat, t))
  498. : t?.('queue.time.asap') ?? 'ASAP'}
  499. </span>
  500. )}
  501. </div>
  502. {/* Options badges */}
  503. <div className="flex flex-wrap items-center gap-1.5 sm:gap-2 mt-1.5 sm:mt-2">
  504. {item.manual_start && (
  505. <span className="text-[10px] sm:text-xs px-1.5 sm:px-2 py-0.5 bg-purple-500/10 text-purple-400 rounded-full border border-purple-500/20 flex items-center gap-1">
  506. <Hand className="w-2.5 h-2.5 sm:w-3 sm:h-3" />
  507. {t('queue.badges.staged')}
  508. </span>
  509. )}
  510. {item.require_previous_success && (
  511. <span className="text-[10px] sm:text-xs px-1.5 sm:px-2 py-0.5 bg-orange-500/10 text-orange-400 rounded-full border border-orange-500/20">
  512. {t('queue.badges.requiresPrevious')}
  513. </span>
  514. )}
  515. {item.auto_off_after && (
  516. <span className="text-[10px] sm:text-xs px-1.5 sm:px-2 py-0.5 bg-blue-500/10 text-blue-400 rounded-full border border-blue-500/20 flex items-center gap-1">
  517. <Power className="w-2.5 h-2.5 sm:w-3 sm:h-3" />
  518. {t('queue.badges.autoPowerOff')}
  519. </span>
  520. )}
  521. {item.gcode_injection && (
  522. <span className="text-[10px] sm:text-xs px-1.5 sm:px-2 py-0.5 bg-emerald-500/10 text-emerald-400 rounded-full border border-emerald-500/20 flex items-center gap-1">
  523. <Code className="w-2.5 h-2.5 sm:w-3 sm:h-3" />
  524. {t('queue.badges.gcodeInjection')}
  525. </span>
  526. )}
  527. </div>
  528. {/* Progress bar for printing items - TODO: integrate with WebSocket */}
  529. {isPrinting && status && (() => {
  530. // Gate progress/remaining/layer on printer actually running this print.
  531. // Between dispatch and RUNNING transition (H2D/P1 MQTT lag), status.progress
  532. // is stale from the previous print — showing 100% then snapping back to 0%
  533. // once the new print starts. Only trust these fields when state is active.
  534. const isActive = status.state === 'RUNNING' || status.state === 'PAUSE';
  535. const progress = isActive ? (status.progress || 0) : 0;
  536. const remaining = isActive ? status.remaining_time : null;
  537. const layerNum = isActive ? status.layer_num : null;
  538. const totalLayers = isActive ? status.total_layers : null;
  539. return (
  540. <div className="mt-2 sm:mt-3">
  541. <div className="flex items-center justify-between text-xs sm:text-sm">
  542. <div className="flex-1 bg-bambu-dark-tertiary rounded-full h-1.5 sm:h-2 mr-3">
  543. <div
  544. className="bg-bambu-green h-1.5 sm:h-2 rounded-full transition-all"
  545. style={{ width: `${progress}%` }}
  546. />
  547. </div>
  548. <span className="text-white">{Math.round(progress)}%</span>
  549. </div>
  550. <div className="flex flex-wrap items-center gap-2 sm:gap-3 mt-1.5 sm:mt-2 text-[10px] sm:text-xs text-bambu-gray">
  551. {remaining != null && remaining > 0 && (
  552. <>
  553. <span className="flex items-center gap-1">
  554. <Clock className="w-3 h-3" />
  555. {formatDuration(remaining * 60)}
  556. </span>
  557. <span className="text-bambu-green font-medium" title={t('printers.estimatedCompletion')}>
  558. ETA {formatETA(remaining, timeFormat, t)}
  559. </span>
  560. </>
  561. )}
  562. {layerNum != null && totalLayers != null && totalLayers > 0 && (
  563. <span className="flex items-center gap-1">
  564. <Layers className="w-3 h-3" />
  565. {layerNum}/{totalLayers}
  566. </span>
  567. )}
  568. </div>
  569. </div>
  570. );
  571. })()}
  572. {/* Waiting reason for model-based assignments */}
  573. {item.waiting_reason && item.status === 'pending' && (
  574. <p className="text-[10px] sm:text-xs text-purple-400 mt-1.5 sm:mt-2 flex items-start gap-1">
  575. <AlertCircle className="w-3 h-3 mt-0.5 flex-shrink-0" />
  576. <span>{item.waiting_reason}</span>
  577. </p>
  578. )}
  579. {/* Error message */}
  580. {item.error_message && (
  581. <p className="text-[10px] sm:text-xs text-red-400 mt-1.5 sm:mt-2 flex items-center gap-1">
  582. <AlertCircle className="w-3 h-3" />
  583. {item.error_message}
  584. </p>
  585. )}
  586. </div>
  587. {/* Status badge + Actions */}
  588. <div className="flex flex-col sm:flex-row items-end sm:items-center gap-2 sm:gap-1 shrink-0" onClick={(e) => e.stopPropagation()}>
  589. <StatusBadge status={item.status} waitingReason={item.waiting_reason} printerState={printerState} t={t} />
  590. <div className="flex items-center gap-0.5 sm:gap-1">
  591. {isPrinting && (
  592. <Button
  593. variant="ghost"
  594. size="sm"
  595. onClick={onStop}
  596. disabled={!hasPermission('printers:control')}
  597. title={!hasPermission('printers:control') ? t('queue.permissions.noStopPrint') : t('queue.actions.stopPrint')}
  598. className="text-red-400 hover:text-red-300 hover:bg-red-500/10 p-1.5 sm:p-2"
  599. >
  600. <StopCircle className="w-4 h-4" />
  601. </Button>
  602. )}
  603. {isPending && (
  604. <>
  605. {item.manual_start && (
  606. <Button
  607. variant="ghost"
  608. size="sm"
  609. onClick={onStart}
  610. disabled={!hasPermission('printers:control')}
  611. title={!hasPermission('printers:control') ? t('queue.permissions.noStartPrint') : t('queue.actions.startPrint')}
  612. className="text-bambu-green hover:text-bambu-green-light hover:bg-bambu-green/10 p-1.5 sm:p-2"
  613. >
  614. <Play className="w-4 h-4" />
  615. </Button>
  616. )}
  617. <Button
  618. variant="ghost"
  619. size="sm"
  620. onClick={onEdit}
  621. disabled={!canModify('queue', 'update', item.created_by_id)}
  622. title={!canModify('queue', 'update', item.created_by_id) ? t('queue.permissions.noEdit') : t('common.edit')}
  623. className="p-1.5 sm:p-2"
  624. >
  625. <Pencil className="w-4 h-4" />
  626. </Button>
  627. <Button
  628. variant="ghost"
  629. size="sm"
  630. onClick={onCancel}
  631. disabled={!canModify('queue', 'delete', item.created_by_id)}
  632. title={!canModify('queue', 'delete', item.created_by_id) ? t('queue.permissions.noCancel') : t('common.cancel')}
  633. className="text-red-400 hover:text-red-300 hover:bg-red-500/10 p-1.5 sm:p-2"
  634. >
  635. <X className="w-4 h-4" />
  636. </Button>
  637. </>
  638. )}
  639. {isHistory && (
  640. <>
  641. <Button
  642. variant="ghost"
  643. size="sm"
  644. onClick={onRequeue}
  645. disabled={!hasPermission('queue:create')}
  646. title={!hasPermission('queue:create') ? t('queue.permissions.noRequeue') : t('queue.actions.requeue')}
  647. className="text-bambu-green hover:text-bambu-green/80 hover:bg-bambu-green/10 p-1.5 sm:p-2"
  648. >
  649. <RefreshCw className="w-4 h-4" />
  650. </Button>
  651. <Button
  652. variant="ghost"
  653. size="sm"
  654. onClick={onRemove}
  655. disabled={!canModify('queue', 'delete', item.created_by_id)}
  656. title={!canModify('queue', 'delete', item.created_by_id) ? t('queue.permissions.noRemove') : t('common.remove')}
  657. className="p-1.5 sm:p-2"
  658. >
  659. <Trash2 className="w-4 h-4" />
  660. </Button>
  661. </>
  662. )}
  663. </div>
  664. </div>
  665. </div>
  666. </div>
  667. );
  668. }
  669. export function QueuePage() {
  670. const { t } = useTranslation();
  671. const queryClient = useQueryClient();
  672. const { showToast } = useToast();
  673. const { hasPermission, hasAnyPermission, canModify } = useAuth();
  674. const [filterPrinter, setFilterPrinter] = useState<number | null>(null);
  675. const [filterStatus, setFilterStatus] = useState<string>('');
  676. const [filterLocation, setFilterLocation] = useState<string>('');
  677. const [showClearHistoryConfirm, setShowClearHistoryConfirm] = useState(false);
  678. const [editItem, setEditItem] = useState<PrintQueueItem | null>(null);
  679. const [requeueItem, setRequeueItem] = useState<PrintQueueItem | null>(null);
  680. const [confirmAction, setConfirmAction] = useState<{
  681. type: 'cancel' | 'remove' | 'stop';
  682. item: PrintQueueItem;
  683. } | null>(null);
  684. const [selectedItems, setSelectedItems] = useState<number[]>([]);
  685. const [showBulkEditModal, setShowBulkEditModal] = useState(false);
  686. const [historySortBy, setHistorySortBy] = useState<'date' | 'name' | 'printer'>(() => {
  687. const saved = localStorage.getItem('queue.historySortBy');
  688. return (saved as 'date' | 'name' | 'printer') || 'date';
  689. });
  690. const [historySortAsc, setHistorySortAsc] = useState(() => {
  691. const saved = localStorage.getItem('queue.historySortAsc');
  692. return saved !== null ? saved === 'true' : false;
  693. });
  694. const [pendingSortBy, setPendingSortBy] = useState<'position' | 'name' | 'printer' | 'time'>(() => {
  695. const saved = localStorage.getItem('queue.pendingSortBy');
  696. return (saved as 'position' | 'name' | 'printer' | 'time') || 'position';
  697. });
  698. const [pendingSortAsc, setPendingSortAsc] = useState(() => {
  699. const saved = localStorage.getItem('queue.pendingSortAsc');
  700. return saved !== null ? saved === 'true' : true;
  701. });
  702. const [historyCollapsed, setHistoryCollapsed] = useState(() => {
  703. return localStorage.getItem('queue.historyCollapsed') !== 'false';
  704. });
  705. const [viewMode, setViewMode] = useState<'list' | 'timeline'>(() => {
  706. return (localStorage.getItem('queue.viewMode') as 'list' | 'timeline') || 'list';
  707. });
  708. // Persist sort settings to localStorage
  709. useEffect(() => {
  710. localStorage.setItem('queue.historySortBy', historySortBy);
  711. }, [historySortBy]);
  712. useEffect(() => {
  713. localStorage.setItem('queue.historySortAsc', String(historySortAsc));
  714. }, [historySortAsc]);
  715. useEffect(() => {
  716. localStorage.setItem('queue.pendingSortBy', pendingSortBy);
  717. }, [pendingSortBy]);
  718. useEffect(() => {
  719. localStorage.setItem('queue.pendingSortAsc', String(pendingSortAsc));
  720. }, [pendingSortAsc]);
  721. useEffect(() => {
  722. localStorage.setItem('queue.historyCollapsed', String(historyCollapsed));
  723. }, [historyCollapsed]);
  724. useEffect(() => {
  725. localStorage.setItem('queue.viewMode', viewMode);
  726. }, [viewMode]);
  727. const sensors = useSensors(
  728. useSensor(PointerSensor, { activationConstraint: { distance: 8 } }),
  729. useSensor(KeyboardSensor, { coordinateGetter: sortableKeyboardCoordinates })
  730. );
  731. const { data: settings } = useQuery({
  732. queryKey: ['settings'],
  733. queryFn: api.getSettings,
  734. });
  735. const timeFormat: TimeFormat = settings?.time_format || 'system';
  736. const { data: queue, isLoading } = useQuery({
  737. queryKey: ['queue', filterPrinter, filterStatus],
  738. queryFn: () => api.getQueue(filterPrinter || undefined, filterStatus || undefined),
  739. refetchInterval: 5000,
  740. });
  741. const { data: printers } = useQuery({
  742. queryKey: ['printers'],
  743. queryFn: () => api.getPrinters(),
  744. });
  745. const sjfMutation = useMutation({
  746. mutationFn: (enabled: boolean) => api.updateSettings({ queue_shortest_first: enabled }),
  747. onSuccess: () => {
  748. queryClient.invalidateQueries({ queryKey: ['settings'] });
  749. },
  750. });
  751. const cancelMutation = useMutation({
  752. mutationFn: (id: number) => api.cancelQueueItem(id),
  753. onSuccess: () => {
  754. queryClient.invalidateQueries({ queryKey: ['queue'] });
  755. showToast(t('queue.toast.cancelled'));
  756. },
  757. onError: () => showToast(t('queue.toast.cancelFailed'), 'error'),
  758. });
  759. const removeMutation = useMutation({
  760. mutationFn: (id: number) => api.removeFromQueue(id),
  761. onSuccess: () => {
  762. queryClient.invalidateQueries({ queryKey: ['queue'] });
  763. showToast(t('queue.toast.removed'));
  764. },
  765. onError: () => showToast(t('queue.toast.removeFailed'), 'error'),
  766. });
  767. const stopMutation = useMutation({
  768. mutationFn: (id: number) => api.stopQueueItem(id),
  769. onSuccess: () => {
  770. queryClient.invalidateQueries({ queryKey: ['queue'] });
  771. showToast(t('queue.toast.stopped'));
  772. },
  773. onError: () => showToast(t('queue.toast.stopFailed'), 'error'),
  774. });
  775. const startMutation = useMutation({
  776. mutationFn: (id: number) => api.startQueueItem(id),
  777. onSuccess: () => {
  778. queryClient.invalidateQueries({ queryKey: ['queue'] });
  779. showToast(t('queue.toast.released'));
  780. },
  781. onError: () => showToast(t('queue.toast.startFailed'), 'error'),
  782. });
  783. const reorderMutation = useMutation({
  784. mutationFn: (items: { id: number; position: number }[]) => api.reorderQueue(items),
  785. onSuccess: () => {
  786. queryClient.invalidateQueries({ queryKey: ['queue'] });
  787. },
  788. onError: () => showToast(t('queue.toast.reorderFailed'), 'error'),
  789. });
  790. const clearHistoryMutation = useMutation({
  791. mutationFn: async () => {
  792. const historyItems = queue?.filter(i =>
  793. ['completed', 'failed', 'skipped', 'cancelled'].includes(i.status)
  794. ) || [];
  795. for (const item of historyItems) {
  796. await api.removeFromQueue(item.id);
  797. }
  798. return historyItems.length;
  799. },
  800. onSuccess: (count) => {
  801. queryClient.invalidateQueries({ queryKey: ['queue'] });
  802. showToast(t('queue.toast.historyCleared', { count }));
  803. },
  804. onError: () => showToast(t('queue.toast.clearHistoryFailed'), 'error'),
  805. });
  806. const bulkUpdateMutation = useMutation({
  807. mutationFn: (data: PrintQueueBulkUpdate) => api.bulkUpdateQueue(data),
  808. onSuccess: (result) => {
  809. queryClient.invalidateQueries({ queryKey: ['queue'] });
  810. setSelectedItems([]);
  811. setShowBulkEditModal(false);
  812. showToast(result.message);
  813. },
  814. onError: () => showToast(t('queue.toast.updateFailed'), 'error'),
  815. });
  816. const bulkCancelMutation = useMutation({
  817. mutationFn: async (ids: number[]) => {
  818. for (const id of ids) {
  819. await api.cancelQueueItem(id);
  820. }
  821. return ids.length;
  822. },
  823. onSuccess: (count) => {
  824. queryClient.invalidateQueries({ queryKey: ['queue'] });
  825. setSelectedItems([]);
  826. showToast(t('queue.toast.bulkCancelled', { count }));
  827. },
  828. onError: () => showToast(t('queue.toast.bulkCancelFailed'), 'error'),
  829. });
  830. const handleToggleSelect = (id: number) => {
  831. setSelectedItems(prev =>
  832. prev.includes(id) ? prev.filter(x => x !== id) : [...prev, id]
  833. );
  834. };
  835. // Get unique locations from printers for the filter dropdown
  836. const uniqueLocations = useMemo(() => {
  837. const locations = new Set<string>();
  838. printers?.forEach(p => {
  839. if (p.location) locations.add(p.location);
  840. });
  841. // Also include locations from queue items (for model-based assignments)
  842. queue?.forEach(item => {
  843. if (item.target_location) locations.add(item.target_location);
  844. });
  845. return Array.from(locations).sort();
  846. }, [printers, queue]);
  847. // Helper to check if a queue item matches the location filter
  848. const matchesLocationFilter = useCallback((item: PrintQueueItem): boolean => {
  849. if (!filterLocation) return true;
  850. // For model-based assignments, check target_location
  851. if (item.target_location) return item.target_location === filterLocation;
  852. // For printer-based assignments, check the printer's location
  853. if (item.printer_id) {
  854. const printer = printers?.find(p => p.id === item.printer_id);
  855. return printer?.location === filterLocation;
  856. }
  857. return false;
  858. }, [filterLocation, printers]);
  859. const pendingItems = useMemo(() => {
  860. let items = queue?.filter(i => i.status === 'pending') || [];
  861. // Apply location filter
  862. if (filterLocation) {
  863. items = items.filter(matchesLocationFilter);
  864. }
  865. // Helper to get scheduled time as timestamp (ASAP/placeholder = 0 for earliest)
  866. const getScheduledTime = (item: PrintQueueItem): number => {
  867. if (!item.scheduled_time) return 0;
  868. const time = parseUTCDate(item.scheduled_time)?.getTime() ?? 0;
  869. // Placeholder dates (> 6 months out) are treated as ASAP
  870. const sixMonthsFromNow = Date.now() + (180 * 24 * 60 * 60 * 1000);
  871. return time > sixMonthsFromNow ? 0 : time;
  872. };
  873. // When SJF is enabled, override sort to match scheduler order
  874. if (settings?.queue_shortest_first) {
  875. return [...items].sort((a, b) => {
  876. // Group by printer first (nulls = model-based, grouped by target_model)
  877. const aPrinter = a.printer_id ?? -(a.target_model?.charCodeAt(0) ?? 0);
  878. const bPrinter = b.printer_id ?? -(b.target_model?.charCodeAt(0) ?? 0);
  879. if (aPrinter !== bPrinter) return aPrinter - bPrinter;
  880. // Within same printer/model: jumped items first (starvation guard)
  881. const aJumped = a.been_jumped ? 1 : 0;
  882. const bJumped = b.been_jumped ? 1 : 0;
  883. if (aJumped !== bJumped) return bJumped - aJumped;
  884. // Shortest print time next (nulls last)
  885. const aTime = a.print_time_seconds ?? Infinity;
  886. const bTime = b.print_time_seconds ?? Infinity;
  887. if (aTime !== bTime) return aTime - bTime;
  888. // Position as tiebreaker
  889. return a.position - b.position;
  890. });
  891. }
  892. return [...items].sort((a, b) => {
  893. let cmp: number;
  894. if (pendingSortBy === 'name') {
  895. const aName = a.archive_name || a.library_file_name || '';
  896. const bName = b.archive_name || b.library_file_name || '';
  897. cmp = aName.localeCompare(bName);
  898. } else if (pendingSortBy === 'printer') {
  899. cmp = (a.printer_name || '').localeCompare(b.printer_name || '');
  900. } else if (pendingSortBy === 'time') {
  901. // Sort by scheduled start time (when print will begin)
  902. cmp = getScheduledTime(a) - getScheduledTime(b);
  903. } else {
  904. cmp = a.position - b.position;
  905. }
  906. return pendingSortAsc ? cmp : -cmp;
  907. });
  908. }, [queue, pendingSortBy, pendingSortAsc, matchesLocationFilter, filterLocation, settings?.queue_shortest_first]);
  909. const handleSelectAll = () => {
  910. const allPendingIds = pendingItems.map(i => i.id);
  911. if (selectedItems.length === allPendingIds.length) {
  912. setSelectedItems([]);
  913. } else {
  914. setSelectedItems(allPendingIds);
  915. }
  916. };
  917. const activeItems = useMemo(() => {
  918. let items = queue?.filter(i => i.status === 'printing') || [];
  919. if (filterLocation) {
  920. items = items.filter(matchesLocationFilter);
  921. }
  922. return items;
  923. }, [queue, filterLocation, matchesLocationFilter]);
  924. // Get unique printer IDs from active items to fetch their statuses
  925. const activePrinterIds = useMemo(() => {
  926. const ids = new Set<number>();
  927. activeItems.forEach(item => {
  928. if (item.printer_id) ids.add(item.printer_id);
  929. });
  930. return Array.from(ids);
  931. }, [activeItems]);
  932. // Fetch printer statuses for printers with active jobs
  933. const printerStatusQueries = useQueries({
  934. queries: activePrinterIds.map(printerId => ({
  935. queryKey: ['printerStatus', printerId],
  936. queryFn: () => api.getPrinterStatus(printerId),
  937. refetchInterval: 5000,
  938. })),
  939. });
  940. // Build a map of printer_id -> state for quick lookup
  941. const printerStateMap = useMemo(() => {
  942. const map: Record<number, string | null> = {};
  943. activePrinterIds.forEach((printerId, index) => {
  944. const result = printerStatusQueries[index];
  945. if (result?.data?.state) {
  946. map[printerId] = result.data.state;
  947. }
  948. });
  949. return map;
  950. }, [activePrinterIds, printerStatusQueries]);
  951. // Build a map of printer_id -> full status for timeline view
  952. const printerStatusMap = useMemo(() => {
  953. const map: Record<number, { progress?: number; remaining_time?: number; state?: string }> = {};
  954. activePrinterIds.forEach((printerId, index) => {
  955. const result = printerStatusQueries[index];
  956. if (result?.data) {
  957. map[printerId] = {
  958. progress: result.data.progress ?? undefined,
  959. remaining_time: result.data.remaining_time ?? undefined,
  960. state: result.data.state ?? undefined,
  961. };
  962. }
  963. });
  964. return map;
  965. }, [activePrinterIds, printerStatusQueries]);
  966. const historyItems = useMemo(() => {
  967. let items = queue?.filter(i => ['completed', 'failed', 'skipped', 'cancelled'].includes(i.status)) || [];
  968. if (filterLocation) {
  969. items = items.filter(matchesLocationFilter);
  970. }
  971. return [...items].sort((a, b) => {
  972. let cmp: number;
  973. if (historySortBy === 'name') {
  974. const aName = a.archive_name || a.library_file_name || '';
  975. const bName = b.archive_name || b.library_file_name || '';
  976. cmp = aName.localeCompare(bName);
  977. } else if (historySortBy === 'printer') {
  978. cmp = (a.printer_name || '').localeCompare(b.printer_name || '');
  979. } else {
  980. // Default: by date - most recent first (desc) is the natural order
  981. cmp = (parseUTCDate(b.completed_at || b.created_at)?.getTime() ?? 0) - (parseUTCDate(a.completed_at || a.created_at)?.getTime() ?? 0);
  982. }
  983. return historySortAsc ? -cmp : cmp;
  984. });
  985. }, [queue, historySortBy, historySortAsc, matchesLocationFilter, filterLocation]);
  986. // Calculate total queue time
  987. const totalQueueTime = useMemo(() => {
  988. return pendingItems.reduce((acc, item) => acc + (item.print_time_seconds || 0), 0);
  989. }, [pendingItems]);
  990. // Calculate total material weight
  991. const totalWeight = useMemo(() => {
  992. return pendingItems.reduce((acc, item) => acc + (item.filament_used_grams || 0), 0);
  993. }, [pendingItems]);
  994. const handleDragEnd = (event: DragEndEvent) => {
  995. const { active, over } = event;
  996. if (!over || active.id === over.id) return;
  997. const oldIndex = pendingItems.findIndex(i => i.id === active.id);
  998. const newIndex = pendingItems.findIndex(i => i.id === over.id);
  999. if (oldIndex !== -1 && newIndex !== -1) {
  1000. const reordered = arrayMove(pendingItems, oldIndex, newIndex);
  1001. const updates = reordered.map((item, index) => ({
  1002. id: item.id,
  1003. position: index + 1,
  1004. }));
  1005. reorderMutation.mutate(updates);
  1006. }
  1007. };
  1008. return (
  1009. <div className="p-4 md:p-8">
  1010. {/* Header */}
  1011. <div className="flex items-center justify-between mb-8">
  1012. <div>
  1013. <h1 className="text-2xl font-bold text-white flex items-center gap-3">
  1014. <ListOrdered className="w-7 h-7 text-bambu-green" />
  1015. {t('queue.title')}
  1016. </h1>
  1017. <p className="text-bambu-gray mt-1">{t('queue.subtitle')}</p>
  1018. </div>
  1019. </div>
  1020. {/* Summary Stats */}
  1021. <QueueStatsBar
  1022. activeCount={activeItems.length}
  1023. pendingCount={pendingItems.length}
  1024. totalTime={totalQueueTime}
  1025. totalWeight={totalWeight}
  1026. historyCount={historyItems.length}
  1027. t={t}
  1028. />
  1029. {/* Filters */}
  1030. <div className="flex flex-wrap items-center gap-2 sm:gap-4 mb-6">
  1031. <select
  1032. className="px-2 sm:px-3 py-2 text-sm sm:text-base bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none min-w-0 flex-1 sm:flex-none"
  1033. value={filterPrinter === -1 ? 'unassigned' : (filterPrinter || '')}
  1034. onChange={(e) => {
  1035. const val = e.target.value;
  1036. if (val === 'unassigned') setFilterPrinter(-1);
  1037. else if (val === '') setFilterPrinter(null);
  1038. else setFilterPrinter(Number(val));
  1039. }}
  1040. >
  1041. <option value="">{t('queue.filter.allPrinters')}</option>
  1042. <option value="unassigned">{t('queue.filter.unassigned')}</option>
  1043. {printers?.map((p) => (
  1044. <option key={p.id} value={p.id}>{p.name}</option>
  1045. ))}
  1046. </select>
  1047. <select
  1048. className="px-2 sm:px-3 py-2 text-sm sm:text-base bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none min-w-0 flex-1 sm:flex-none"
  1049. value={filterStatus}
  1050. onChange={(e) => setFilterStatus(e.target.value)}
  1051. >
  1052. <option value="">{t('queue.filter.allStatus')}</option>
  1053. <option value="pending">{t('queue.status.pending')}</option>
  1054. <option value="printing">{t('queue.status.printing')}</option>
  1055. <option value="completed">{t('queue.status.completed')}</option>
  1056. <option value="failed">{t('queue.status.failed')}</option>
  1057. <option value="skipped">{t('queue.status.skipped')}</option>
  1058. <option value="cancelled">{t('queue.status.cancelled')}</option>
  1059. </select>
  1060. {uniqueLocations.length > 0 && (
  1061. <select
  1062. className="px-2 sm:px-3 py-2 text-sm sm:text-base bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none min-w-0 flex-1 sm:flex-none"
  1063. value={filterLocation}
  1064. onChange={(e) => setFilterLocation(e.target.value)}
  1065. >
  1066. <option value="">{t('queue.filter.allLocations')}</option>
  1067. {uniqueLocations.map((loc) => (
  1068. <option key={loc} value={loc}>{loc}</option>
  1069. ))}
  1070. </select>
  1071. )}
  1072. <div className="hidden sm:block flex-1" />
  1073. {historyItems.length > 0 && (
  1074. <Button
  1075. className="w-full sm:w-auto"
  1076. variant="secondary"
  1077. size="sm"
  1078. onClick={() => setShowClearHistoryConfirm(true)}
  1079. disabled={!hasPermission('queue:delete_all')}
  1080. title={!hasPermission('queue:delete_all') ? t('queue.permissions.noClearHistory') : undefined}
  1081. >
  1082. <Trash2 className="w-4 h-4" />
  1083. {t('queue.clearHistory')}
  1084. </Button>
  1085. )}
  1086. </div>
  1087. {/* View Mode Toggle + SJF */}
  1088. <div className="flex items-center gap-3 mb-6">
  1089. <div className="hidden sm:flex items-center border border-bambu-dark-tertiary rounded-lg overflow-hidden">
  1090. <button
  1091. className={`p-2 transition-colors ${viewMode === 'list' ? 'bg-bambu-green text-white' : 'bg-bambu-dark text-bambu-gray hover:text-white'}`}
  1092. onClick={() => setViewMode('list')}
  1093. title={t('queue.timeline.listView')}
  1094. >
  1095. <List className="w-4 h-4" />
  1096. </button>
  1097. <button
  1098. className={`p-2 transition-colors ${viewMode === 'timeline' ? 'bg-bambu-green text-white' : 'bg-bambu-dark text-bambu-gray hover:text-white'}`}
  1099. onClick={() => setViewMode('timeline')}
  1100. title={t('queue.timeline.timelineView')}
  1101. >
  1102. <GanttChart className="w-4 h-4" />
  1103. </button>
  1104. </div>
  1105. <button
  1106. onClick={() => {
  1107. const newValue = !(settings?.queue_shortest_first ?? false);
  1108. sjfMutation.mutate(newValue);
  1109. }}
  1110. className={`flex items-center gap-1 px-2 py-1.5 text-xs rounded-lg border transition-colors ${
  1111. settings?.queue_shortest_first
  1112. ? 'bg-bambu-green/20 border-bambu-green text-bambu-green'
  1113. : 'bg-bambu-dark-secondary border-bambu-dark-tertiary text-bambu-gray hover:text-white hover:border-bambu-gray'
  1114. }`}
  1115. title={t('queue.sjf.tooltip', 'Shortest Job First — scheduler prioritizes shorter prints')}
  1116. >
  1117. <Snail className="w-4 h-4" />
  1118. <span className="hidden sm:inline">{t('queue.sjf.label', 'SJF')}</span>
  1119. <span className={`w-1.5 h-1.5 rounded-full ${settings?.queue_shortest_first ? 'bg-bambu-green' : 'bg-bambu-gray'}`} />
  1120. </button>
  1121. </div>
  1122. {isLoading ? (
  1123. <div className="text-center py-12 text-bambu-gray">{t('common.loading')}</div>
  1124. ) : queue?.length === 0 ? (
  1125. <Card className="p-12 text-center border-dashed">
  1126. <Calendar className="w-16 h-16 text-bambu-gray mx-auto mb-4 opacity-50" />
  1127. <h3 className="text-xl font-medium text-white mb-2">{t('queue.empty.title')}</h3>
  1128. <p className="text-bambu-gray max-w-md mx-auto">
  1129. {t('queue.empty.description')}
  1130. </p>
  1131. </Card>
  1132. ) : viewMode === 'timeline' ? (
  1133. <QueueTimelineView
  1134. queueItems={queue || []}
  1135. printerStatuses={printerStatusMap}
  1136. onItemClick={(item) => {
  1137. if (['completed', 'failed', 'skipped', 'cancelled'].includes(item.status)) {
  1138. setRequeueItem(item);
  1139. } else if (item.status === 'pending') {
  1140. setEditItem(item);
  1141. } else if (item.status === 'printing') {
  1142. setConfirmAction({ type: 'stop', item });
  1143. }
  1144. }}
  1145. t={t}
  1146. />
  1147. ) : (
  1148. <div className="space-y-6 sm:space-y-8">
  1149. {/* Active Prints */}
  1150. {activeItems.length > 0 && (
  1151. <div>
  1152. <h2 className="text-base sm:text-lg font-semibold text-white mb-3 sm:mb-4 flex items-center gap-2">
  1153. <div className="w-2 h-2 rounded-full bg-blue-400 animate-pulse" />
  1154. {t('queue.sections.currentlyPrinting')}
  1155. </h2>
  1156. <div className="space-y-2 sm:space-y-3">
  1157. {activeItems.map((item) => (
  1158. <SortableQueueItem
  1159. key={item.id}
  1160. item={item}
  1161. onEdit={() => {}}
  1162. onCancel={() => {}}
  1163. onRemove={() => {}}
  1164. onStop={() => setConfirmAction({ type: 'stop', item })}
  1165. onRequeue={() => {}}
  1166. onStart={() => {}}
  1167. timeFormat={timeFormat}
  1168. hasPermission={hasPermission}
  1169. canModify={canModify}
  1170. printerState={item.printer_id ? printerStateMap[item.printer_id] : null}
  1171. t={t}
  1172. />
  1173. ))}
  1174. </div>
  1175. </div>
  1176. )}
  1177. {/* Pending Queue */}
  1178. {pendingItems.length > 0 && (
  1179. <div>
  1180. <div className="flex flex-wrap items-center justify-between gap-2 mb-3 sm:mb-4">
  1181. <h2 className="text-base sm:text-lg font-semibold text-white flex items-center gap-2">
  1182. <Clock className="w-4 h-4 sm:w-5 sm:h-5 text-yellow-400" />
  1183. {t('queue.sections.queued')}
  1184. <span className="text-xs sm:text-sm font-normal text-bambu-gray">
  1185. ({t('queue.itemCount', { count: pendingItems.length })})
  1186. </span>
  1187. <span className="hidden sm:inline text-xs text-bambu-gray ml-2" title={t('queue.reorderHint')}>
  1188. {t('queue.dragToReorder')}
  1189. </span>
  1190. </h2>
  1191. <div className="flex items-center gap-2">
  1192. <select
  1193. className="px-2 sm:px-3 py-1.5 text-xs sm:text-sm bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1194. value={pendingSortBy}
  1195. onChange={(e) => setPendingSortBy(e.target.value as 'position' | 'name' | 'printer' | 'time')}
  1196. >
  1197. <option value="position">{t('queue.sort.byPosition')}</option>
  1198. <option value="name">{t('queue.sort.byName')}</option>
  1199. <option value="printer">{t('queue.sort.byPrinter')}</option>
  1200. <option value="time">{t('queue.sort.bySchedule')}</option>
  1201. </select>
  1202. <Button
  1203. variant="ghost"
  1204. size="sm"
  1205. onClick={() => setPendingSortAsc(!pendingSortAsc)}
  1206. title={pendingSortAsc ? t('common.ascending') : t('common.descending')}
  1207. className="px-2"
  1208. >
  1209. {pendingSortAsc ? <ArrowUp className="w-4 h-4" /> : <ArrowDown className="w-4 h-4" />}
  1210. </Button>
  1211. </div>
  1212. </div>
  1213. {/* Bulk action toolbar */}
  1214. <div className="flex flex-wrap items-center gap-2 sm:gap-3 mb-3 sm:mb-4 p-2 sm:p-3 bg-bambu-dark rounded-lg">
  1215. <Button
  1216. variant="ghost"
  1217. size="sm"
  1218. onClick={handleSelectAll}
  1219. className="flex items-center gap-1.5 sm:gap-2 text-xs sm:text-sm"
  1220. >
  1221. {selectedItems.length === pendingItems.length && pendingItems.length > 0 ? (
  1222. <CheckSquare className="w-4 h-4 text-bambu-green" />
  1223. ) : (
  1224. <Square className="w-4 h-4" />
  1225. )}
  1226. {selectedItems.length === pendingItems.length && pendingItems.length > 0 ? t('queue.bulkEdit.deselectAll') : t('queue.bulkEdit.selectAll')}
  1227. </Button>
  1228. {selectedItems.length > 0 && (
  1229. <>
  1230. <span className="text-xs sm:text-sm text-bambu-gray">
  1231. {t('queue.bulkEdit.selected', { count: selectedItems.length })}
  1232. </span>
  1233. <div className="hidden sm:block h-4 w-px bg-bambu-dark-tertiary" />
  1234. <Button
  1235. variant="ghost"
  1236. size="sm"
  1237. onClick={() => setShowBulkEditModal(true)}
  1238. className="flex items-center gap-1.5 sm:gap-2 text-xs sm:text-sm text-bambu-green hover:text-bambu-green-light"
  1239. disabled={!hasAnyPermission('queue:update_own', 'queue:update_all')}
  1240. title={!hasAnyPermission('queue:update_own', 'queue:update_all') ? t('queue.permissions.noEditItems') : t('queue.bulkEdit.editSelected')}
  1241. >
  1242. <Pencil className="w-3.5 h-3.5 sm:w-4 sm:h-4" />
  1243. <span className="hidden sm:inline">{t('queue.bulkEdit.editSelected')}</span>
  1244. </Button>
  1245. <Button
  1246. variant="ghost"
  1247. size="sm"
  1248. onClick={() => bulkCancelMutation.mutate(selectedItems)}
  1249. className="flex items-center gap-1.5 sm:gap-2 text-xs sm:text-sm text-red-400 hover:text-red-300"
  1250. disabled={bulkCancelMutation.isPending || !hasAnyPermission('queue:delete_own', 'queue:delete_all')}
  1251. title={!hasAnyPermission('queue:delete_own', 'queue:delete_all') ? t('queue.permissions.noCancelItems') : t('queue.bulkEdit.cancelSelected')}
  1252. >
  1253. <X className="w-3.5 h-3.5 sm:w-4 sm:h-4" />
  1254. <span className="hidden sm:inline">{t('queue.bulkEdit.cancelSelected')}</span>
  1255. </Button>
  1256. </>
  1257. )}
  1258. </div>
  1259. <DndContext
  1260. sensors={sensors}
  1261. collisionDetection={closestCenter}
  1262. onDragEnd={handleDragEnd}
  1263. >
  1264. <SortableContext
  1265. items={pendingItems.map(i => i.id)}
  1266. strategy={verticalListSortingStrategy}
  1267. >
  1268. <div className="space-y-2 sm:space-y-3">
  1269. {pendingItems.map((item, index) => (
  1270. <SortableQueueItem
  1271. key={item.id}
  1272. item={item}
  1273. position={index + 1}
  1274. onEdit={() => setEditItem(item)}
  1275. onCancel={() => setConfirmAction({ type: 'cancel', item })}
  1276. onRemove={() => {}}
  1277. onStop={() => {}}
  1278. onRequeue={() => {}}
  1279. onStart={() => startMutation.mutate(item.id)}
  1280. timeFormat={timeFormat}
  1281. isSelected={selectedItems.includes(item.id)}
  1282. onToggleSelect={() => handleToggleSelect(item.id)}
  1283. hasPermission={hasPermission}
  1284. canModify={canModify}
  1285. t={t}
  1286. />
  1287. ))}
  1288. </div>
  1289. </SortableContext>
  1290. </DndContext>
  1291. </div>
  1292. )}
  1293. {/* History */}
  1294. {historyItems.length > 0 && (
  1295. <div>
  1296. <div className="flex flex-wrap items-center justify-between gap-2 mb-3 sm:mb-4">
  1297. <button
  1298. onClick={() => setHistoryCollapsed(!historyCollapsed)}
  1299. className="text-base sm:text-lg font-semibold text-white flex items-center gap-2 hover:text-bambu-green transition-colors"
  1300. >
  1301. {historyCollapsed ? <ChevronRight className="w-4 h-4 sm:w-5 sm:h-5" /> : <ChevronDown className="w-4 h-4 sm:w-5 sm:h-5" />}
  1302. {t('queue.sections.history')}
  1303. <span className="text-xs sm:text-sm font-normal text-bambu-gray">
  1304. ({t('queue.itemCount', { count: historyItems.length })})
  1305. </span>
  1306. </button>
  1307. {!historyCollapsed && (
  1308. <div className="flex items-center gap-2">
  1309. <select
  1310. className="px-2 sm:px-3 py-1.5 text-xs sm:text-sm bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white focus:border-bambu-green focus:outline-none"
  1311. value={historySortBy}
  1312. onChange={(e) => setHistorySortBy(e.target.value as 'date' | 'name' | 'printer')}
  1313. >
  1314. <option value="date">{t('queue.sort.byDate')}</option>
  1315. <option value="name">{t('queue.sort.byName')}</option>
  1316. <option value="printer">{t('queue.sort.byPrinter')}</option>
  1317. </select>
  1318. <Button
  1319. variant="ghost"
  1320. size="sm"
  1321. onClick={() => setHistorySortAsc(!historySortAsc)}
  1322. title={historySortAsc ? t('queue.sort.ascendingOldest') : t('queue.sort.descendingNewest')}
  1323. className="px-2"
  1324. >
  1325. {historySortAsc ? <ArrowUp className="w-4 h-4" /> : <ArrowDown className="w-4 h-4" />}
  1326. </Button>
  1327. </div>
  1328. )}
  1329. </div>
  1330. {!historyCollapsed && (
  1331. <div className="space-y-1.5 sm:space-y-2">
  1332. {historyItems.slice(0, 50).map((item) => (
  1333. <CompactHistoryRow
  1334. key={item.id}
  1335. item={item}
  1336. onRemove={() => setConfirmAction({ type: 'remove', item })}
  1337. onRequeue={() => setRequeueItem(item)}
  1338. timeFormat={timeFormat}
  1339. hasPermission={hasPermission}
  1340. canModify={canModify}
  1341. t={t}
  1342. />
  1343. ))}
  1344. </div>
  1345. )}
  1346. </div>
  1347. )}
  1348. </div>
  1349. )}
  1350. {/* Edit Modal */}
  1351. {editItem && (
  1352. <PrintModal
  1353. mode="edit-queue-item"
  1354. archiveId={editItem.archive_id ?? undefined}
  1355. libraryFileId={editItem.library_file_id ?? undefined}
  1356. archiveName={editItem.archive_name || editItem.library_file_name || `File #${editItem.archive_id || editItem.library_file_id}`}
  1357. queueItem={editItem}
  1358. onClose={() => setEditItem(null)}
  1359. />
  1360. )}
  1361. {/* Re-queue Modal */}
  1362. {requeueItem && (
  1363. <PrintModal
  1364. mode="add-to-queue"
  1365. archiveId={requeueItem.archive_id ?? undefined}
  1366. libraryFileId={requeueItem.library_file_id ?? undefined}
  1367. archiveName={requeueItem.archive_name || requeueItem.library_file_name || `File #${requeueItem.archive_id || requeueItem.library_file_id}`}
  1368. onClose={() => setRequeueItem(null)}
  1369. />
  1370. )}
  1371. {/* Confirm Action Modal */}
  1372. {confirmAction && (
  1373. <ConfirmModal
  1374. title={
  1375. confirmAction.type === 'cancel' ? t('queue.confirm.cancelTitle') :
  1376. confirmAction.type === 'stop' ? t('queue.confirm.stopTitle') :
  1377. t('queue.confirm.removeTitle')
  1378. }
  1379. message={
  1380. confirmAction.type === 'cancel'
  1381. ? t('queue.confirm.cancelMessage', { name: confirmAction.item.archive_name || confirmAction.item.library_file_name || t('queue.confirm.thisPrint') })
  1382. : confirmAction.type === 'stop'
  1383. ? t('queue.confirm.stopMessage', { name: confirmAction.item.archive_name || confirmAction.item.library_file_name || t('queue.confirm.thisPrint') })
  1384. : t('queue.confirm.removeMessage', { name: confirmAction.item.archive_name || confirmAction.item.library_file_name || t('queue.confirm.thisItem') })
  1385. }
  1386. confirmText={
  1387. confirmAction.type === 'cancel' ? t('queue.confirm.cancelButton') :
  1388. confirmAction.type === 'stop' ? t('queue.confirm.stopButton') :
  1389. t('common.remove')
  1390. }
  1391. variant="danger"
  1392. onConfirm={() => {
  1393. if (confirmAction.type === 'cancel') {
  1394. cancelMutation.mutate(confirmAction.item.id);
  1395. } else if (confirmAction.type === 'stop') {
  1396. stopMutation.mutate(confirmAction.item.id);
  1397. } else {
  1398. removeMutation.mutate(confirmAction.item.id);
  1399. }
  1400. setConfirmAction(null);
  1401. }}
  1402. onCancel={() => setConfirmAction(null)}
  1403. />
  1404. )}
  1405. {/* Clear History Confirm Modal */}
  1406. {showClearHistoryConfirm && (
  1407. <ConfirmModal
  1408. title={t('queue.confirm.clearHistoryTitle')}
  1409. message={t('queue.confirm.clearHistoryMessage', { count: historyItems.length })}
  1410. confirmText={t('queue.clearHistory')}
  1411. variant="danger"
  1412. onConfirm={() => {
  1413. clearHistoryMutation.mutate();
  1414. setShowClearHistoryConfirm(false);
  1415. }}
  1416. onCancel={() => setShowClearHistoryConfirm(false)}
  1417. />
  1418. )}
  1419. {/* Bulk Edit Modal */}
  1420. {showBulkEditModal && (
  1421. <BulkEditModal
  1422. selectedCount={selectedItems.length}
  1423. printers={printers?.map(p => ({ id: p.id, name: p.name })) || []}
  1424. onSave={(data) => {
  1425. if (Object.keys(data).length > 0) {
  1426. bulkUpdateMutation.mutate({ item_ids: selectedItems, ...data });
  1427. }
  1428. }}
  1429. onClose={() => setShowBulkEditModal(false)}
  1430. isSaving={bulkUpdateMutation.isPending}
  1431. canControlPrinter={hasPermission('printers:control')}
  1432. t={t}
  1433. />
  1434. )}
  1435. </div>
  1436. );
  1437. }