QueuePage.tsx 63 KB

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