QueuePage.tsx 67 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600
  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, ApiError } 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. {/* Filament-short flag from the dispatch pre-flight (#1496). */}
  580. {item.filament_short && item.status === 'pending' && (
  581. <p
  582. className="text-[10px] sm:text-xs text-yellow-400 mt-1.5 sm:mt-2 flex items-start gap-1"
  583. title={t('queue.filamentShort.rowTooltip')}
  584. >
  585. <AlertCircle className="w-3 h-3 mt-0.5 flex-shrink-0" />
  586. <span>{t('queue.filamentShort.rowBadge')}</span>
  587. </p>
  588. )}
  589. {/* Error message */}
  590. {item.error_message && (
  591. <p className="text-[10px] sm:text-xs text-red-400 mt-1.5 sm:mt-2 flex items-center gap-1">
  592. <AlertCircle className="w-3 h-3" />
  593. {item.error_message}
  594. </p>
  595. )}
  596. </div>
  597. {/* Status badge + Actions */}
  598. <div className="flex flex-col sm:flex-row items-end sm:items-center gap-2 sm:gap-1 shrink-0" onClick={(e) => e.stopPropagation()}>
  599. <StatusBadge status={item.status} waitingReason={item.waiting_reason} printerState={printerState} t={t} />
  600. <div className="flex items-center gap-0.5 sm:gap-1">
  601. {isPrinting && (
  602. <Button
  603. variant="ghost"
  604. size="sm"
  605. onClick={onStop}
  606. disabled={!hasPermission('printers:control')}
  607. title={!hasPermission('printers:control') ? t('queue.permissions.noStopPrint') : t('queue.actions.stopPrint')}
  608. className="text-red-400 hover:text-red-300 hover:bg-red-500/10 p-1.5 sm:p-2"
  609. >
  610. <StopCircle className="w-4 h-4" />
  611. </Button>
  612. )}
  613. {isPending && (
  614. <>
  615. {item.manual_start && (
  616. <Button
  617. variant="ghost"
  618. size="sm"
  619. onClick={onStart}
  620. disabled={!hasPermission('printers:control')}
  621. title={!hasPermission('printers:control') ? t('queue.permissions.noStartPrint') : t('queue.actions.startPrint')}
  622. className="text-bambu-green hover:text-bambu-green-light hover:bg-bambu-green/10 p-1.5 sm:p-2"
  623. >
  624. <Play className="w-4 h-4" />
  625. </Button>
  626. )}
  627. <Button
  628. variant="ghost"
  629. size="sm"
  630. onClick={onEdit}
  631. disabled={!canModify('queue', 'update', item.created_by_id)}
  632. title={!canModify('queue', 'update', item.created_by_id) ? t('queue.permissions.noEdit') : t('common.edit')}
  633. className="p-1.5 sm:p-2"
  634. >
  635. <Pencil className="w-4 h-4" />
  636. </Button>
  637. <Button
  638. variant="ghost"
  639. size="sm"
  640. onClick={onCancel}
  641. disabled={!canModify('queue', 'delete', item.created_by_id)}
  642. title={!canModify('queue', 'delete', item.created_by_id) ? t('queue.permissions.noCancel') : t('common.cancel')}
  643. className="text-red-400 hover:text-red-300 hover:bg-red-500/10 p-1.5 sm:p-2"
  644. >
  645. <X className="w-4 h-4" />
  646. </Button>
  647. </>
  648. )}
  649. {isHistory && (
  650. <>
  651. <Button
  652. variant="ghost"
  653. size="sm"
  654. onClick={onRequeue}
  655. disabled={!hasPermission('queue:create')}
  656. title={!hasPermission('queue:create') ? t('queue.permissions.noRequeue') : t('queue.actions.requeue')}
  657. className="text-bambu-green hover:text-bambu-green/80 hover:bg-bambu-green/10 p-1.5 sm:p-2"
  658. >
  659. <RefreshCw className="w-4 h-4" />
  660. </Button>
  661. <Button
  662. variant="ghost"
  663. size="sm"
  664. onClick={onRemove}
  665. disabled={!canModify('queue', 'delete', item.created_by_id)}
  666. title={!canModify('queue', 'delete', item.created_by_id) ? t('queue.permissions.noRemove') : t('common.remove')}
  667. className="p-1.5 sm:p-2"
  668. >
  669. <Trash2 className="w-4 h-4" />
  670. </Button>
  671. </>
  672. )}
  673. </div>
  674. </div>
  675. </div>
  676. </div>
  677. );
  678. }
  679. export function QueuePage() {
  680. const { t } = useTranslation();
  681. const queryClient = useQueryClient();
  682. const { showToast } = useToast();
  683. const { hasPermission, hasAnyPermission, canModify } = useAuth();
  684. const [filterPrinter, setFilterPrinter] = useState<number | null>(null);
  685. const [filterStatus, setFilterStatus] = useState<string>('');
  686. const [filterLocation, setFilterLocation] = useState<string>('');
  687. const [showClearHistoryConfirm, setShowClearHistoryConfirm] = useState(false);
  688. const [editItem, setEditItem] = useState<PrintQueueItem | null>(null);
  689. const [requeueItem, setRequeueItem] = useState<PrintQueueItem | null>(null);
  690. const [confirmAction, setConfirmAction] = useState<{
  691. type: 'cancel' | 'remove' | 'stop';
  692. item: PrintQueueItem;
  693. } | null>(null);
  694. const [selectedItems, setSelectedItems] = useState<number[]>([]);
  695. const [showBulkEditModal, setShowBulkEditModal] = useState(false);
  696. const [historySortBy, setHistorySortBy] = useState<'date' | 'name' | 'printer'>(() => {
  697. const saved = localStorage.getItem('queue.historySortBy');
  698. return (saved as 'date' | 'name' | 'printer') || 'date';
  699. });
  700. const [historySortAsc, setHistorySortAsc] = useState(() => {
  701. const saved = localStorage.getItem('queue.historySortAsc');
  702. return saved !== null ? saved === 'true' : false;
  703. });
  704. const [pendingSortBy, setPendingSortBy] = useState<'position' | 'name' | 'printer' | 'time'>(() => {
  705. const saved = localStorage.getItem('queue.pendingSortBy');
  706. return (saved as 'position' | 'name' | 'printer' | 'time') || 'position';
  707. });
  708. const [pendingSortAsc, setPendingSortAsc] = useState(() => {
  709. const saved = localStorage.getItem('queue.pendingSortAsc');
  710. return saved !== null ? saved === 'true' : true;
  711. });
  712. const [historyCollapsed, setHistoryCollapsed] = useState(() => {
  713. return localStorage.getItem('queue.historyCollapsed') !== 'false';
  714. });
  715. const [viewMode, setViewMode] = useState<'list' | 'timeline'>(() => {
  716. return (localStorage.getItem('queue.viewMode') as 'list' | 'timeline') || 'list';
  717. });
  718. // Persist sort settings to localStorage
  719. useEffect(() => {
  720. localStorage.setItem('queue.historySortBy', historySortBy);
  721. }, [historySortBy]);
  722. useEffect(() => {
  723. localStorage.setItem('queue.historySortAsc', String(historySortAsc));
  724. }, [historySortAsc]);
  725. useEffect(() => {
  726. localStorage.setItem('queue.pendingSortBy', pendingSortBy);
  727. }, [pendingSortBy]);
  728. useEffect(() => {
  729. localStorage.setItem('queue.pendingSortAsc', String(pendingSortAsc));
  730. }, [pendingSortAsc]);
  731. useEffect(() => {
  732. localStorage.setItem('queue.historyCollapsed', String(historyCollapsed));
  733. }, [historyCollapsed]);
  734. useEffect(() => {
  735. localStorage.setItem('queue.viewMode', viewMode);
  736. }, [viewMode]);
  737. const sensors = useSensors(
  738. useSensor(PointerSensor, { activationConstraint: { distance: 8 } }),
  739. useSensor(KeyboardSensor, { coordinateGetter: sortableKeyboardCoordinates })
  740. );
  741. const { data: settings } = useQuery({
  742. queryKey: ['settings'],
  743. queryFn: api.getSettings,
  744. });
  745. const timeFormat: TimeFormat = settings?.time_format || 'system';
  746. const { data: queue, isLoading } = useQuery({
  747. queryKey: ['queue', filterPrinter, filterStatus],
  748. queryFn: () => api.getQueue(filterPrinter || undefined, filterStatus || undefined),
  749. refetchInterval: 5000,
  750. });
  751. const { data: printers } = useQuery({
  752. queryKey: ['printers'],
  753. queryFn: () => api.getPrinters(),
  754. });
  755. const sjfMutation = useMutation({
  756. mutationFn: (enabled: boolean) => api.updateSettings({ queue_shortest_first: enabled }),
  757. onSuccess: () => {
  758. queryClient.invalidateQueries({ queryKey: ['settings'] });
  759. },
  760. });
  761. const cancelMutation = useMutation({
  762. mutationFn: (id: number) => api.cancelQueueItem(id),
  763. onSuccess: () => {
  764. queryClient.invalidateQueries({ queryKey: ['queue'] });
  765. showToast(t('queue.toast.cancelled'));
  766. },
  767. onError: () => showToast(t('queue.toast.cancelFailed'), 'error'),
  768. });
  769. const removeMutation = useMutation({
  770. mutationFn: (id: number) => api.removeFromQueue(id),
  771. onSuccess: () => {
  772. queryClient.invalidateQueries({ queryKey: ['queue'] });
  773. showToast(t('queue.toast.removed'));
  774. },
  775. onError: () => showToast(t('queue.toast.removeFailed'), 'error'),
  776. });
  777. const stopMutation = useMutation({
  778. mutationFn: (id: number) => api.stopQueueItem(id),
  779. onSuccess: () => {
  780. queryClient.invalidateQueries({ queryKey: ['queue'] });
  781. showToast(t('queue.toast.stopped'));
  782. },
  783. onError: () => showToast(t('queue.toast.stopFailed'), 'error'),
  784. });
  785. // Filament-deficit confirmation state (#1496). When the backend returns
  786. // 409 with `code=insufficient_filament` we stash the deficit + item id
  787. // here; the modal at the bottom of the page reads it and the "Print
  788. // Anyway" path re-issues the start with `skipFilamentCheck=true`.
  789. const [filamentShortConfirm, setFilamentShortConfirm] = useState<{
  790. itemId: number;
  791. deficit: Array<{
  792. slot_id: number;
  793. required_grams: number;
  794. remaining_grams: number | null;
  795. filament_type?: string | null;
  796. }>;
  797. } | null>(null);
  798. const startMutation = useMutation({
  799. mutationFn: ({ id, skipFilamentCheck }: { id: number; skipFilamentCheck?: boolean }) =>
  800. api.startQueueItem(id, { skipFilamentCheck }),
  801. onSuccess: () => {
  802. queryClient.invalidateQueries({ queryKey: ['queue'] });
  803. showToast(t('queue.toast.released'));
  804. setFilamentShortConfirm(null);
  805. },
  806. onError: (error: unknown, variables) => {
  807. if (error instanceof ApiError && error.status === 409 && error.code === 'insufficient_filament') {
  808. const deficitRaw = (error.detail?.deficit ?? []) as Array<{
  809. slot_id: number;
  810. required_grams: number;
  811. remaining_grams: number | null;
  812. filament_type?: string | null;
  813. }>;
  814. setFilamentShortConfirm({ itemId: variables.id, deficit: deficitRaw });
  815. return;
  816. }
  817. showToast(t('queue.toast.startFailed'), 'error');
  818. },
  819. });
  820. const reorderMutation = useMutation({
  821. mutationFn: (items: { id: number; position: number }[]) => api.reorderQueue(items),
  822. onSuccess: () => {
  823. queryClient.invalidateQueries({ queryKey: ['queue'] });
  824. },
  825. onError: () => showToast(t('queue.toast.reorderFailed'), 'error'),
  826. });
  827. const clearHistoryMutation = useMutation({
  828. mutationFn: async () => {
  829. const historyItems = queue?.filter(i =>
  830. ['completed', 'failed', 'skipped', 'cancelled'].includes(i.status)
  831. ) || [];
  832. for (const item of historyItems) {
  833. await api.removeFromQueue(item.id);
  834. }
  835. return historyItems.length;
  836. },
  837. onSuccess: (count) => {
  838. queryClient.invalidateQueries({ queryKey: ['queue'] });
  839. showToast(t('queue.toast.historyCleared', { count }));
  840. },
  841. onError: () => showToast(t('queue.toast.clearHistoryFailed'), 'error'),
  842. });
  843. const bulkUpdateMutation = useMutation({
  844. mutationFn: (data: PrintQueueBulkUpdate) => api.bulkUpdateQueue(data),
  845. onSuccess: (result) => {
  846. queryClient.invalidateQueries({ queryKey: ['queue'] });
  847. setSelectedItems([]);
  848. setShowBulkEditModal(false);
  849. showToast(result.message);
  850. },
  851. onError: () => showToast(t('queue.toast.updateFailed'), 'error'),
  852. });
  853. const bulkCancelMutation = useMutation({
  854. mutationFn: async (ids: number[]) => {
  855. for (const id of ids) {
  856. await api.cancelQueueItem(id);
  857. }
  858. return ids.length;
  859. },
  860. onSuccess: (count) => {
  861. queryClient.invalidateQueries({ queryKey: ['queue'] });
  862. setSelectedItems([]);
  863. showToast(t('queue.toast.bulkCancelled', { count }));
  864. },
  865. onError: () => showToast(t('queue.toast.bulkCancelFailed'), 'error'),
  866. });
  867. const handleToggleSelect = (id: number) => {
  868. setSelectedItems(prev =>
  869. prev.includes(id) ? prev.filter(x => x !== id) : [...prev, id]
  870. );
  871. };
  872. // Get unique locations from printers for the filter dropdown
  873. const uniqueLocations = useMemo(() => {
  874. const locations = new Set<string>();
  875. printers?.forEach(p => {
  876. if (p.location) locations.add(p.location);
  877. });
  878. // Also include locations from queue items (for model-based assignments)
  879. queue?.forEach(item => {
  880. if (item.target_location) locations.add(item.target_location);
  881. });
  882. return Array.from(locations).sort();
  883. }, [printers, queue]);
  884. // Helper to check if a queue item matches the location filter
  885. const matchesLocationFilter = useCallback((item: PrintQueueItem): boolean => {
  886. if (!filterLocation) return true;
  887. // For model-based assignments, check target_location
  888. if (item.target_location) return item.target_location === filterLocation;
  889. // For printer-based assignments, check the printer's location
  890. if (item.printer_id) {
  891. const printer = printers?.find(p => p.id === item.printer_id);
  892. return printer?.location === filterLocation;
  893. }
  894. return false;
  895. }, [filterLocation, printers]);
  896. const pendingItems = useMemo(() => {
  897. let items = queue?.filter(i => i.status === 'pending') || [];
  898. // Apply location filter
  899. if (filterLocation) {
  900. items = items.filter(matchesLocationFilter);
  901. }
  902. // Helper to get scheduled time as timestamp (ASAP/placeholder = 0 for earliest)
  903. const getScheduledTime = (item: PrintQueueItem): number => {
  904. if (!item.scheduled_time) return 0;
  905. const time = parseUTCDate(item.scheduled_time)?.getTime() ?? 0;
  906. // Placeholder dates (> 6 months out) are treated as ASAP
  907. const sixMonthsFromNow = Date.now() + (180 * 24 * 60 * 60 * 1000);
  908. return time > sixMonthsFromNow ? 0 : time;
  909. };
  910. // When SJF is enabled, override sort to match scheduler order
  911. if (settings?.queue_shortest_first) {
  912. return [...items].sort((a, b) => {
  913. // Group by printer first (nulls = model-based, grouped by target_model)
  914. const aPrinter = a.printer_id ?? -(a.target_model?.charCodeAt(0) ?? 0);
  915. const bPrinter = b.printer_id ?? -(b.target_model?.charCodeAt(0) ?? 0);
  916. if (aPrinter !== bPrinter) return aPrinter - bPrinter;
  917. // Within same printer/model: jumped items first (starvation guard)
  918. const aJumped = a.been_jumped ? 1 : 0;
  919. const bJumped = b.been_jumped ? 1 : 0;
  920. if (aJumped !== bJumped) return bJumped - aJumped;
  921. // Shortest print time next (nulls last)
  922. const aTime = a.print_time_seconds ?? Infinity;
  923. const bTime = b.print_time_seconds ?? Infinity;
  924. if (aTime !== bTime) return aTime - bTime;
  925. // Position as tiebreaker
  926. return a.position - b.position;
  927. });
  928. }
  929. return [...items].sort((a, b) => {
  930. let cmp: number;
  931. if (pendingSortBy === 'name') {
  932. const aName = a.archive_name || a.library_file_name || '';
  933. const bName = b.archive_name || b.library_file_name || '';
  934. cmp = aName.localeCompare(bName);
  935. } else if (pendingSortBy === 'printer') {
  936. cmp = (a.printer_name || '').localeCompare(b.printer_name || '');
  937. } else if (pendingSortBy === 'time') {
  938. // Sort by scheduled start time (when print will begin)
  939. cmp = getScheduledTime(a) - getScheduledTime(b);
  940. } else {
  941. cmp = a.position - b.position;
  942. }
  943. return pendingSortAsc ? cmp : -cmp;
  944. });
  945. }, [queue, pendingSortBy, pendingSortAsc, matchesLocationFilter, filterLocation, settings?.queue_shortest_first]);
  946. const handleSelectAll = () => {
  947. const allPendingIds = pendingItems.map(i => i.id);
  948. if (selectedItems.length === allPendingIds.length) {
  949. setSelectedItems([]);
  950. } else {
  951. setSelectedItems(allPendingIds);
  952. }
  953. };
  954. const activeItems = useMemo(() => {
  955. let items = queue?.filter(i => i.status === 'printing') || [];
  956. if (filterLocation) {
  957. items = items.filter(matchesLocationFilter);
  958. }
  959. return items;
  960. }, [queue, filterLocation, matchesLocationFilter]);
  961. // Get unique printer IDs from active items to fetch their statuses
  962. const activePrinterIds = useMemo(() => {
  963. const ids = new Set<number>();
  964. activeItems.forEach(item => {
  965. if (item.printer_id) ids.add(item.printer_id);
  966. });
  967. return Array.from(ids);
  968. }, [activeItems]);
  969. // Fetch printer statuses for printers with active jobs
  970. const printerStatusQueries = useQueries({
  971. queries: activePrinterIds.map(printerId => ({
  972. queryKey: ['printerStatus', printerId],
  973. queryFn: () => api.getPrinterStatus(printerId),
  974. refetchInterval: 5000,
  975. })),
  976. });
  977. // Build a map of printer_id -> state for quick lookup
  978. const printerStateMap = useMemo(() => {
  979. const map: Record<number, string | null> = {};
  980. activePrinterIds.forEach((printerId, index) => {
  981. const result = printerStatusQueries[index];
  982. if (result?.data?.state) {
  983. map[printerId] = result.data.state;
  984. }
  985. });
  986. return map;
  987. }, [activePrinterIds, printerStatusQueries]);
  988. // Build a map of printer_id -> full status for timeline view
  989. const printerStatusMap = useMemo(() => {
  990. const map: Record<number, { progress?: number; remaining_time?: number; state?: string }> = {};
  991. activePrinterIds.forEach((printerId, index) => {
  992. const result = printerStatusQueries[index];
  993. if (result?.data) {
  994. map[printerId] = {
  995. progress: result.data.progress ?? undefined,
  996. remaining_time: result.data.remaining_time ?? undefined,
  997. state: result.data.state ?? undefined,
  998. };
  999. }
  1000. });
  1001. return map;
  1002. }, [activePrinterIds, printerStatusQueries]);
  1003. const historyItems = useMemo(() => {
  1004. let items = queue?.filter(i => ['completed', 'failed', 'skipped', 'cancelled'].includes(i.status)) || [];
  1005. if (filterLocation) {
  1006. items = items.filter(matchesLocationFilter);
  1007. }
  1008. return [...items].sort((a, b) => {
  1009. let cmp: number;
  1010. if (historySortBy === 'name') {
  1011. const aName = a.archive_name || a.library_file_name || '';
  1012. const bName = b.archive_name || b.library_file_name || '';
  1013. cmp = aName.localeCompare(bName);
  1014. } else if (historySortBy === 'printer') {
  1015. cmp = (a.printer_name || '').localeCompare(b.printer_name || '');
  1016. } else {
  1017. // Default: by date - most recent first (desc) is the natural order
  1018. cmp = (parseUTCDate(b.completed_at || b.created_at)?.getTime() ?? 0) - (parseUTCDate(a.completed_at || a.created_at)?.getTime() ?? 0);
  1019. }
  1020. return historySortAsc ? -cmp : cmp;
  1021. });
  1022. }, [queue, historySortBy, historySortAsc, matchesLocationFilter, filterLocation]);
  1023. // Calculate total queue time
  1024. const totalQueueTime = useMemo(() => {
  1025. return pendingItems.reduce((acc, item) => acc + (item.print_time_seconds || 0), 0);
  1026. }, [pendingItems]);
  1027. // Calculate total material weight
  1028. const totalWeight = useMemo(() => {
  1029. return pendingItems.reduce((acc, item) => acc + (item.filament_used_grams || 0), 0);
  1030. }, [pendingItems]);
  1031. const handleDragEnd = (event: DragEndEvent) => {
  1032. const { active, over } = event;
  1033. if (!over || active.id === over.id) return;
  1034. const oldIndex = pendingItems.findIndex(i => i.id === active.id);
  1035. const newIndex = pendingItems.findIndex(i => i.id === over.id);
  1036. if (oldIndex !== -1 && newIndex !== -1) {
  1037. const reordered = arrayMove(pendingItems, oldIndex, newIndex);
  1038. const updates = reordered.map((item, index) => ({
  1039. id: item.id,
  1040. position: index + 1,
  1041. }));
  1042. reorderMutation.mutate(updates);
  1043. }
  1044. };
  1045. return (
  1046. <div className="p-4 md:p-8">
  1047. {/* Header */}
  1048. <div className="flex items-center justify-between mb-8">
  1049. <div>
  1050. <h1 className="text-2xl font-bold text-white flex items-center gap-3">
  1051. <ListOrdered className="w-7 h-7 text-bambu-green" />
  1052. {t('queue.title')}
  1053. </h1>
  1054. <p className="text-bambu-gray mt-1">{t('queue.subtitle')}</p>
  1055. </div>
  1056. </div>
  1057. {/* Summary Stats */}
  1058. <QueueStatsBar
  1059. activeCount={activeItems.length}
  1060. pendingCount={pendingItems.length}
  1061. totalTime={totalQueueTime}
  1062. totalWeight={totalWeight}
  1063. historyCount={historyItems.length}
  1064. t={t}
  1065. />
  1066. {/* Filters */}
  1067. <div className="flex flex-wrap items-center gap-2 sm:gap-4 mb-6">
  1068. <select
  1069. 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"
  1070. value={filterPrinter === -1 ? 'unassigned' : (filterPrinter || '')}
  1071. onChange={(e) => {
  1072. const val = e.target.value;
  1073. if (val === 'unassigned') setFilterPrinter(-1);
  1074. else if (val === '') setFilterPrinter(null);
  1075. else setFilterPrinter(Number(val));
  1076. }}
  1077. >
  1078. <option value="">{t('queue.filter.allPrinters')}</option>
  1079. <option value="unassigned">{t('queue.filter.unassigned')}</option>
  1080. {printers?.map((p) => (
  1081. <option key={p.id} value={p.id}>{p.name}</option>
  1082. ))}
  1083. </select>
  1084. <select
  1085. 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"
  1086. value={filterStatus}
  1087. onChange={(e) => setFilterStatus(e.target.value)}
  1088. >
  1089. <option value="">{t('queue.filter.allStatus')}</option>
  1090. <option value="pending">{t('queue.status.pending')}</option>
  1091. <option value="printing">{t('queue.status.printing')}</option>
  1092. <option value="completed">{t('queue.status.completed')}</option>
  1093. <option value="failed">{t('queue.status.failed')}</option>
  1094. <option value="skipped">{t('queue.status.skipped')}</option>
  1095. <option value="cancelled">{t('queue.status.cancelled')}</option>
  1096. </select>
  1097. {uniqueLocations.length > 0 && (
  1098. <select
  1099. 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"
  1100. value={filterLocation}
  1101. onChange={(e) => setFilterLocation(e.target.value)}
  1102. >
  1103. <option value="">{t('queue.filter.allLocations')}</option>
  1104. {uniqueLocations.map((loc) => (
  1105. <option key={loc} value={loc}>{loc}</option>
  1106. ))}
  1107. </select>
  1108. )}
  1109. <div className="hidden sm:block flex-1" />
  1110. {historyItems.length > 0 && (
  1111. <Button
  1112. className="w-full sm:w-auto"
  1113. variant="secondary"
  1114. size="sm"
  1115. onClick={() => setShowClearHistoryConfirm(true)}
  1116. disabled={!hasPermission('queue:delete_all')}
  1117. title={!hasPermission('queue:delete_all') ? t('queue.permissions.noClearHistory') : undefined}
  1118. >
  1119. <Trash2 className="w-4 h-4" />
  1120. {t('queue.clearHistory')}
  1121. </Button>
  1122. )}
  1123. </div>
  1124. {/* View Mode Toggle + SJF */}
  1125. <div className="flex items-center gap-3 mb-6">
  1126. <div className="hidden sm:flex items-center border border-bambu-dark-tertiary rounded-lg overflow-hidden">
  1127. <button
  1128. className={`p-2 transition-colors ${viewMode === 'list' ? 'bg-bambu-green text-white' : 'bg-bambu-dark text-bambu-gray hover:text-white'}`}
  1129. onClick={() => setViewMode('list')}
  1130. title={t('queue.timeline.listView')}
  1131. >
  1132. <List className="w-4 h-4" />
  1133. </button>
  1134. <button
  1135. className={`p-2 transition-colors ${viewMode === 'timeline' ? 'bg-bambu-green text-white' : 'bg-bambu-dark text-bambu-gray hover:text-white'}`}
  1136. onClick={() => setViewMode('timeline')}
  1137. title={t('queue.timeline.timelineView')}
  1138. >
  1139. <GanttChart className="w-4 h-4" />
  1140. </button>
  1141. </div>
  1142. <button
  1143. onClick={() => {
  1144. const newValue = !(settings?.queue_shortest_first ?? false);
  1145. sjfMutation.mutate(newValue);
  1146. }}
  1147. className={`flex items-center gap-1 px-2 py-1.5 text-xs rounded-lg border transition-colors ${
  1148. settings?.queue_shortest_first
  1149. ? 'bg-bambu-green/20 border-bambu-green text-bambu-green'
  1150. : 'bg-bambu-dark-secondary border-bambu-dark-tertiary text-bambu-gray hover:text-white hover:border-bambu-gray'
  1151. }`}
  1152. title={t('queue.sjf.tooltip', 'Shortest Job First — scheduler prioritizes shorter prints')}
  1153. >
  1154. <Snail className="w-4 h-4" />
  1155. <span className="hidden sm:inline">{t('queue.sjf.label', 'SJF')}</span>
  1156. <span className={`w-1.5 h-1.5 rounded-full ${settings?.queue_shortest_first ? 'bg-bambu-green' : 'bg-bambu-gray'}`} />
  1157. </button>
  1158. </div>
  1159. {isLoading ? (
  1160. <div className="text-center py-12 text-bambu-gray">{t('common.loading')}</div>
  1161. ) : queue?.length === 0 ? (
  1162. <Card className="p-12 text-center border-dashed">
  1163. <Calendar className="w-16 h-16 text-bambu-gray mx-auto mb-4 opacity-50" />
  1164. <h3 className="text-xl font-medium text-white mb-2">{t('queue.empty.title')}</h3>
  1165. <p className="text-bambu-gray max-w-md mx-auto">
  1166. {t('queue.empty.description')}
  1167. </p>
  1168. </Card>
  1169. ) : viewMode === 'timeline' ? (
  1170. <QueueTimelineView
  1171. queueItems={queue || []}
  1172. printerStatuses={printerStatusMap}
  1173. onItemClick={(item) => {
  1174. if (['completed', 'failed', 'skipped', 'cancelled'].includes(item.status)) {
  1175. setRequeueItem(item);
  1176. } else if (item.status === 'pending') {
  1177. setEditItem(item);
  1178. } else if (item.status === 'printing') {
  1179. setConfirmAction({ type: 'stop', item });
  1180. }
  1181. }}
  1182. t={t}
  1183. />
  1184. ) : (
  1185. <div className="space-y-6 sm:space-y-8">
  1186. {/* Active Prints */}
  1187. {activeItems.length > 0 && (
  1188. <div>
  1189. <h2 className="text-base sm:text-lg font-semibold text-white mb-3 sm:mb-4 flex items-center gap-2">
  1190. <div className="w-2 h-2 rounded-full bg-blue-400 animate-pulse" />
  1191. {t('queue.sections.currentlyPrinting')}
  1192. </h2>
  1193. <div className="space-y-2 sm:space-y-3">
  1194. {activeItems.map((item) => (
  1195. <SortableQueueItem
  1196. key={item.id}
  1197. item={item}
  1198. onEdit={() => {}}
  1199. onCancel={() => {}}
  1200. onRemove={() => {}}
  1201. onStop={() => setConfirmAction({ type: 'stop', item })}
  1202. onRequeue={() => {}}
  1203. onStart={() => {}}
  1204. timeFormat={timeFormat}
  1205. hasPermission={hasPermission}
  1206. canModify={canModify}
  1207. printerState={item.printer_id ? printerStateMap[item.printer_id] : null}
  1208. t={t}
  1209. />
  1210. ))}
  1211. </div>
  1212. </div>
  1213. )}
  1214. {/* Pending Queue */}
  1215. {pendingItems.length > 0 && (
  1216. <div>
  1217. <div className="flex flex-wrap items-center justify-between gap-2 mb-3 sm:mb-4">
  1218. <h2 className="text-base sm:text-lg font-semibold text-white flex items-center gap-2">
  1219. <Clock className="w-4 h-4 sm:w-5 sm:h-5 text-yellow-400" />
  1220. {t('queue.sections.queued')}
  1221. <span className="text-xs sm:text-sm font-normal text-bambu-gray">
  1222. ({t('queue.itemCount', { count: pendingItems.length })})
  1223. </span>
  1224. <span className="hidden sm:inline text-xs text-bambu-gray ml-2" title={t('queue.reorderHint')}>
  1225. {t('queue.dragToReorder')}
  1226. </span>
  1227. </h2>
  1228. <div className="flex items-center gap-2">
  1229. <select
  1230. 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"
  1231. value={pendingSortBy}
  1232. onChange={(e) => setPendingSortBy(e.target.value as 'position' | 'name' | 'printer' | 'time')}
  1233. >
  1234. <option value="position">{t('queue.sort.byPosition')}</option>
  1235. <option value="name">{t('queue.sort.byName')}</option>
  1236. <option value="printer">{t('queue.sort.byPrinter')}</option>
  1237. <option value="time">{t('queue.sort.bySchedule')}</option>
  1238. </select>
  1239. <Button
  1240. variant="ghost"
  1241. size="sm"
  1242. onClick={() => setPendingSortAsc(!pendingSortAsc)}
  1243. title={pendingSortAsc ? t('common.ascending') : t('common.descending')}
  1244. className="px-2"
  1245. >
  1246. {pendingSortAsc ? <ArrowUp className="w-4 h-4" /> : <ArrowDown className="w-4 h-4" />}
  1247. </Button>
  1248. </div>
  1249. </div>
  1250. {/* Bulk action toolbar */}
  1251. <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">
  1252. <Button
  1253. variant="ghost"
  1254. size="sm"
  1255. onClick={handleSelectAll}
  1256. className="flex items-center gap-1.5 sm:gap-2 text-xs sm:text-sm"
  1257. >
  1258. {selectedItems.length === pendingItems.length && pendingItems.length > 0 ? (
  1259. <CheckSquare className="w-4 h-4 text-bambu-green" />
  1260. ) : (
  1261. <Square className="w-4 h-4" />
  1262. )}
  1263. {selectedItems.length === pendingItems.length && pendingItems.length > 0 ? t('queue.bulkEdit.deselectAll') : t('queue.bulkEdit.selectAll')}
  1264. </Button>
  1265. {selectedItems.length > 0 && (
  1266. <>
  1267. <span className="text-xs sm:text-sm text-bambu-gray">
  1268. {t('queue.bulkEdit.selected', { count: selectedItems.length })}
  1269. </span>
  1270. <div className="hidden sm:block h-4 w-px bg-bambu-dark-tertiary" />
  1271. <Button
  1272. variant="ghost"
  1273. size="sm"
  1274. onClick={() => setShowBulkEditModal(true)}
  1275. className="flex items-center gap-1.5 sm:gap-2 text-xs sm:text-sm text-bambu-green hover:text-bambu-green-light"
  1276. disabled={!hasAnyPermission('queue:update_own', 'queue:update_all')}
  1277. title={!hasAnyPermission('queue:update_own', 'queue:update_all') ? t('queue.permissions.noEditItems') : t('queue.bulkEdit.editSelected')}
  1278. >
  1279. <Pencil className="w-3.5 h-3.5 sm:w-4 sm:h-4" />
  1280. <span className="hidden sm:inline">{t('queue.bulkEdit.editSelected')}</span>
  1281. </Button>
  1282. <Button
  1283. variant="ghost"
  1284. size="sm"
  1285. onClick={() => bulkCancelMutation.mutate(selectedItems)}
  1286. className="flex items-center gap-1.5 sm:gap-2 text-xs sm:text-sm text-red-400 hover:text-red-300"
  1287. disabled={bulkCancelMutation.isPending || !hasAnyPermission('queue:delete_own', 'queue:delete_all')}
  1288. title={!hasAnyPermission('queue:delete_own', 'queue:delete_all') ? t('queue.permissions.noCancelItems') : t('queue.bulkEdit.cancelSelected')}
  1289. >
  1290. <X className="w-3.5 h-3.5 sm:w-4 sm:h-4" />
  1291. <span className="hidden sm:inline">{t('queue.bulkEdit.cancelSelected')}</span>
  1292. </Button>
  1293. </>
  1294. )}
  1295. </div>
  1296. <DndContext
  1297. sensors={sensors}
  1298. collisionDetection={closestCenter}
  1299. onDragEnd={handleDragEnd}
  1300. >
  1301. <SortableContext
  1302. items={pendingItems.map(i => i.id)}
  1303. strategy={verticalListSortingStrategy}
  1304. >
  1305. <div className="space-y-2 sm:space-y-3">
  1306. {pendingItems.map((item, index) => (
  1307. <SortableQueueItem
  1308. key={item.id}
  1309. item={item}
  1310. position={index + 1}
  1311. onEdit={() => setEditItem(item)}
  1312. onCancel={() => setConfirmAction({ type: 'cancel', item })}
  1313. onRemove={() => {}}
  1314. onStop={() => {}}
  1315. onRequeue={() => {}}
  1316. onStart={() => startMutation.mutate({ id: item.id })}
  1317. timeFormat={timeFormat}
  1318. isSelected={selectedItems.includes(item.id)}
  1319. onToggleSelect={() => handleToggleSelect(item.id)}
  1320. hasPermission={hasPermission}
  1321. canModify={canModify}
  1322. t={t}
  1323. />
  1324. ))}
  1325. </div>
  1326. </SortableContext>
  1327. </DndContext>
  1328. </div>
  1329. )}
  1330. {/* History */}
  1331. {historyItems.length > 0 && (
  1332. <div>
  1333. <div className="flex flex-wrap items-center justify-between gap-2 mb-3 sm:mb-4">
  1334. <button
  1335. onClick={() => setHistoryCollapsed(!historyCollapsed)}
  1336. className="text-base sm:text-lg font-semibold text-white flex items-center gap-2 hover:text-bambu-green transition-colors"
  1337. >
  1338. {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" />}
  1339. {t('queue.sections.history')}
  1340. <span className="text-xs sm:text-sm font-normal text-bambu-gray">
  1341. ({t('queue.itemCount', { count: historyItems.length })})
  1342. </span>
  1343. </button>
  1344. {!historyCollapsed && (
  1345. <div className="flex items-center gap-2">
  1346. <select
  1347. 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"
  1348. value={historySortBy}
  1349. onChange={(e) => setHistorySortBy(e.target.value as 'date' | 'name' | 'printer')}
  1350. >
  1351. <option value="date">{t('queue.sort.byDate')}</option>
  1352. <option value="name">{t('queue.sort.byName')}</option>
  1353. <option value="printer">{t('queue.sort.byPrinter')}</option>
  1354. </select>
  1355. <Button
  1356. variant="ghost"
  1357. size="sm"
  1358. onClick={() => setHistorySortAsc(!historySortAsc)}
  1359. title={historySortAsc ? t('queue.sort.ascendingOldest') : t('queue.sort.descendingNewest')}
  1360. className="px-2"
  1361. >
  1362. {historySortAsc ? <ArrowUp className="w-4 h-4" /> : <ArrowDown className="w-4 h-4" />}
  1363. </Button>
  1364. </div>
  1365. )}
  1366. </div>
  1367. {!historyCollapsed && (
  1368. <div className="space-y-1.5 sm:space-y-2">
  1369. {historyItems.slice(0, 50).map((item) => (
  1370. <CompactHistoryRow
  1371. key={item.id}
  1372. item={item}
  1373. onRemove={() => setConfirmAction({ type: 'remove', item })}
  1374. onRequeue={() => setRequeueItem(item)}
  1375. timeFormat={timeFormat}
  1376. hasPermission={hasPermission}
  1377. canModify={canModify}
  1378. t={t}
  1379. />
  1380. ))}
  1381. </div>
  1382. )}
  1383. </div>
  1384. )}
  1385. </div>
  1386. )}
  1387. {/* Edit Modal */}
  1388. {editItem && (
  1389. <PrintModal
  1390. mode="edit-queue-item"
  1391. archiveId={editItem.archive_id ?? undefined}
  1392. libraryFileId={editItem.library_file_id ?? undefined}
  1393. archiveName={editItem.archive_name || editItem.library_file_name || `File #${editItem.archive_id || editItem.library_file_id}`}
  1394. queueItem={editItem}
  1395. onClose={() => setEditItem(null)}
  1396. />
  1397. )}
  1398. {/* Re-queue Modal */}
  1399. {requeueItem && (
  1400. <PrintModal
  1401. mode="add-to-queue"
  1402. archiveId={requeueItem.archive_id ?? undefined}
  1403. libraryFileId={requeueItem.library_file_id ?? undefined}
  1404. archiveName={requeueItem.archive_name || requeueItem.library_file_name || `File #${requeueItem.archive_id || requeueItem.library_file_id}`}
  1405. onClose={() => setRequeueItem(null)}
  1406. />
  1407. )}
  1408. {/* Confirm Action Modal */}
  1409. {filamentShortConfirm && (
  1410. <ConfirmModal
  1411. title={t('queue.filamentShort.confirmTitle')}
  1412. message={
  1413. t('queue.filamentShort.confirmIntro') + '\n\n' +
  1414. filamentShortConfirm.deficit
  1415. .map((d) =>
  1416. t('queue.filamentShort.lineItem', {
  1417. slot: d.slot_id,
  1418. required: Math.round(d.required_grams),
  1419. remaining:
  1420. d.remaining_grams == null
  1421. ? t('queue.filamentShort.unknown')
  1422. : Math.round(d.remaining_grams),
  1423. }),
  1424. )
  1425. .join('\n')
  1426. }
  1427. confirmText={t('queue.filamentShort.printAnyway')}
  1428. variant="warning"
  1429. onConfirm={() => {
  1430. startMutation.mutate({ id: filamentShortConfirm.itemId, skipFilamentCheck: true });
  1431. }}
  1432. onCancel={() => setFilamentShortConfirm(null)}
  1433. />
  1434. )}
  1435. {confirmAction && (
  1436. <ConfirmModal
  1437. title={
  1438. confirmAction.type === 'cancel' ? t('queue.confirm.cancelTitle') :
  1439. confirmAction.type === 'stop' ? t('queue.confirm.stopTitle') :
  1440. t('queue.confirm.removeTitle')
  1441. }
  1442. message={
  1443. confirmAction.type === 'cancel'
  1444. ? t('queue.confirm.cancelMessage', { name: confirmAction.item.archive_name || confirmAction.item.library_file_name || t('queue.confirm.thisPrint') })
  1445. : confirmAction.type === 'stop'
  1446. ? t('queue.confirm.stopMessage', { name: confirmAction.item.archive_name || confirmAction.item.library_file_name || t('queue.confirm.thisPrint') })
  1447. : t('queue.confirm.removeMessage', { name: confirmAction.item.archive_name || confirmAction.item.library_file_name || t('queue.confirm.thisItem') })
  1448. }
  1449. confirmText={
  1450. confirmAction.type === 'cancel' ? t('queue.confirm.cancelButton') :
  1451. confirmAction.type === 'stop' ? t('queue.confirm.stopButton') :
  1452. t('common.remove')
  1453. }
  1454. variant="danger"
  1455. onConfirm={() => {
  1456. if (confirmAction.type === 'cancel') {
  1457. cancelMutation.mutate(confirmAction.item.id);
  1458. } else if (confirmAction.type === 'stop') {
  1459. stopMutation.mutate(confirmAction.item.id);
  1460. } else {
  1461. removeMutation.mutate(confirmAction.item.id);
  1462. }
  1463. setConfirmAction(null);
  1464. }}
  1465. onCancel={() => setConfirmAction(null)}
  1466. />
  1467. )}
  1468. {/* Clear History Confirm Modal */}
  1469. {showClearHistoryConfirm && (
  1470. <ConfirmModal
  1471. title={t('queue.confirm.clearHistoryTitle')}
  1472. message={t('queue.confirm.clearHistoryMessage', { count: historyItems.length })}
  1473. confirmText={t('queue.clearHistory')}
  1474. variant="danger"
  1475. onConfirm={() => {
  1476. clearHistoryMutation.mutate();
  1477. setShowClearHistoryConfirm(false);
  1478. }}
  1479. onCancel={() => setShowClearHistoryConfirm(false)}
  1480. />
  1481. )}
  1482. {/* Bulk Edit Modal */}
  1483. {showBulkEditModal && (
  1484. <BulkEditModal
  1485. selectedCount={selectedItems.length}
  1486. printers={printers?.map(p => ({ id: p.id, name: p.name })) || []}
  1487. onSave={(data) => {
  1488. if (Object.keys(data).length > 0) {
  1489. bulkUpdateMutation.mutate({ item_ids: selectedItems, ...data });
  1490. }
  1491. }}
  1492. onClose={() => setShowBulkEditModal(false)}
  1493. isSaving={bulkUpdateMutation.isPending}
  1494. canControlPrinter={hasPermission('printers:control')}
  1495. t={t}
  1496. />
  1497. )}
  1498. </div>
  1499. );
  1500. }