InventoryPage.tsx 99 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504150515061507150815091510151115121513151415151516151715181519152015211522152315241525152615271528152915301531153215331534153515361537153815391540154115421543154415451546154715481549155015511552155315541555155615571558155915601561156215631564156515661567156815691570157115721573157415751576157715781579158015811582158315841585158615871588158915901591159215931594159515961597159815991600160116021603160416051606160716081609161016111612161316141615161616171618161916201621162216231624162516261627162816291630163116321633163416351636163716381639164016411642164316441645164616471648164916501651165216531654165516561657165816591660166116621663166416651666166716681669167016711672167316741675167616771678167916801681168216831684168516861687168816891690169116921693169416951696169716981699170017011702170317041705170617071708170917101711171217131714171517161717171817191720172117221723172417251726172717281729173017311732173317341735173617371738173917401741174217431744174517461747174817491750175117521753175417551756175717581759176017611762176317641765176617671768176917701771177217731774177517761777177817791780178117821783178417851786178717881789179017911792179317941795179617971798179918001801180218031804180518061807180818091810181118121813181418151816181718181819182018211822182318241825182618271828182918301831183218331834183518361837183818391840184118421843184418451846184718481849185018511852185318541855185618571858185918601861186218631864186518661867186818691870187118721873187418751876187718781879188018811882188318841885188618871888188918901891189218931894189518961897189818991900190119021903190419051906190719081909191019111912191319141915191619171918191919201921192219231924192519261927192819291930193119321933193419351936193719381939194019411942194319441945194619471948194919501951195219531954195519561957195819591960196119621963196419651966196719681969197019711972197319741975197619771978197919801981198219831984198519861987198819891990199119921993199419951996199719981999200020012002200320042005200620072008200920102011201220132014201520162017201820192020202120222023202420252026202720282029203020312032203320342035203620372038203920402041204220432044204520462047204820492050205120522053205420552056205720582059206020612062206320642065206620672068206920702071207220732074207520762077207820792080208120822083208420852086208720882089209020912092209320942095209620972098209921002101210221032104210521062107210821092110211121122113211421152116211721182119212021212122212321242125212621272128212921302131213221332134213521362137213821392140214121422143214421452146214721482149215021512152215321542155215621572158215921602161216221632164216521662167216821692170217121722173217421752176217721782179218021812182218321842185218621872188218921902191219221932194219521962197219821992200220122022203220422052206220722082209221022112212221322142215221622172218221922202221222222232224222522262227222822292230223122322233223422352236223722382239224022412242224322442245224622472248224922502251225222532254225522562257225822592260226122622263226422652266226722682269227022712272227322742275227622772278227922802281228222832284228522862287228822892290229122922293229422952296229722982299230023012302230323042305230623072308230923102311231223132314231523162317231823192320232123222323232423252326232723282329
  1. import { useState, useMemo, useEffect, useRef, useCallback, type ReactNode } from 'react';
  2. import { useSearchParams } from 'react-router-dom';
  3. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  4. import { useTranslation } from 'react-i18next';
  5. import {
  6. Plus, Loader2, Trash2, Archive, RotateCcw, Edit2, Package,
  7. Search, ChevronLeft, ChevronRight, ChevronsLeft, ChevronsRight,
  8. TrendingDown, Layers, Printer, AlertTriangle, X, Clock, LayoutGrid, TableProperties, Columns,
  9. ArrowUp, ArrowDown, ArrowUpDown, Group, ChevronDown, Check, RefreshCw, TrendingUp, Lock, Copy, Eraser,
  10. } from 'lucide-react';
  11. import { ForecastPanel } from '../components/ForecastPanel';
  12. import { api, spoolbuddyApi, ApiError } from '../api/client';
  13. import type { InventorySpool, SpoolCatalogEntry } from '../api/client';
  14. import { Button } from '../components/Button';
  15. import { FilamentSwatch } from '../components/FilamentSwatch';
  16. import { buildFilamentBackground } from '../components/filamentSwatchHelpers';
  17. import {SpoolFormModal, type SpoolFormMode} from '../components/SpoolFormModal';
  18. import { ConfirmModal } from '../components/ConfirmModal';
  19. import { ColumnConfigModal, type ColumnConfig } from '../components/ColumnConfigModal';
  20. import { LabelTemplatePickerModal } from '../components/LabelTemplatePickerModal';
  21. import { useToast } from '../contexts/ToastContext';
  22. import { useAuth } from '../contexts/AuthContext';
  23. import { resolveSpoolColorName } from '../utils/colors';
  24. import { getCurrencySymbol } from '../utils/currency';
  25. import { formatDateInput, parseUTCDate, type DateFormat } from '../utils/date';
  26. import { formatSlotLabel } from '../utils/amsHelpers';
  27. import { filterSpoolsByQuery } from '../utils/inventorySearch';
  28. import { aggregateGroupSpool } from '../utils/inventoryGrouping';
  29. type ArchiveFilter = 'active' | 'archived';
  30. type UsageFilter = 'all' | 'used' | 'new' | 'lowstock';
  31. type ViewMode = 'table' | 'cards' | 'forecast';
  32. type SortDirection = 'asc' | 'desc';
  33. type SortState = { column: string; direction: SortDirection } | null;
  34. type DisplayItem =
  35. | { type: 'single'; spool: InventorySpool }
  36. | { type: 'group'; key: string; spools: InventorySpool[]; representative: InventorySpool };
  37. function spoolGroupKey(s: InventorySpool): string {
  38. // Include extra_colors + effect_type so the "Group similar" toggle does
  39. // not collapse two spools that share the base colour but differ on
  40. // gradient stops or visual effect (#1154).
  41. return `${s.material}|${s.subtype || ''}|${s.brand || ''}|${s.color_name || ''}|${s.rgba || ''}|${s.extra_colors || ''}|${s.effect_type || ''}|${s.label_weight}`;
  42. }
  43. // Column definitions for the inventory table
  44. const COLUMN_CONFIG_KEY = 'bambuddy-inventory-columns';
  45. const DEFAULT_COLUMNS: ColumnConfig[] = [
  46. { id: 'id', label: '#', visible: true },
  47. { id: 'added_time', label: 'Added', visible: true },
  48. { id: 'encode_time', label: 'Encoded', visible: false },
  49. { id: 'last_used_time', label: 'Last Used', visible: false },
  50. { id: 'rgba', label: 'Color', visible: true },
  51. { id: 'material', label: 'Material', visible: true },
  52. { id: 'subtype', label: 'Subtype', visible: true },
  53. { id: 'color_name', label: 'Color Name', visible: false },
  54. { id: 'brand', label: 'Brand', visible: true },
  55. { id: 'slicer_filament', label: 'Slicer Filament', visible: false },
  56. { id: 'location', label: 'Location', visible: true },
  57. { id: 'storage_location', label: 'Storage Location', visible: false },
  58. { id: 'label_weight', label: 'Label', visible: true },
  59. { id: 'net', label: 'Net', visible: true },
  60. { id: 'gross', label: 'Gross', visible: false },
  61. { id: 'added_full', label: 'Full', visible: false },
  62. { id: 'used', label: 'Used', visible: false },
  63. { id: 'printed_total', label: 'Printed Total', visible: false },
  64. { id: 'printed_since_weight', label: 'Printed Since Weight', visible: false },
  65. { id: 'note', label: 'Note', visible: false },
  66. { id: 'pa_k', label: 'PA(K)', visible: true },
  67. { id: 'tag_id', label: 'Tag ID', visible: false },
  68. { id: 'data_origin', label: 'Data Origin', visible: false },
  69. { id: 'tag_type', label: 'Linked Tag Type', visible: false },
  70. { id: 'stock', label: 'Stock', visible: false },
  71. { id: 'remaining', label: 'Remaining', visible: true },
  72. { id: 'spool_name', label: 'Spool', visible: false },
  73. { id: 'cost_per_kg', label: 'Cost/kg', visible: false },
  74. { id: 'weight_check', label: 'Weight Check', visible: false },
  75. ];
  76. function loadColumnConfig(): ColumnConfig[] {
  77. try {
  78. const stored = localStorage.getItem(COLUMN_CONFIG_KEY);
  79. if (stored) {
  80. const parsed = JSON.parse(stored) as ColumnConfig[];
  81. const defaultIds = new Set(DEFAULT_COLUMNS.map((c) => c.id));
  82. const storedIds = new Set(parsed.map((c) => c.id));
  83. // Keep stored columns that still exist in defaults
  84. const validStored = parsed.filter((c) => defaultIds.has(c.id));
  85. // Add any new default columns not in stored config
  86. const newColumns = DEFAULT_COLUMNS.filter((c) => !storedIds.has(c.id));
  87. return [...validStored, ...newColumns];
  88. }
  89. } catch {
  90. // Ignore errors
  91. }
  92. return DEFAULT_COLUMNS.map((c) => ({ ...c }));
  93. }
  94. function saveColumnConfig(config: ColumnConfig[]) {
  95. try {
  96. localStorage.setItem(COLUMN_CONFIG_KEY, JSON.stringify(config));
  97. } catch {
  98. // Ignore errors
  99. }
  100. }
  101. function formatWeight(g: number, useKg = false): string {
  102. if (useKg && g >= 1000) return `${(g / 1000).toFixed(1)}kg`;
  103. return `${Math.round(g)}g`;
  104. }
  105. // Material color mapping for pills
  106. const MATERIAL_COLORS: Record<string, string> = {
  107. PLA: 'bg-green-500/20 text-green-400',
  108. ABS: 'bg-red-500/20 text-red-400',
  109. PETG: 'bg-blue-500/20 text-blue-400',
  110. TPU: 'bg-purple-500/20 text-purple-400',
  111. ASA: 'bg-orange-500/20 text-orange-400',
  112. PA: 'bg-yellow-500/20 text-yellow-400',
  113. PC: 'bg-cyan-500/20 text-cyan-400',
  114. PET: 'bg-sky-500/20 text-sky-400',
  115. };
  116. type TFn = (key: string, opts?: Record<string, unknown>) => string;
  117. function formatInventoryDate(dateStr: string | null, dateFormat: DateFormat = 'system'): string {
  118. if (!dateStr) return '-';
  119. const date = parseUTCDate(dateStr);
  120. if (!date) return '-';
  121. return formatDateInput(date, dateFormat);
  122. }
  123. // Slim shape for the LOCATION column — only the fields actually rendered.
  124. // Sourced from either local SpoolAssignment (lokal) or SpoolmanSlotAssignment
  125. // (Spoolman mode), so we can't reuse SpoolAssignment without dummy values.
  126. type LocationDisplay = {
  127. printer_id: number;
  128. printer_name: string | null;
  129. ams_id: number;
  130. tray_id: number;
  131. ams_label: string | null;
  132. };
  133. type CellCtx = {
  134. spool: InventorySpool;
  135. remaining: number;
  136. pct: number;
  137. assignmentMap: Record<number, LocationDisplay>;
  138. catalogMap: Record<number, SpoolCatalogEntry>;
  139. currencySymbol: string;
  140. dateFormat: DateFormat;
  141. t: TFn;
  142. onSyncWeight?: (spool: InventorySpool) => void;
  143. };
  144. // Column header labels (25 columns — matching SpoolBuddy exactly)
  145. const columnHeaders: Record<string, (t: TFn) => string> = {
  146. id: () => '#',
  147. added_time: () => 'Added',
  148. encode_time: () => 'Encoded',
  149. last_used_time: () => 'Last Used',
  150. rgba: (t) => t('inventory.color'),
  151. material: (t) => t('inventory.material'),
  152. subtype: (t) => t('inventory.subtype'),
  153. color_name: (t) => t('inventory.colorName'),
  154. brand: (t) => t('inventory.brand'),
  155. slicer_filament: (t) => t('inventory.slicerFilament'),
  156. location: () => 'Location',
  157. storage_location: (t) => t('inventory.storageLocation'),
  158. label_weight: (t) => t('inventory.labelWeight'),
  159. net: (t) => t('inventory.net'),
  160. gross: () => 'Gross',
  161. added_full: () => 'Full',
  162. used: (t) => t('inventory.weightUsed'),
  163. printed_total: () => 'Printed Total',
  164. printed_since_weight: () => 'Printed Since Weight',
  165. note: (t) => t('inventory.note'),
  166. pa_k: () => 'PA(K)',
  167. tag_id: () => 'Tag ID',
  168. data_origin: () => 'Data Origin',
  169. tag_type: () => 'Linked Tag Type',
  170. stock: (t) => t('inventory.stock'),
  171. remaining: (t) => t('inventory.remaining'),
  172. spool_name: (t) => t('inventory.spoolName'),
  173. cost_per_kg: (t) => t('inventory.costPerKg'),
  174. weight_check: (t) => t('inventory.weightCheck'),
  175. };
  176. // Column cell renderers (25 columns — matching SpoolBuddy exactly)
  177. const columnCells: Record<string, (ctx: CellCtx) => ReactNode> = {
  178. id: ({ spool }) => (
  179. <span className="text-sm font-medium text-white">{spool.id}</span>
  180. ),
  181. added_time: ({ spool, dateFormat }) => (
  182. <span className="text-sm text-bambu-gray">{formatInventoryDate(spool.created_at, dateFormat)}</span>
  183. ),
  184. encode_time: ({ spool, dateFormat }) => (
  185. <span className="text-sm text-bambu-gray">{formatInventoryDate(spool.encode_time, dateFormat)}</span>
  186. ),
  187. last_used_time: ({ spool, dateFormat }) => (
  188. <span className="text-sm text-bambu-gray">{spool.last_used ? formatInventoryDate(spool.last_used, dateFormat) : 'Never'}</span>
  189. ),
  190. rgba: ({ spool }) => (
  191. <div className="flex items-center justify-center">
  192. <FilamentSwatch
  193. rgba={spool.rgba}
  194. extraColors={spool.extra_colors}
  195. effectType={spool.effect_type}
  196. effectSize="table"
  197. subtype={spool.subtype}
  198. />
  199. </div>
  200. ),
  201. material: ({ spool }) => (
  202. <span className="text-sm text-white">{spool.material}</span>
  203. ),
  204. subtype: ({ spool }) => (
  205. <span className="text-sm text-bambu-gray">{spool.subtype || '-'}</span>
  206. ),
  207. color_name: ({ spool }) => (
  208. <span className="text-sm text-bambu-gray">{resolveSpoolColorName(spool.color_name, spool.rgba) || '-'}</span>
  209. ),
  210. brand: ({ spool }) => (
  211. <span className="text-sm text-bambu-gray">{spool.brand || '-'}</span>
  212. ),
  213. slicer_filament: ({ spool }) => (
  214. <span className="text-sm text-bambu-gray" title={spool.slicer_filament || undefined}>
  215. {spool.slicer_filament_name || spool.slicer_filament || '-'}
  216. </span>
  217. ),
  218. location: ({ spool, assignmentMap }) => {
  219. const assignment = assignmentMap[spool.id];
  220. if (!assignment) return <span className="text-sm text-bambu-gray">-</span>;
  221. const printerLabel = assignment.printer_name || `Printer ${assignment.printer_id}`;
  222. const isExternal = assignment.ams_id === 254 || assignment.ams_id === 255;
  223. const isHt = !isExternal && assignment.ams_id >= 128;
  224. const slotLabel = formatSlotLabel(assignment.ams_id, assignment.tray_id, isHt, isExternal);
  225. return (
  226. <span className="inline-flex items-center px-1.5 py-0.5 rounded text-xs font-medium bg-purple-500/20 text-purple-400">
  227. {printerLabel} {slotLabel}{assignment.ams_label ? ` (${assignment.ams_label})` : ''}
  228. </span>
  229. );
  230. },
  231. storage_location: ({ spool }) => {
  232. if (!spool.storage_location) return <span className="text-sm text-bambu-gray">-</span>;
  233. return (
  234. <span className="inline-flex items-center px-1.5 py-0.5 rounded text-xs font-medium bg-blue-500/20 text-blue-400">
  235. {spool.storage_location}
  236. </span>
  237. );
  238. },
  239. label_weight: ({ spool }) => (
  240. <span className="text-sm text-white">{formatWeight(spool.label_weight)}</span>
  241. ),
  242. net: ({ remaining }) => (
  243. <span className="text-sm text-white">{formatWeight(remaining)}</span>
  244. ),
  245. gross: ({ spool, remaining }) => (
  246. <span className="text-sm text-bambu-gray">{formatWeight(remaining + spool.core_weight)}</span>
  247. ),
  248. added_full: ({ spool }) => (
  249. <span className="text-sm text-bambu-gray">{spool.added_full == null ? '-' : spool.added_full ? 'Yes' : 'No'}</span>
  250. ),
  251. used: ({ spool }) => (
  252. <span className="text-sm text-bambu-gray">{spool.weight_used > 0 ? formatWeight(spool.weight_used) : '-'}</span>
  253. ),
  254. printed_total: () => (
  255. <span className="text-sm text-bambu-gray/50">-</span>
  256. ),
  257. printed_since_weight: () => (
  258. <span className="text-sm text-bambu-gray/50">-</span>
  259. ),
  260. note: ({ spool }) => (
  261. <span className="text-sm text-bambu-gray max-w-[150px] truncate block" title={spool.note || undefined}>{spool.note || '-'}</span>
  262. ),
  263. pa_k: ({ spool }) => {
  264. const count = spool.k_profiles?.length ?? 0;
  265. if (count === 0) return <span className="text-sm text-bambu-gray">-</span>;
  266. return (
  267. <span className="inline-flex items-center px-1.5 py-0.5 rounded text-xs font-medium bg-bambu-green/20 text-bambu-green">
  268. K
  269. </span>
  270. );
  271. },
  272. tag_id: ({ spool }) => {
  273. const tag = spool.tag_uid || spool.tray_uuid;
  274. if (!tag) return <span className="text-sm text-bambu-gray/50">-</span>;
  275. return (
  276. <span className="text-sm text-bambu-gray font-mono" title={tag}>
  277. {tag.length > 12 ? `${tag.slice(0, 6)}...${tag.slice(-4)}` : tag}
  278. </span>
  279. );
  280. },
  281. data_origin: ({ spool }) => (
  282. <span className="text-sm text-bambu-gray">{spool.data_origin || '-'}</span>
  283. ),
  284. tag_type: ({ spool }) => (
  285. <span className="text-sm text-bambu-gray">{spool.tag_type || '-'}</span>
  286. ),
  287. stock: ({ spool, t }) => {
  288. if (!spool.slicer_filament) {
  289. return (
  290. <span className="inline-flex items-center px-1.5 py-0.5 rounded text-xs font-medium bg-amber-500/20 text-amber-400">
  291. {t('inventory.stock')}
  292. </span>
  293. );
  294. }
  295. return <span className="text-sm text-bambu-gray">-</span>;
  296. },
  297. remaining: ({ remaining, pct }) => (
  298. <div className="flex items-center gap-2">
  299. <div className="flex-1 h-2 bg-bambu-dark-tertiary rounded-full overflow-hidden">
  300. <div
  301. className={`h-full rounded-full ${pct > 50 ? 'bg-bambu-green' : pct > 20 ? 'bg-yellow-500' : 'bg-red-500'}`}
  302. style={{ width: `${Math.min(pct, 100)}%` }}
  303. />
  304. </div>
  305. <span className="text-xs text-bambu-gray min-w-[40px] text-right">{Math.round(remaining)}g</span>
  306. </div>
  307. ),
  308. spool_name: ({ spool, catalogMap }) => {
  309. const entry = spool.core_weight_catalog_id != null ? catalogMap[spool.core_weight_catalog_id] : undefined;
  310. return <span className="text-sm text-bambu-gray">{entry?.name || '-'}</span>;
  311. },
  312. cost_per_kg: ({ spool, currencySymbol }) => (
  313. <span className="text-sm text-bambu-gray">
  314. {spool.cost_per_kg != null ? `${currencySymbol}${spool.cost_per_kg.toFixed(2)}` : '-'}
  315. </span>
  316. ),
  317. weight_check: ({ spool, onSyncWeight }) => {
  318. const scaleWeight = spool.last_scale_weight;
  319. if (scaleWeight == null) return <span className="text-sm text-bambu-gray/50" title="No scale measurement">-</span>;
  320. const coreWeight = spool.core_weight || 0;
  321. const calculatedWeight = Math.max(0, spool.label_weight - spool.weight_used) + coreWeight;
  322. // Edge case: scale < core_weight means spool is empty or not on scale — treat as match
  323. let difference: number;
  324. let isMatch: boolean;
  325. if (scaleWeight < coreWeight) {
  326. difference = scaleWeight - coreWeight;
  327. isMatch = true;
  328. } else {
  329. difference = scaleWeight - calculatedWeight;
  330. isMatch = Math.abs(difference) <= 50;
  331. }
  332. const diffStr = difference > 0 ? `+${Math.round(difference)}` : `${Math.round(difference)}`;
  333. const tooltip = isMatch
  334. ? `Scale: ${Math.round(scaleWeight)}g\nCalculated: ${Math.round(calculatedWeight)}g\nDifference: ${diffStr}g (within tolerance)`
  335. : `Scale: ${Math.round(scaleWeight)}g\nCalculated: ${Math.round(calculatedWeight)}g\nDifference: ${diffStr}g (mismatch!)`;
  336. return (
  337. <div
  338. className={`flex items-center gap-1 text-sm font-medium ${isMatch ? 'text-green-400' : 'text-yellow-400'}`}
  339. title={tooltip}
  340. >
  341. <span>{Math.round(scaleWeight)}g</span>
  342. {isMatch ? (
  343. <Check className="w-3 h-3" />
  344. ) : (
  345. <>
  346. <AlertTriangle className="w-3 h-3" />
  347. {onSyncWeight && (
  348. <button
  349. type="button"
  350. onClick={(e) => {
  351. e.stopPropagation();
  352. e.preventDefault();
  353. onSyncWeight(spool);
  354. }}
  355. className="p-1 hover:bg-bambu-green/20 rounded transition-colors text-bambu-green"
  356. title="Sync: trust scale weight and reset tracking"
  357. >
  358. <RefreshCw className="w-3.5 h-3.5" />
  359. </button>
  360. )}
  361. </>
  362. )}
  363. </div>
  364. );
  365. },
  366. };
  367. // Sort value extractors — return a comparable value for each sortable column
  368. const columnSortValues: Record<string, (spool: InventorySpool, assignmentMap: Record<number, LocationDisplay>) => string | number> = {
  369. id: (s) => s.id,
  370. added_time: (s) => s.created_at || '',
  371. encode_time: (s) => s.encode_time || '',
  372. last_used_time: (s) => s.last_used || '',
  373. material: (s) => (s.material || '').toLowerCase(),
  374. subtype: (s) => (s.subtype || '').toLowerCase(),
  375. color_name: (s) => (s.color_name || '').toLowerCase(),
  376. brand: (s) => (s.brand || '').toLowerCase(),
  377. slicer_filament: (s) => (s.slicer_filament_name || s.slicer_filament || '').toLowerCase(),
  378. location: (s, am) => {
  379. const a = am[s.id];
  380. if (!a) return '';
  381. const isExt = a.ams_id === 254 || a.ams_id === 255;
  382. const isHt = !isExt && a.ams_id >= 128;
  383. const label = a.ams_label ? ` (${a.ams_label})` : '';
  384. return `${a.printer_name || ''} ${formatSlotLabel(a.ams_id, a.tray_id, isHt, isExt)}${label}`;
  385. },
  386. storage_location: (s) => (s.storage_location || '').toLowerCase(),
  387. label_weight: (s) => s.label_weight,
  388. net: (s) => Math.max(0, s.label_weight - s.weight_used),
  389. gross: (s) => Math.max(0, s.label_weight - s.weight_used) + s.core_weight,
  390. used: (s) => s.weight_used,
  391. remaining: (s) => s.label_weight > 0 ? Math.max(0, s.label_weight - s.weight_used) / s.label_weight : 0,
  392. note: (s) => (s.note || '').toLowerCase(),
  393. data_origin: (s) => (s.data_origin || '').toLowerCase(),
  394. tag_type: (s) => (s.tag_type || '').toLowerCase(),
  395. stock: (s) => s.slicer_filament ? 1 : 0,
  396. spool_name: (s) => s.core_weight_catalog_id ?? 0,
  397. cost_per_kg: (s) => s.cost_per_kg ?? 0,
  398. weight_check: (s) => {
  399. if (s.last_scale_weight == null) return -1;
  400. const expectedGross = Math.max(0, s.label_weight - s.weight_used) + s.core_weight;
  401. return Math.abs(s.last_scale_weight - expectedGross);
  402. },
  403. };
  404. const SORT_STATE_KEY = 'bambuddy-inventory-sort';
  405. function loadSortState(): SortState {
  406. try {
  407. const stored = localStorage.getItem(SORT_STATE_KEY);
  408. if (stored) return JSON.parse(stored);
  409. } catch { /* ignore */ }
  410. return null;
  411. }
  412. function saveSortState(state: SortState) {
  413. try {
  414. if (state) {
  415. localStorage.setItem(SORT_STATE_KEY, JSON.stringify(state));
  416. } else {
  417. localStorage.removeItem(SORT_STATE_KEY);
  418. }
  419. } catch { /* ignore */ }
  420. }
  421. // Wrapper: detects Spoolman mode and passes it to the shared inventory UI
  422. export default function InventoryPageRouter() {
  423. const { data: spoolmanSettings } = useQuery({
  424. queryKey: ['spoolman-settings'],
  425. queryFn: api.getSpoolmanSettings,
  426. staleTime: 5 * 60 * 1000,
  427. });
  428. const spoolmanModeReady = spoolmanSettings !== undefined;
  429. const spoolmanMode =
  430. spoolmanSettings?.spoolman_enabled === 'true' && !!spoolmanSettings?.spoolman_url;
  431. return <InventoryPage spoolmanMode={spoolmanMode} spoolmanModeReady={spoolmanModeReady} />;
  432. }
  433. function InventoryPage({ spoolmanMode = false, spoolmanModeReady = true }: { spoolmanMode?: boolean; spoolmanModeReady?: boolean }) {
  434. const { t } = useTranslation();
  435. const queryClient = useQueryClient();
  436. const { showToast } = useToast();
  437. const { hasPermission, loading: authLoading } = useAuth();
  438. const canViewForecast = !authLoading && hasPermission('inventory:forecast_read');
  439. const [searchParams, setSearchParams] = useSearchParams();
  440. const [formModal, setFormModal] = useState<{ spool?: InventorySpool | null; mode: SpoolFormMode } | null>(null);
  441. const deepLinkHandled = useRef(false);
  442. const [confirmAction, setConfirmAction] = useState<
  443. | { type: 'delete' | 'archive' | 'reset-usage'; spoolId: number }
  444. | { type: 'reset-all-usage' }
  445. | null
  446. >(null);
  447. // Label printing (#809). null = closed; otherwise the IDs to print labels for.
  448. const [labelPickerSpoolIds, setLabelPickerSpoolIds] = useState<number[] | null>(null);
  449. // Filter state
  450. const [archiveFilter, setArchiveFilter] = useState<ArchiveFilter>('active');
  451. const [usageFilter, setUsageFilter] = useState<UsageFilter>('all');
  452. const [materialFilter, setMaterialFilter] = useState('');
  453. const [brandFilter, setBrandFilter] = useState('');
  454. const [categoryFilter, setCategoryFilter] = useState('');
  455. const [spoolFilter, setSpoolFilter] = useState('');
  456. const [stockFilter, setStockFilter] = useState<'all' | 'stock' | 'configured'>('all');
  457. // #1400: storage-location dropdown. Uses the sentinel `__none__` for the
  458. // "no storage location set" group, same pattern as the category filter so
  459. // users can find unfiled spools.
  460. const [storageLocationFilter, setStorageLocationFilter] = useState('');
  461. const [search, setSearch] = useState('');
  462. const [viewMode, setViewMode] = useState<ViewMode>('table');
  463. const [sortState, setSortState] = useState<SortState>(loadSortState);
  464. const [columnConfig, setColumnConfig] = useState<ColumnConfig[]>(loadColumnConfig);
  465. const [showColumnModal, setShowColumnModal] = useState(false);
  466. const [groupSimilar, setGroupSimilar] = useState(() => {
  467. try {
  468. return localStorage.getItem('bambuddy-inventory-group') === 'true';
  469. } catch { return false; }
  470. });
  471. const [expandedGroups, setExpandedGroups] = useState<Set<string>>(new Set());
  472. // Pagination state (pageSize persisted to localStorage)
  473. const [pageIndex, setPageIndex] = useState(0);
  474. const [pageSize, setPageSize] = useState(() => {
  475. try {
  476. const stored = localStorage.getItem('bambuddy-inventory-pageSize');
  477. if (stored) {
  478. const n = Number(stored);
  479. if ([15, 30, 50, 100, -1].includes(n)) return n;
  480. }
  481. } catch { /* ignore */ }
  482. return 15;
  483. });
  484. const { data: settings } = useQuery({
  485. queryKey: ['settings'],
  486. queryFn: api.getSettings,
  487. });
  488. const dateFormat: DateFormat = settings?.date_format || 'system';
  489. // Query key and fetch function differ based on data source
  490. const spoolsQueryKey = spoolmanMode ? ['spoolman-inventory-spools'] : ['inventory-spools'];
  491. const { data: spools, isLoading } = useQuery({
  492. queryKey: spoolsQueryKey,
  493. queryFn: () =>
  494. spoolmanMode ? api.getSpoolmanInventorySpools(true) : api.getSpools(true),
  495. refetchInterval: 30000,
  496. });
  497. // Deep-link: open edit modal for ?spool=<id>
  498. // Prefer the already-loaded spool list (no extra API call); fall back to a
  499. // targeted fetch for the rare case where the full list hasn't arrived yet.
  500. const _rawSpoolParam = searchParams.get('spool');
  501. // Only accept strings of digits representing a positive integer — guards against
  502. // NaN (Number('abc')), 0, negatives, and floats like '1.5' that would produce
  503. // an invalid path parameter and trigger unnecessary 422 responses from the API.
  504. const deepLinkSpoolId =
  505. _rawSpoolParam && /^\d+$/.test(_rawSpoolParam) && Number(_rawSpoolParam) > 0
  506. ? Number(_rawSpoolParam)
  507. : null;
  508. const deepLinkInList = spools?.find((s) => s.id === deepLinkSpoolId) ?? null;
  509. const clearDeepLinkParam = useCallback(() => {
  510. deepLinkHandled.current = true;
  511. setSearchParams((prev) => { prev.delete('spool'); return prev; }, { replace: true });
  512. }, [setSearchParams]);
  513. // Targeted fetch — only fires when mode is known and spool isn't in the list yet
  514. const { data: deepLinkSpool, isError: deepLinkFetchFailed, error: deepLinkError } = useQuery({
  515. queryKey: spoolmanMode
  516. ? ['spoolman-inventory-spool', deepLinkSpoolId]
  517. : ['inventory-spool', deepLinkSpoolId],
  518. queryFn: () =>
  519. spoolmanMode
  520. ? api.getSpoolmanInventorySpool(deepLinkSpoolId!)
  521. : api.getSpool(deepLinkSpoolId!),
  522. enabled: spoolmanModeReady && deepLinkSpoolId !== null && deepLinkInList === null,
  523. staleTime: Infinity,
  524. retry: (failureCount, error) =>
  525. failureCount < 2 && !(error instanceof ApiError && error.status === 404),
  526. });
  527. useEffect(() => {
  528. if (deepLinkHandled.current) return;
  529. // Case 1: spool is already in the fetched list
  530. if (spoolmanModeReady && deepLinkSpoolId && deepLinkInList) {
  531. clearDeepLinkParam();
  532. setFormModal({ spool: deepLinkInList, mode: 'edit' });
  533. return;
  534. }
  535. // Case 2: spool was fetched individually
  536. if (deepLinkSpool) {
  537. clearDeepLinkParam();
  538. setFormModal({ spool: deepLinkSpool, mode: 'edit' });
  539. return;
  540. }
  541. // Case 3: fetch failed
  542. if (deepLinkFetchFailed) {
  543. clearDeepLinkParam();
  544. const is404 = deepLinkError instanceof ApiError && deepLinkError.status === 404;
  545. showToast(t(is404 ? 'inventory.deepLinkSpoolNotFound' : 'inventory.deepLinkFetchFailed'), 'error');
  546. }
  547. }, [
  548. spoolmanModeReady,
  549. deepLinkSpoolId,
  550. deepLinkInList,
  551. deepLinkSpool,
  552. deepLinkFetchFailed,
  553. deepLinkError,
  554. clearDeepLinkParam,
  555. showToast,
  556. t,
  557. ]);
  558. const { data: assignments } = useQuery({
  559. queryKey: ['spool-assignments'],
  560. queryFn: () => api.getAssignments(),
  561. refetchInterval: 30000,
  562. });
  563. // Spoolman-mode slot assignments. spool.id IS the spoolman_spool_id, so this
  564. // feeds into the same assignmentMap that the LOCATION column reads.
  565. const {
  566. data: spoolmanSlotAssignments = [],
  567. isError: spoolmanSlotAssignmentsError,
  568. } = useQuery({
  569. queryKey: ['spoolman-slot-assignments-all'],
  570. queryFn: () => api.getSpoolmanSlotAssignments(),
  571. enabled: spoolmanMode,
  572. refetchInterval: 30000,
  573. staleTime: 10000,
  574. retry: 1,
  575. });
  576. // Surface a single toast when the slot-assignment endpoint goes down — the
  577. // LOCATION column would otherwise silently show "-" for every Spoolman spool.
  578. // useRef guard prevents repeated toasts during refetchInterval polls.
  579. const slotErrorToastShown = useRef(false);
  580. useEffect(() => {
  581. if (spoolmanSlotAssignmentsError && !slotErrorToastShown.current) {
  582. slotErrorToastShown.current = true;
  583. showToast(t('inventory.spoolmanUnreachable'), 'error');
  584. } else if (!spoolmanSlotAssignmentsError) {
  585. slotErrorToastShown.current = false;
  586. }
  587. }, [spoolmanSlotAssignmentsError, showToast, t]);
  588. const { data: catalogEntries } = useQuery({
  589. queryKey: ['spool-catalog'],
  590. queryFn: () => api.getSpoolCatalog(),
  591. });
  592. const deleteMutation = useMutation({
  593. mutationFn: (id: number) =>
  594. spoolmanMode ? api.deleteSpoolmanInventorySpool(id) : api.deleteSpool(id),
  595. onSuccess: () => {
  596. queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  597. showToast(t('inventory.spoolDeleted'), 'success');
  598. },
  599. onError: (error: Error) => {
  600. if (error instanceof ApiError && error.status === 404) {
  601. showToast(t('inventory.deleteSpoolNotFound'), 'error');
  602. } else if (error instanceof ApiError && error.status === 503) {
  603. showToast(t('inventory.spoolmanUnreachable'), 'error');
  604. } else {
  605. showToast(t('inventory.deleteFailed'), 'error');
  606. }
  607. },
  608. });
  609. const archiveMutation = useMutation({
  610. mutationFn: (id: number) =>
  611. spoolmanMode ? api.archiveSpoolmanInventorySpool(id) : api.archiveSpool(id),
  612. onSuccess: () => {
  613. queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  614. showToast(t('inventory.spoolArchived'), 'success');
  615. },
  616. onError: (error: Error) => {
  617. if (error instanceof ApiError && error.status === 404) {
  618. showToast(t('inventory.archiveSpoolNotFound'), 'error');
  619. } else if (error instanceof ApiError && error.status === 503) {
  620. showToast(t('inventory.spoolmanUnreachable'), 'error');
  621. } else {
  622. showToast(t('inventory.archiveFailed'), 'error');
  623. }
  624. },
  625. });
  626. const restoreMutation = useMutation({
  627. mutationFn: (id: number) =>
  628. spoolmanMode ? api.restoreSpoolmanInventorySpool(id) : api.restoreSpool(id),
  629. onSuccess: () => {
  630. queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  631. showToast(t('inventory.spoolRestored'), 'success');
  632. },
  633. onError: (error: Error) => {
  634. if (error instanceof ApiError && error.status === 404) {
  635. showToast(t('inventory.restoreSpoolNotFound'), 'error');
  636. } else if (error instanceof ApiError && error.status === 503) {
  637. showToast(t('inventory.spoolmanUnreachable'), 'error');
  638. } else {
  639. showToast(t('inventory.restoreFailed'), 'error');
  640. }
  641. },
  642. });
  643. const resetUsageMutation = useMutation({
  644. mutationFn: (id: number) =>
  645. spoolmanMode ? api.resetSpoolmanInventorySpoolUsage(id) : api.resetSpoolUsage(id),
  646. onSuccess: () => {
  647. queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  648. showToast(t('inventory.usageReset'), 'success');
  649. },
  650. onError: () => {
  651. showToast(t('inventory.resetUsageFailed'), 'error');
  652. },
  653. });
  654. const bulkResetUsageMutation = useMutation({
  655. mutationFn: (ids: number[]) =>
  656. spoolmanMode ? api.bulkResetSpoolmanInventorySpoolUsage(ids) : api.bulkResetSpoolUsage(ids),
  657. onSuccess: (data) => {
  658. queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  659. showToast(t('inventory.allUsageReset', { count: data.reset }), 'success');
  660. },
  661. onError: () => {
  662. showToast(t('inventory.resetUsageFailed'), 'error');
  663. },
  664. });
  665. // Spool IDs the "Reset all usage" button bulk-targets. Includes archived
  666. // spools too — without them, the broadened "Total Consumed" stat (which
  667. // sums archived consumption per the #1390 follow-up) would stay non-zero
  668. // after a Reset-all click, surprising the user. Backend reset endpoints
  669. // (both internal and Spoolman) already accept archived IDs without a
  670. // route-level guard, so this just removes the frontend filter.
  671. const resetableSpoolIds = useMemo(
  672. () => (spools ?? []).map((s) => s.id),
  673. [spools],
  674. );
  675. const handleSyncWeight = async (spool: InventorySpool) => {
  676. if (spool.last_scale_weight == null) return;
  677. try {
  678. if (spoolmanMode) {
  679. await api.syncSpoolmanSpoolWeight(spool.id, spool.last_scale_weight);
  680. } else {
  681. await spoolbuddyApi.updateSpoolWeight(spool.id, spool.last_scale_weight);
  682. }
  683. queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  684. const spoolName = [spool.brand, spool.material, spool.color_name].filter(Boolean).join(' ');
  685. showToast(`Synced "${spoolName}" to scale weight`, 'success');
  686. } catch (e) {
  687. const is404 = e instanceof ApiError && e.status === 404;
  688. const is503 = e instanceof ApiError && e.status === 503;
  689. if (is404) showToast(t('inventory.syncWeightSpoolNotFound'), 'error');
  690. else if (is503) showToast(t('inventory.syncWeightSpoolmanUnreachable'), 'error');
  691. else showToast(t('inventory.syncWeightFailed'), 'error');
  692. }
  693. };
  694. // Low stock threshold from backend settings
  695. const lowStockThreshold = settings?.low_stock_threshold ?? 20;
  696. const [showThresholdInput, setShowThresholdInput] = useState(false);
  697. const [thresholdInput, setThresholdInput] = useState(lowStockThreshold.toString());
  698. // Sync thresholdInput when lowStockThreshold changes and input is not shown
  699. useEffect(() => {
  700. if (!showThresholdInput) {
  701. setThresholdInput(lowStockThreshold.toString());
  702. }
  703. }, [lowStockThreshold, showThresholdInput]);
  704. const updateThresholdMutation = useMutation({
  705. mutationFn: (threshold: number) => api.updateSettings({ low_stock_threshold: threshold }),
  706. onSuccess: () => {
  707. queryClient.invalidateQueries({ queryKey: ['settings'] });
  708. showToast(t('common.saved'), 'success');
  709. setShowThresholdInput(false);
  710. },
  711. onError: () => {
  712. showToast(t('inventory.lowStockThresholdError'), 'error');
  713. },
  714. });
  715. // Stats calculation.
  716. //
  717. // "Total Consumed" sums over ALL spools (active AND archived) because it's
  718. // a running counter — past consumption of a now-archived spool is real
  719. // history and silently dropping it on archive made the running total
  720. // collapse mysteriously (#1390 follow-up). The other aggregates
  721. // (totalWeight, lowStock, byMaterial, activeCount) describe what's
  722. // currently in active inventory and stay active-only.
  723. const stats = useMemo(() => {
  724. if (!spools) return null;
  725. let totalWeight = 0;
  726. let totalConsumed = 0;
  727. let lowStock = 0;
  728. let activeCount = 0;
  729. const byMaterial: Record<string, { count: number; weight: number }> = {};
  730. for (const s of spools) {
  731. // "Total Consumed" is the resettable counter (weight_used - baseline)
  732. // rather than raw weight_used so the per-spool / bulk eraser zeroes
  733. // the stat without inflating remaining back to label_weight (#1390).
  734. // Computed before the archived-skip below so archived consumption
  735. // stays in the running total.
  736. totalConsumed += Math.max(0, s.weight_used - (s.weight_used_baseline ?? 0));
  737. if (s.archived_at) continue;
  738. activeCount++;
  739. const remaining = Math.max(0, s.label_weight - s.weight_used);
  740. totalWeight += remaining;
  741. const pct = s.label_weight > 0 ? (remaining / s.label_weight) * 100 : 0;
  742. const threshold = s.low_stock_threshold_pct ?? lowStockThreshold;
  743. if (pct < threshold) lowStock++;
  744. const mat = s.material || 'Unknown';
  745. if (!byMaterial[mat]) byMaterial[mat] = { count: 0, weight: 0 };
  746. byMaterial[mat].count++;
  747. byMaterial[mat].weight += remaining;
  748. }
  749. return { totalWeight, totalConsumed, lowStock, byMaterial, totalSpools: activeCount };
  750. }, [spools, lowStockThreshold]);
  751. const inPrinterCount =
  752. (assignments?.length ?? 0) + (spoolmanMode ? spoolmanSlotAssignments.length : 0);
  753. const currencySymbol = getCurrencySymbol(settings?.currency || 'USD');
  754. // Map spool_id -> location display data for the LOCATION column.
  755. // Local SpoolAssignment entries first, then Spoolman SlotAssignment fills in
  756. // remaining IDs. Local wins on collision (defensive — modes are exclusive in
  757. // practice, but a stray pair with the same numeric id would otherwise be
  758. // unpredictable). spool.id IS the spoolman_spool_id in Spoolman mode.
  759. const assignmentMap = useMemo<Record<number, LocationDisplay>>(() => {
  760. const map: Record<number, LocationDisplay> = {};
  761. for (const a of assignments || []) {
  762. map[a.spool_id] = {
  763. printer_id: a.printer_id,
  764. printer_name: a.printer_name,
  765. ams_id: a.ams_id,
  766. tray_id: a.tray_id,
  767. ams_label: a.ams_label ?? null,
  768. };
  769. }
  770. for (const a of spoolmanSlotAssignments) {
  771. // Defensive: skip malformed entries (missing or invalid spool id, ams id,
  772. // tray id). The Pydantic response model on the backend should already
  773. // reject these, but MITM proxies and stale CDN responses can drop fields.
  774. if (
  775. typeof a?.spoolman_spool_id !== 'number' ||
  776. a.spoolman_spool_id <= 0 ||
  777. typeof a.printer_id !== 'number' ||
  778. typeof a.ams_id !== 'number' ||
  779. typeof a.tray_id !== 'number'
  780. ) continue;
  781. if (!map[a.spoolman_spool_id]) {
  782. map[a.spoolman_spool_id] = {
  783. printer_id: a.printer_id,
  784. printer_name: a.printer_name ?? null,
  785. ams_id: a.ams_id,
  786. tray_id: a.tray_id,
  787. ams_label: a.ams_label ?? null,
  788. };
  789. }
  790. }
  791. return map;
  792. }, [assignments, spoolmanSlotAssignments]);
  793. // Map catalog_id -> catalog entry for spool name column
  794. const catalogMap = useMemo(() => {
  795. const map: Record<number, SpoolCatalogEntry> = {};
  796. for (const e of catalogEntries || []) {
  797. map[e.id] = e;
  798. }
  799. return map;
  800. }, [catalogEntries]);
  801. // Top materials by weight for stat card pills
  802. const topMaterials = useMemo(() => {
  803. if (!stats) return [];
  804. return Object.entries(stats.byMaterial)
  805. .sort((a, b) => b[1].weight - a[1].weight)
  806. .slice(0, 4);
  807. }, [stats]);
  808. // Filtering pipeline
  809. const filteredSpools = useMemo(() => {
  810. let filtered = spools || [];
  811. // Archive filter
  812. if (archiveFilter === 'active') {
  813. filtered = filtered.filter((s) => !s.archived_at);
  814. } else {
  815. filtered = filtered.filter((s) => !!s.archived_at);
  816. }
  817. // Usage filter
  818. if (usageFilter === 'used') {
  819. filtered = filtered.filter((s) => s.weight_used > 0);
  820. } else if (usageFilter === 'new') {
  821. filtered = filtered.filter((s) => s.weight_used === 0);
  822. } else if (usageFilter === 'lowstock') {
  823. filtered = filtered.filter((s) => {
  824. const remaining = Math.max(0, s.label_weight - s.weight_used);
  825. const pct = s.label_weight > 0 ? (remaining / s.label_weight) * 100 : 0;
  826. const threshold = s.low_stock_threshold_pct ?? lowStockThreshold;
  827. return pct < threshold;
  828. });
  829. }
  830. // Material dropdown
  831. if (materialFilter) {
  832. filtered = filtered.filter((s) => s.material === materialFilter);
  833. }
  834. // Brand dropdown
  835. if (brandFilter) {
  836. filtered = filtered.filter((s) => s.brand === brandFilter);
  837. }
  838. // Category dropdown (#729)
  839. if (categoryFilter) {
  840. if (categoryFilter === '__none__') {
  841. filtered = filtered.filter((s) => !s.category);
  842. } else {
  843. filtered = filtered.filter((s) => s.category === categoryFilter);
  844. }
  845. }
  846. // Spool name dropdown
  847. if (spoolFilter) {
  848. const catalogId = Number(spoolFilter);
  849. filtered = filtered.filter((s) => s.core_weight_catalog_id === catalogId);
  850. }
  851. // Storage location dropdown (#1400). `__none__` lets the user find
  852. // spools that haven't been assigned a storage location yet.
  853. if (storageLocationFilter) {
  854. if (storageLocationFilter === '__none__') {
  855. filtered = filtered.filter((s) => !s.storage_location?.trim());
  856. } else {
  857. filtered = filtered.filter((s) => s.storage_location?.trim() === storageLocationFilter);
  858. }
  859. }
  860. // Stock filter
  861. if (stockFilter === 'stock') {
  862. filtered = filtered.filter((s) => !s.slicer_filament);
  863. } else if (stockFilter === 'configured') {
  864. filtered = filtered.filter((s) => !!s.slicer_filament);
  865. }
  866. // Global search
  867. if (search) {
  868. filtered = filterSpoolsByQuery(filtered, search);
  869. }
  870. return filtered;
  871. }, [spools, archiveFilter, usageFilter, materialFilter, brandFilter, categoryFilter, spoolFilter, stockFilter, storageLocationFilter, search, lowStockThreshold]);
  872. // Reset page on filter changes
  873. const resetPage = () => setPageIndex(0);
  874. // Unique values for filter dropdowns
  875. const uniqueMaterials = [...new Set(spools?.map((s) => s.material) || [])].sort();
  876. const uniqueBrands = [...new Set(spools?.map((s) => s.brand).filter(Boolean) || [])].sort() as string[];
  877. const uniqueCategories = [...new Set(spools?.map((s) => s.category?.trim()).filter(Boolean) as string[] || [])].sort();
  878. const hasUncategorized = (spools ?? []).some((s) => !s.category);
  879. const uniqueSpoolCatalogIds = [...new Set(spools?.map((s) => s.core_weight_catalog_id).filter((id): id is number => id != null) || [])].sort((a, b) => {
  880. const nameA = (catalogMap[a]?.name || '').toLowerCase();
  881. const nameB = (catalogMap[b]?.name || '').toLowerCase();
  882. return nameA.localeCompare(nameB);
  883. });
  884. // #1400: storage-location distinct values. `.trim()` so accidental
  885. // trailing whitespace doesn't show up as a separate option.
  886. const uniqueStorageLocations = [...new Set(spools?.map((s) => s.storage_location?.trim()).filter(Boolean) as string[] || [])].sort();
  887. const hasUnsetStorageLocation = (spools ?? []).some((s) => !s.storage_location?.trim());
  888. // Check if any filters are non-default
  889. const hasActiveFilters = archiveFilter !== 'active' || usageFilter !== 'all' || !!materialFilter || !!brandFilter || !!categoryFilter || !!spoolFilter || !!storageLocationFilter || stockFilter !== 'all' || !!search;
  890. const handleColumnConfigSave = (config: ColumnConfig[]) => {
  891. setColumnConfig(config);
  892. saveColumnConfig(config);
  893. };
  894. // Visible column IDs in order
  895. const visibleColumns = useMemo(
  896. () => columnConfig.filter((c) => c.visible).map((c) => c.id),
  897. [columnConfig]
  898. );
  899. const handleSort = (colId: string) => {
  900. if (!columnSortValues[colId]) return; // Not sortable
  901. setSortState((prev) => {
  902. let next: SortState;
  903. if (prev?.column === colId) {
  904. // Toggle direction, or clear on third click
  905. next = prev.direction === 'asc' ? { column: colId, direction: 'desc' } : null;
  906. } else {
  907. next = { column: colId, direction: 'asc' };
  908. }
  909. saveSortState(next);
  910. return next;
  911. });
  912. resetPage();
  913. };
  914. // Sort filtered spools
  915. const sortedSpools = useMemo(() => {
  916. if (!sortState) return filteredSpools;
  917. const extractor = columnSortValues[sortState.column];
  918. if (!extractor) return filteredSpools;
  919. const sorted = [...filteredSpools].sort((a, b) => {
  920. const va = extractor(a, assignmentMap);
  921. const vb = extractor(b, assignmentMap);
  922. if (va < vb) return sortState.direction === 'asc' ? -1 : 1;
  923. if (va > vb) return sortState.direction === 'asc' ? 1 : -1;
  924. return 0;
  925. });
  926. return sorted;
  927. }, [filteredSpools, sortState, assignmentMap]);
  928. // Group similar spools when toggle is active
  929. const displayItems = useMemo((): DisplayItem[] => {
  930. if (!groupSimilar) return sortedSpools.map((s) => ({ type: 'single' as const, spool: s }));
  931. const groups = new Map<string, InventorySpool[]>();
  932. for (const spool of sortedSpools) {
  933. // Only group unused & unassigned spools
  934. if (spool.weight_used > 0 || assignmentMap[spool.id]) {
  935. // Will be added as singles in the walk below
  936. } else {
  937. const key = spoolGroupKey(spool);
  938. const arr = groups.get(key);
  939. if (arr) arr.push(spool);
  940. else groups.set(key, [spool]);
  941. }
  942. }
  943. const items: DisplayItem[] = [];
  944. const processedKeys = new Set<string>();
  945. // Walk sortedSpools order so groups appear at the position of their first member
  946. for (const spool of sortedSpools) {
  947. if (spool.weight_used > 0 || assignmentMap[spool.id]) {
  948. items.push({ type: 'single', spool });
  949. continue;
  950. }
  951. const key = spoolGroupKey(spool);
  952. if (processedKeys.has(key)) continue;
  953. processedKeys.add(key);
  954. const members = groups.get(key)!;
  955. if (members.length === 1) {
  956. items.push({ type: 'single', spool: members[0] });
  957. } else {
  958. items.push({ type: 'group', key, spools: members, representative: members[0] });
  959. }
  960. }
  961. return items;
  962. }, [sortedSpools, groupSimilar, assignmentMap]);
  963. // Pagination (after sorting) — pageSize -1 means "All"
  964. const showAll = pageSize === -1;
  965. const totalDisplayItems = displayItems.length;
  966. const effectivePageSize = showAll ? totalDisplayItems || 1 : pageSize;
  967. const totalPages = Math.max(1, Math.ceil(totalDisplayItems / effectivePageSize));
  968. const safePageIndex = showAll ? 0 : Math.min(pageIndex, totalPages - 1);
  969. const pagedItems = showAll
  970. ? displayItems
  971. : displayItems.slice(safePageIndex * effectivePageSize, (safePageIndex + 1) * effectivePageSize);
  972. const toggleGroupSimilar = () => {
  973. const next = !groupSimilar;
  974. setGroupSimilar(next);
  975. setExpandedGroups(new Set());
  976. resetPage();
  977. try { localStorage.setItem('bambuddy-inventory-group', String(next)); } catch { /* ignore */ }
  978. };
  979. const toggleGroupExpand = (key: string) => {
  980. setExpandedGroups((prev) => {
  981. const next = new Set(prev);
  982. if (next.has(key)) next.delete(key);
  983. else next.add(key);
  984. return next;
  985. });
  986. };
  987. const handlePageSizeChange = (size: number) => {
  988. setPageSize(size);
  989. setPageIndex(0);
  990. try { localStorage.setItem('bambuddy-inventory-pageSize', String(size)); } catch { /* ignore */ }
  991. };
  992. const clearAllFilters = () => {
  993. setArchiveFilter('active');
  994. setUsageFilter('all');
  995. setMaterialFilter('');
  996. setBrandFilter('');
  997. setCategoryFilter('');
  998. setSpoolFilter('');
  999. setStorageLocationFilter('');
  1000. setStockFilter('all');
  1001. setSearch('');
  1002. resetPage();
  1003. };
  1004. return (
  1005. <div className="p-4 md:p-8 space-y-6">
  1006. {/* Header */}
  1007. <div className="flex items-center justify-between">
  1008. <div>
  1009. <h1 className="text-2xl font-bold text-white flex items-center gap-3">
  1010. <Package className="w-7 h-7 text-bambu-green" />
  1011. {t('inventory.title')}
  1012. </h1>
  1013. <p className="text-bambu-gray mt-1">{t('inventory.subtitle')}</p>
  1014. </div>
  1015. <div className="flex items-center gap-2">
  1016. <Button
  1017. variant="secondary"
  1018. disabled={filteredSpools.length === 0}
  1019. // Pre-select every visible spool so the user lands in "all
  1020. // checked", then refines downward in the modal. Per-card icon
  1021. // pre-selects only that spool — both flows share the same picker.
  1022. onClick={() => setLabelPickerSpoolIds(filteredSpools.map((s) => s.id))}
  1023. title={
  1024. filteredSpools.length === 0
  1025. ? t('inventory.labels.noSpoolsTitle', 'No spools to label')
  1026. : t('inventory.labels.bulkTitle', 'Pick spools to print labels for from the {{count}} currently shown', { count: filteredSpools.length })
  1027. }
  1028. >
  1029. <Printer className="w-4 h-4" />
  1030. {t('inventory.labels.printLabels', 'Print labels…')}
  1031. </Button>
  1032. <Button onClick={() => setFormModal({ spool: null, mode: 'create' })}>
  1033. <Plus className="w-4 h-4" />
  1034. {t('inventory.addSpool')}
  1035. </Button>
  1036. </div>
  1037. </div>
  1038. {/* Stats Bar */}
  1039. {stats && !isLoading && (
  1040. <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-5 gap-3">
  1041. {/* Total Inventory */}
  1042. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  1043. <div className="flex items-center gap-2 mb-1">
  1044. <Package className="w-4 h-4 text-bambu-green" />
  1045. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.totalInventory')}</span>
  1046. </div>
  1047. <div className="text-xl font-bold text-white">{formatWeight(stats.totalWeight, true)}</div>
  1048. <div className="text-xs text-bambu-gray mt-1">{stats.totalSpools} {stats.totalSpools !== 1 ? t('inventory.spools') : t('inventory.spool')}</div>
  1049. </div>
  1050. {/* Total Consumed */}
  1051. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  1052. <div className="flex items-center justify-between gap-2 mb-1">
  1053. <div className="flex items-center gap-2">
  1054. <TrendingDown className="w-4 h-4 text-blue-400" />
  1055. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.totalConsumed')}</span>
  1056. </div>
  1057. {stats.totalConsumed > 0 && resetableSpoolIds.length > 0 && (
  1058. <button
  1059. onClick={() => setConfirmAction({ type: 'reset-all-usage' })}
  1060. className="p-1 text-bambu-gray hover:text-red-400 rounded transition-colors"
  1061. title={t('inventory.resetAllUsageTooltip')}
  1062. aria-label={t('inventory.resetAllUsage')}
  1063. >
  1064. <Eraser className="w-3.5 h-3.5" />
  1065. </button>
  1066. )}
  1067. </div>
  1068. <div className="text-xl font-bold text-white">{formatWeight(stats.totalConsumed, true)}</div>
  1069. <div className="text-xs text-bambu-gray mt-1">{t('inventory.sinceTracking')}</div>
  1070. </div>
  1071. {/* By Material */}
  1072. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  1073. <div className="flex items-center gap-2 mb-1">
  1074. <Layers className="w-4 h-4 text-green-400" />
  1075. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.byMaterial')}</span>
  1076. </div>
  1077. <div className="flex flex-wrap gap-1.5 mt-1">
  1078. {topMaterials.map(([mat, data]) => (
  1079. <span
  1080. key={mat}
  1081. className={`inline-flex items-center gap-1 px-2 py-0.5 rounded-full text-xs font-medium ${MATERIAL_COLORS[mat] || 'bg-bambu-dark-tertiary text-bambu-gray'}`}
  1082. >
  1083. {mat} <span className="opacity-70">{formatWeight(data.weight, true)}</span>
  1084. </span>
  1085. ))}
  1086. </div>
  1087. </div>
  1088. {/* In Printer */}
  1089. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  1090. <div className="flex items-center gap-2 mb-1">
  1091. <Printer className="w-4 h-4 text-purple-400" />
  1092. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.inPrinter')}</span>
  1093. </div>
  1094. <div className="text-xl font-bold text-white">{inPrinterCount}</div>
  1095. <div className="text-xs text-bambu-gray mt-1">{t('inventory.loadedInAms')}</div>
  1096. </div>
  1097. {/* Low Stock */}
  1098. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  1099. <div className="flex items-center gap-2 mb-1">
  1100. <AlertTriangle className="w-4 h-4 text-yellow-400" />
  1101. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.lowStock')}</span>
  1102. </div>
  1103. <div className={`text-xl font-bold ${stats.lowStock > 0 ? 'text-yellow-400' : 'text-white'}`}>{stats.lowStock}</div>
  1104. <div className="text-xs text-bambu-gray mt-1 flex items-center gap-2">
  1105. {showThresholdInput ? (
  1106. <form
  1107. onSubmit={e => {
  1108. e.preventDefault();
  1109. const val = parseFloat(thresholdInput);
  1110. if (!isNaN(val) && val >= 0.1 && val <= 99.9) {
  1111. updateThresholdMutation.mutate(val);
  1112. } else {
  1113. showToast(t('inventory.lowStockThresholdError'), 'error');
  1114. }
  1115. }}
  1116. className="flex items-center gap-2"
  1117. >
  1118. <span className="text-xs text-bambu-gray">{'<'}</span>
  1119. <input
  1120. type="text"
  1121. inputMode="decimal"
  1122. pattern="^\d{0,2}(\.\d?)?$"
  1123. maxLength={4}
  1124. value={thresholdInput}
  1125. onChange={e => {
  1126. // Only allow up to 2 digits before decimal and 1 after
  1127. const val = e.target.value.replace(/[^\d.]/g, '');
  1128. if (/^\d{0,2}(\.\d?)?$/.test(val)) {
  1129. setThresholdInput(val);
  1130. }
  1131. }}
  1132. className="px-1.5 py-1 rounded border border-bambu-dark-tertiary text-xs text-white bg-bambu-dark-secondary focus:outline-none focus:border-bambu-green w-14 text-center"
  1133. onWheel={e => e.currentTarget.blur()}
  1134. disabled={updateThresholdMutation.isPending}
  1135. />
  1136. <span className="text-xs text-bambu-gray">%</span>
  1137. <Button type="submit" size="sm" disabled={updateThresholdMutation.isPending}>{t('common.save')}</Button>
  1138. <Button type="button" size="sm" variant="ghost" onClick={() => setShowThresholdInput(false)} disabled={updateThresholdMutation.isPending}>{t('common.cancel')}</Button>
  1139. </form>
  1140. ) : (
  1141. <>
  1142. <span className="text-bambu-gray">{'< '}{lowStockThreshold}%</span>
  1143. <button
  1144. className="p-1.5 text-bambu-gray hover:text-white rounded transition-colors"
  1145. title={t('common.edit')}
  1146. onClick={() => {
  1147. setThresholdInput(lowStockThreshold.toString());
  1148. setShowThresholdInput(true);
  1149. }}
  1150. >
  1151. <Edit2 className="w-4 h-4" />
  1152. </button>
  1153. </>
  1154. )}
  1155. </div>
  1156. </div>
  1157. </div>
  1158. )}
  1159. {/* Toolbar: Search + View toggle */}
  1160. <div className="flex flex-col sm:flex-row gap-3 items-start sm:items-center justify-between">
  1161. <div className={`relative flex-1 max-w-md ${viewMode === 'forecast' ? 'invisible pointer-events-none' : ''}`}>
  1162. <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray/50" />
  1163. <input
  1164. type="text"
  1165. value={search}
  1166. onChange={(e) => { setSearch(e.target.value); resetPage(); }}
  1167. placeholder={t('inventory.search')}
  1168. className="w-full pl-10 pr-8 py-2 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-lg text-white text-sm placeholder:text-bambu-gray/50 focus:outline-none focus:border-bambu-green"
  1169. />
  1170. {search && (
  1171. <button
  1172. onClick={() => { setSearch(''); resetPage(); }}
  1173. className="absolute right-3 top-1/2 -translate-y-1/2 text-bambu-gray hover:text-white"
  1174. >
  1175. <X className="w-4 h-4" />
  1176. </button>
  1177. )}
  1178. </div>
  1179. <div className="flex items-center gap-2">
  1180. {/* Columns button (table view only) */}
  1181. {viewMode === 'table' && (
  1182. <button
  1183. onClick={() => setShowColumnModal(true)}
  1184. className="flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium text-bambu-gray border border-bambu-dark-tertiary rounded-lg hover:bg-bambu-dark-tertiary transition-colors"
  1185. title={t('inventory.configureColumns')}
  1186. >
  1187. <Columns className="w-4 h-4" />
  1188. <span className="hidden sm:inline">{t('inventory.columns')}</span>
  1189. </button>
  1190. )}
  1191. {/* Group similar toggle — hidden in forecast mode */}
  1192. {viewMode !== 'forecast' && (
  1193. <button
  1194. onClick={toggleGroupSimilar}
  1195. className={`flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium border rounded-lg transition-colors ${
  1196. groupSimilar
  1197. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  1198. : 'text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  1199. }`}
  1200. title={t('inventory.groupSimilar')}
  1201. >
  1202. <Group className="w-4 h-4" />
  1203. <span className="hidden sm:inline">{t('inventory.groupSimilar')}</span>
  1204. </button>
  1205. )}
  1206. {/* Table / Cards toggle */}
  1207. <div className="flex bg-bambu-dark-primary border border-bambu-dark-tertiary rounded-lg overflow-hidden">
  1208. <button
  1209. onClick={() => setViewMode('table')}
  1210. className={`flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium transition-colors ${
  1211. viewMode === 'table'
  1212. ? 'bg-bambu-green text-white'
  1213. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1214. }`}
  1215. >
  1216. <TableProperties className="w-4 h-4" />
  1217. <span className="hidden sm:inline">{t('inventory.table')}</span>
  1218. </button>
  1219. <button
  1220. onClick={() => setViewMode('cards')}
  1221. className={`flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium transition-colors ${
  1222. viewMode === 'cards'
  1223. ? 'bg-bambu-green text-white'
  1224. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1225. }`}
  1226. >
  1227. <LayoutGrid className="w-4 h-4" />
  1228. <span className="hidden sm:inline">{t('inventory.cards')}</span>
  1229. </button>
  1230. <button
  1231. onClick={() => canViewForecast && setViewMode('forecast')}
  1232. disabled={!canViewForecast}
  1233. title={canViewForecast ? undefined : t('forecast.noReadAccess')}
  1234. className={`flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium transition-colors disabled:opacity-40 disabled:cursor-not-allowed ${
  1235. viewMode === 'forecast'
  1236. ? 'bg-bambu-green text-white'
  1237. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1238. }`}
  1239. >
  1240. {canViewForecast ? <TrendingUp className="w-4 h-4" /> : <Lock className="w-4 h-4" />}
  1241. <span className="hidden sm:inline">{t('forecast.title')}</span>
  1242. </button>
  1243. </div>
  1244. </div>
  1245. </div>
  1246. {/* Filter chips row — hidden in forecast mode */}
  1247. <div className={`flex flex-wrap items-center gap-2 ${viewMode === 'forecast' ? 'hidden' : ''}`}>
  1248. {/* Active / Archived chips */}
  1249. <div className="flex items-center rounded-lg border border-bambu-dark-tertiary overflow-hidden">
  1250. <button
  1251. onClick={() => { setArchiveFilter('active'); resetPage(); }}
  1252. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  1253. archiveFilter === 'active'
  1254. ? 'bg-bambu-green/20 text-bambu-green'
  1255. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1256. }`}
  1257. >
  1258. <Package className="w-3.5 h-3.5" />
  1259. {t('inventory.active')}
  1260. </button>
  1261. <button
  1262. onClick={() => { setArchiveFilter('archived'); resetPage(); }}
  1263. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  1264. archiveFilter === 'archived'
  1265. ? 'bg-bambu-green/20 text-bambu-green'
  1266. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1267. }`}
  1268. >
  1269. <Archive className="w-3.5 h-3.5" />
  1270. {t('inventory.archived')}
  1271. </button>
  1272. </div>
  1273. <div className="w-px h-5 bg-bambu-dark-tertiary" />
  1274. {/* All / Used / New chips */}
  1275. <div className="flex items-center rounded-lg border border-bambu-dark-tertiary overflow-hidden">
  1276. <button
  1277. onClick={() => { setUsageFilter('all'); resetPage(); }}
  1278. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  1279. usageFilter === 'all'
  1280. ? 'bg-bambu-green/20 text-bambu-green'
  1281. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1282. }`}
  1283. >
  1284. {t('inventory.all')}
  1285. </button>
  1286. <button
  1287. onClick={() => { setUsageFilter('used'); resetPage(); }}
  1288. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  1289. usageFilter === 'used'
  1290. ? 'bg-bambu-green/20 text-bambu-green'
  1291. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1292. }`}
  1293. >
  1294. <Clock className="w-3.5 h-3.5" />
  1295. {t('inventory.used')}
  1296. </button>
  1297. <button
  1298. onClick={() => { setUsageFilter('new'); resetPage(); }}
  1299. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  1300. usageFilter === 'new'
  1301. ? 'bg-bambu-green/20 text-bambu-green'
  1302. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1303. }`}
  1304. >
  1305. {t('inventory.new')}
  1306. </button>
  1307. <button
  1308. onClick={() => { setUsageFilter('lowstock'); resetPage(); }}
  1309. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  1310. usageFilter === 'lowstock'
  1311. ? 'bg-yellow-500/20 text-yellow-400'
  1312. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1313. }`}
  1314. >
  1315. <AlertTriangle className="w-3.5 h-3.5" />
  1316. {t('inventory.lowStock')}
  1317. </button>
  1318. </div>
  1319. {/* Stock filter chips */}
  1320. <div className="flex items-center rounded-lg border border-bambu-dark-tertiary overflow-hidden">
  1321. <button
  1322. onClick={() => { setStockFilter('all'); resetPage(); }}
  1323. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  1324. stockFilter === 'all'
  1325. ? 'bg-bambu-green/20 text-bambu-green'
  1326. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1327. }`}
  1328. >
  1329. {t('inventory.all')}
  1330. </button>
  1331. <button
  1332. onClick={() => { setStockFilter('stock'); resetPage(); }}
  1333. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  1334. stockFilter === 'stock'
  1335. ? 'bg-amber-500/20 text-amber-400'
  1336. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1337. }`}
  1338. >
  1339. {t('inventory.stock')}
  1340. </button>
  1341. <button
  1342. onClick={() => { setStockFilter('configured'); resetPage(); }}
  1343. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  1344. stockFilter === 'configured'
  1345. ? 'bg-bambu-green/20 text-bambu-green'
  1346. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  1347. }`}
  1348. >
  1349. {t('inventory.configured')}
  1350. </button>
  1351. </div>
  1352. <div className="w-px h-5 bg-bambu-dark-tertiary" />
  1353. {/* Material dropdown chip */}
  1354. <select
  1355. value={materialFilter}
  1356. onChange={(e) => { setMaterialFilter(e.target.value); resetPage(); }}
  1357. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  1358. materialFilter
  1359. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  1360. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  1361. }`}
  1362. >
  1363. <option value="">{t('inventory.material')}</option>
  1364. {uniqueMaterials.map((m) => (
  1365. <option key={m} value={m}>{m}</option>
  1366. ))}
  1367. </select>
  1368. {/* Brand dropdown chip */}
  1369. <select
  1370. value={brandFilter}
  1371. onChange={(e) => { setBrandFilter(e.target.value); resetPage(); }}
  1372. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  1373. brandFilter
  1374. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  1375. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  1376. }`}
  1377. >
  1378. <option value="">{t('inventory.brand')}</option>
  1379. {uniqueBrands.map((b) => (
  1380. <option key={b} value={b}>{b}</option>
  1381. ))}
  1382. </select>
  1383. {/* Category dropdown chip (#729) — only render once at least one
  1384. spool carries a category, otherwise it's noise. */}
  1385. {(uniqueCategories.length > 0 || categoryFilter) && (
  1386. <select
  1387. value={categoryFilter}
  1388. onChange={(e) => { setCategoryFilter(e.target.value); resetPage(); }}
  1389. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  1390. categoryFilter
  1391. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  1392. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  1393. }`}
  1394. >
  1395. <option value="">{t('inventory.category')}</option>
  1396. {uniqueCategories.map((c) => (
  1397. <option key={c} value={c}>{c}</option>
  1398. ))}
  1399. {hasUncategorized && (
  1400. <option value="__none__">{t('inventory.categoryNone')}</option>
  1401. )}
  1402. </select>
  1403. )}
  1404. {/* Spool name dropdown chip */}
  1405. {uniqueSpoolCatalogIds.length > 0 && (
  1406. <select
  1407. value={spoolFilter}
  1408. onChange={(e) => { setSpoolFilter(e.target.value); resetPage(); }}
  1409. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  1410. spoolFilter
  1411. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  1412. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  1413. }`}
  1414. >
  1415. <option value="">{t('inventory.spoolName')}</option>
  1416. {uniqueSpoolCatalogIds.map((id) => (
  1417. <option key={id} value={id}>{catalogMap[id]?.name || `#${id}`}</option>
  1418. ))}
  1419. </select>
  1420. )}
  1421. {/* Storage location dropdown chip (#1400) — only render when at
  1422. least one spool carries a storage location, otherwise it's noise
  1423. (matches the category chip pattern). */}
  1424. {(uniqueStorageLocations.length > 0 || storageLocationFilter) && (
  1425. <select
  1426. value={storageLocationFilter}
  1427. onChange={(e) => { setStorageLocationFilter(e.target.value); resetPage(); }}
  1428. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  1429. storageLocationFilter
  1430. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  1431. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  1432. }`}
  1433. >
  1434. <option value="">{t('inventory.storageLocation')}</option>
  1435. {uniqueStorageLocations.map((loc) => (
  1436. <option key={loc} value={loc}>{loc}</option>
  1437. ))}
  1438. {hasUnsetStorageLocation && (
  1439. <option value="__none__">{t('inventory.storageLocationNone')}</option>
  1440. )}
  1441. </select>
  1442. )}
  1443. {/* Clear filters */}
  1444. {hasActiveFilters && (
  1445. <>
  1446. <div className="w-px h-5 bg-bambu-dark-tertiary" />
  1447. <button
  1448. onClick={clearAllFilters}
  1449. className="flex items-center gap-1 text-xs text-bambu-gray hover:text-bambu-green transition-colors"
  1450. >
  1451. <X className="w-3.5 h-3.5" />
  1452. {t('inventory.clearFilters')}
  1453. </button>
  1454. </>
  1455. )}
  1456. {/* Results count — hidden in forecast mode */}
  1457. {viewMode !== 'forecast' && (
  1458. <span className="ml-auto text-xs text-bambu-gray">
  1459. {sortedSpools.length} {sortedSpools.length !== 1 ? t('inventory.spools') : t('inventory.spool')}
  1460. {groupSimilar && totalDisplayItems < sortedSpools.length && ` (${totalDisplayItems} ${t('inventory.groupedRows')})`}
  1461. </span>
  1462. )}
  1463. </div>
  1464. {/* Content */}
  1465. {isLoading ? (
  1466. <div className="flex justify-center py-16">
  1467. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  1468. </div>
  1469. ) : viewMode === 'forecast' ? (
  1470. /* Forecast view */
  1471. <ForecastPanel spools={spools || []} />
  1472. ) : viewMode === 'cards' ? (
  1473. /* Cards view */
  1474. pagedItems.length > 0 ? (
  1475. <>
  1476. <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4">
  1477. {pagedItems.map((item) => {
  1478. if (item.type === 'group') {
  1479. const { key, spools: groupSpools, representative: rep } = item;
  1480. // Total remaining filament across the group (#1368) — the
  1481. // headline number for the collapsed card, vs one member's.
  1482. const groupRemaining = groupSpools.reduce(
  1483. (sum, s) => sum + Math.max(0, s.label_weight - s.weight_used),
  1484. 0,
  1485. );
  1486. const groupBannerStyle = buildFilamentBackground({
  1487. rgba: rep.rgba,
  1488. extraColors: rep.extra_colors,
  1489. effectType: rep.effect_type,
  1490. subtype: rep.subtype,
  1491. effectSize: 'groupheader',
  1492. });
  1493. const isExpanded = expandedGroups.has(key);
  1494. return (
  1495. <div key={`group-${key}`} className="col-span-full">
  1496. {/* Group header card */}
  1497. <div
  1498. className="bg-bambu-dark-secondary rounded-lg overflow-hidden border border-bambu-green/30 hover:border-bambu-green transition-colors cursor-pointer"
  1499. onClick={() => toggleGroupExpand(key)}
  1500. >
  1501. <div className="h-10 flex items-center px-4 gap-3" style={groupBannerStyle}>
  1502. <span className="bg-white/90 text-gray-800 px-3 py-0.5 rounded-full text-sm font-medium">
  1503. {resolveSpoolColorName(rep.color_name, rep.rgba) || '-'}
  1504. </span>
  1505. </div>
  1506. <div className="px-4 py-3 flex items-center justify-between">
  1507. <div className="flex items-center gap-3">
  1508. <ChevronDown className={`w-4 h-4 text-bambu-gray transition-transform ${isExpanded ? '' : '-rotate-90'}`} />
  1509. <div>
  1510. <h3 className="font-semibold text-white">{rep.material}{rep.subtype ? ` ${rep.subtype}` : ''}</h3>
  1511. <p className="text-sm text-bambu-gray">{rep.brand || '-'}</p>
  1512. </div>
  1513. </div>
  1514. <div className="flex items-center gap-2">
  1515. <span className="text-sm text-bambu-gray" title={t('inventory.remaining')}>
  1516. {formatWeight(groupRemaining)}
  1517. </span>
  1518. <span className="text-xs font-medium bg-bambu-green/20 text-bambu-green px-2 py-0.5 rounded-full">
  1519. {t('inventory.groupedSpools', { count: groupSpools.length })}
  1520. </span>
  1521. </div>
  1522. </div>
  1523. </div>
  1524. {/* Expanded individual spools */}
  1525. {isExpanded && (
  1526. <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4 mt-2 ml-4">
  1527. {groupSpools.map((spool) => {
  1528. const remaining = Math.max(0, spool.label_weight - spool.weight_used);
  1529. const pct = spool.label_weight > 0 ? (remaining / spool.label_weight) * 100 : 0;
  1530. return (
  1531. <SpoolCard
  1532. key={spool.id}
  1533. spool={spool}
  1534. remaining={remaining}
  1535. pct={pct}
  1536. onClick={() => setFormModal({ spool, mode: 'edit' })}
  1537. onPrintLabel={() => setLabelPickerSpoolIds([spool.id])}
  1538. onCopy={() => setFormModal({ spool: spool, mode: 'copy' })}
  1539. t={t}
  1540. />
  1541. );
  1542. })}
  1543. </div>
  1544. )}
  1545. </div>
  1546. );
  1547. }
  1548. const spool = item.spool;
  1549. const remaining = Math.max(0, spool.label_weight - spool.weight_used);
  1550. const pct = spool.label_weight > 0 ? (remaining / spool.label_weight) * 100 : 0;
  1551. return (
  1552. <SpoolCard
  1553. key={spool.id}
  1554. spool={spool}
  1555. remaining={remaining}
  1556. pct={pct}
  1557. onClick={() => setFormModal({ spool, mode: 'edit' })}
  1558. onPrintLabel={() => setLabelPickerSpoolIds([spool.id])}
  1559. onCopy={() => setFormModal({ spool: spool, mode: 'copy' })}
  1560. t={t}
  1561. />
  1562. );
  1563. })}
  1564. </div>
  1565. {/* Pagination for cards */}
  1566. <PaginationBar
  1567. pageIndex={safePageIndex}
  1568. pageSize={pageSize}
  1569. totalRows={totalDisplayItems}
  1570. totalPages={totalPages}
  1571. onPageChange={setPageIndex}
  1572. onPageSizeChange={handlePageSizeChange}
  1573. t={t}
  1574. />
  1575. </>
  1576. ) : (
  1577. <EmptyFilterState
  1578. hasFilters={hasActiveFilters}
  1579. onAddSpool={() => setFormModal({ spool: null, mode: 'create' })}
  1580. t={t}
  1581. />
  1582. )
  1583. ) : (
  1584. /* Table view */
  1585. pagedItems.length > 0 ? (
  1586. <div className="bg-bambu-dark-secondary rounded-lg overflow-hidden border border-bambu-dark-tertiary">
  1587. <div className="overflow-x-auto">
  1588. <table className="w-full">
  1589. <thead>
  1590. <tr className="border-b border-bambu-dark-tertiary bg-bambu-dark-tertiary/30">
  1591. {visibleColumns.map((colId) => {
  1592. const sortable = !!columnSortValues[colId];
  1593. const isActive = sortState?.column === colId;
  1594. return (
  1595. <th
  1596. key={colId}
  1597. className={`text-left py-3 px-4 text-xs font-medium uppercase tracking-wide select-none ${colId === 'remaining' ? 'min-w-[150px]' : ''} ${
  1598. sortable ? 'cursor-pointer hover:text-bambu-green transition-colors' : ''
  1599. } ${isActive ? 'text-bambu-green' : 'text-bambu-gray'}`}
  1600. onClick={sortable ? () => handleSort(colId) : undefined}
  1601. >
  1602. <span className="inline-flex items-center gap-1">
  1603. {columnHeaders[colId]?.(t) ?? colId}
  1604. {sortable && (
  1605. isActive
  1606. ? sortState.direction === 'asc'
  1607. ? <ArrowUp className="w-3 h-3" />
  1608. : <ArrowDown className="w-3 h-3" />
  1609. : <ArrowUpDown className="w-3 h-3 opacity-30" />
  1610. )}
  1611. </span>
  1612. </th>
  1613. );
  1614. })}
  1615. <th className="text-right py-3 px-4 text-xs font-medium text-bambu-gray uppercase tracking-wide">{t('common.actions')}</th>
  1616. </tr>
  1617. </thead>
  1618. <tbody>
  1619. {pagedItems.map((item) => {
  1620. if (item.type === 'group') {
  1621. const { key, spools: groupSpools } = item;
  1622. const isExpanded = expandedGroups.has(key);
  1623. // Header row shows group totals (#1368): an aggregate
  1624. // spool plus remaining / pct summed across all members.
  1625. const headerSpool = aggregateGroupSpool(groupSpools);
  1626. const remaining = Math.max(0, headerSpool.label_weight - headerSpool.weight_used);
  1627. const pct = headerSpool.label_weight > 0 ? (remaining / headerSpool.label_weight) * 100 : 0;
  1628. return (
  1629. <SpoolTableGroup
  1630. key={`group-${key}`}
  1631. spools={groupSpools}
  1632. headerSpool={headerSpool}
  1633. remaining={remaining}
  1634. pct={pct}
  1635. isExpanded={isExpanded}
  1636. onToggle={() => toggleGroupExpand(key)}
  1637. onEdit={(s) => setFormModal({ spool: s, mode: 'edit' })}
  1638. onCopy={(s) => setFormModal({ spool: s, mode: 'copy' })}
  1639. onArchive={(id) => setConfirmAction({ type: 'archive', spoolId: id })}
  1640. onDelete={(id) => setConfirmAction({ type: 'delete', spoolId: id })}
  1641. onPrintLabel={(id) => setLabelPickerSpoolIds([id])}
  1642. onResetUsage={(id) => setConfirmAction({ type: 'reset-usage', spoolId: id })}
  1643. visibleColumns={visibleColumns}
  1644. assignmentMap={assignmentMap}
  1645. catalogMap={catalogMap}
  1646. currencySymbol={currencySymbol}
  1647. dateFormat={dateFormat}
  1648. t={t}
  1649. onSyncWeight={handleSyncWeight}
  1650. />
  1651. );
  1652. }
  1653. const spool = item.spool;
  1654. const remaining = Math.max(0, spool.label_weight - spool.weight_used);
  1655. const pct = spool.label_weight > 0 ? (remaining / spool.label_weight) * 100 : 0;
  1656. return (
  1657. <SpoolTableRow
  1658. key={spool.id}
  1659. spool={spool}
  1660. remaining={remaining}
  1661. pct={pct}
  1662. onEdit={() => setFormModal({ spool, mode: 'edit' })}
  1663. onCopy={() => setFormModal({ spool: spool, mode: 'copy' })}
  1664. onRestore={() => restoreMutation.mutate(spool.id)}
  1665. onArchive={() => setConfirmAction({ type: 'archive', spoolId: spool.id })}
  1666. onDelete={() => setConfirmAction({ type: 'delete', spoolId: spool.id })}
  1667. onPrintLabel={() => setLabelPickerSpoolIds([spool.id])}
  1668. onResetUsage={() => setConfirmAction({ type: 'reset-usage', spoolId: spool.id })}
  1669. visibleColumns={visibleColumns}
  1670. assignmentMap={assignmentMap}
  1671. catalogMap={catalogMap}
  1672. currencySymbol={currencySymbol}
  1673. dateFormat={dateFormat}
  1674. t={t}
  1675. onSyncWeight={handleSyncWeight}
  1676. />
  1677. );
  1678. })}
  1679. </tbody>
  1680. </table>
  1681. </div>
  1682. {/* Pagination inside card footer */}
  1683. <div className="flex items-center justify-between px-4 py-3 bg-bambu-dark-tertiary/50 border-t border-bambu-dark-tertiary text-sm">
  1684. <span className="text-bambu-gray">
  1685. {showAll
  1686. ? `${totalDisplayItems} ${totalDisplayItems !== 1 ? t('inventory.spools') : t('inventory.spool')}`
  1687. : <>{t('inventory.showing')} {safePageIndex * effectivePageSize + 1} {t('inventory.to')}{' '}
  1688. {Math.min((safePageIndex + 1) * effectivePageSize, totalDisplayItems)}{' '}
  1689. {t('inventory.of')} {totalDisplayItems} {t('inventory.spools')}</>
  1690. }
  1691. </span>
  1692. <div className="flex items-center gap-2">
  1693. <span className="text-bambu-gray">{t('inventory.show')}</span>
  1694. <select
  1695. value={pageSize}
  1696. onChange={(e) => handlePageSizeChange(Number(e.target.value))}
  1697. className="px-2 py-1 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded text-white text-sm focus:outline-none focus:border-bambu-green"
  1698. >
  1699. {[15, 30, 50, 100].map((n) => (
  1700. <option key={n} value={n}>{n}</option>
  1701. ))}
  1702. <option value={-1}>{t('inventory.all')}</option>
  1703. </select>
  1704. {!showAll && (
  1705. <>
  1706. <button
  1707. onClick={() => setPageIndex(0)}
  1708. disabled={safePageIndex === 0}
  1709. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1710. title="First page"
  1711. >
  1712. <ChevronsLeft className="w-4 h-4" />
  1713. </button>
  1714. <button
  1715. onClick={() => setPageIndex((p) => Math.max(0, p - 1))}
  1716. disabled={safePageIndex === 0}
  1717. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1718. >
  1719. <ChevronLeft className="w-4 h-4" />
  1720. </button>
  1721. <span className="text-bambu-gray px-2 whitespace-nowrap">
  1722. {t('inventory.page')} {safePageIndex + 1} {t('inventory.of')} {totalPages}
  1723. </span>
  1724. <button
  1725. onClick={() => setPageIndex((p) => Math.min(totalPages - 1, p + 1))}
  1726. disabled={safePageIndex >= totalPages - 1}
  1727. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1728. >
  1729. <ChevronRight className="w-4 h-4" />
  1730. </button>
  1731. <button
  1732. onClick={() => setPageIndex(totalPages - 1)}
  1733. disabled={safePageIndex >= totalPages - 1}
  1734. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1735. title="Last page"
  1736. >
  1737. <ChevronsRight className="w-4 h-4" />
  1738. </button>
  1739. </>
  1740. )}
  1741. </div>
  1742. </div>
  1743. </div>
  1744. ) : (
  1745. <EmptyFilterState
  1746. hasFilters={hasActiveFilters}
  1747. onAddSpool={() => setFormModal({ spool: null, mode: 'create' })}
  1748. t={t}
  1749. />
  1750. )
  1751. )}
  1752. {/* Spool Form Modal */}
  1753. {formModal !== null && (
  1754. <SpoolFormModal
  1755. isOpen={true}
  1756. onClose={() => setFormModal(null)}
  1757. spool={formModal.spool}
  1758. mode={formModal.mode}
  1759. currencySymbol={currencySymbol}
  1760. spoolmanMode={spoolmanMode}
  1761. spoolsQueryKey={spoolsQueryKey}
  1762. />
  1763. )}
  1764. {/* Confirm Modal (delete / archive / reset-usage / reset-all-usage) */}
  1765. {confirmAction && (
  1766. <ConfirmModal
  1767. title={
  1768. confirmAction.type === 'delete' ? t('common.delete') :
  1769. confirmAction.type === 'archive' ? t('inventory.archive') :
  1770. confirmAction.type === 'reset-usage' ? t('inventory.resetUsage') :
  1771. t('inventory.resetAllUsage')
  1772. }
  1773. message={
  1774. confirmAction.type === 'delete' ? t('inventory.deleteConfirm') :
  1775. confirmAction.type === 'archive' ? t('inventory.archiveConfirm') :
  1776. confirmAction.type === 'reset-usage' ? t('inventory.resetUsageConfirm') :
  1777. t('inventory.resetAllUsageConfirm', { count: resetableSpoolIds.length })
  1778. }
  1779. confirmText={
  1780. confirmAction.type === 'delete' ? t('common.delete') :
  1781. confirmAction.type === 'archive' ? t('inventory.archive') :
  1782. t('inventory.resetUsage')
  1783. }
  1784. variant={confirmAction.type === 'archive' ? 'warning' : 'danger'}
  1785. onConfirm={() => {
  1786. if (confirmAction.type === 'delete') {
  1787. deleteMutation.mutate(confirmAction.spoolId);
  1788. } else if (confirmAction.type === 'archive') {
  1789. archiveMutation.mutate(confirmAction.spoolId);
  1790. } else if (confirmAction.type === 'reset-usage') {
  1791. resetUsageMutation.mutate(confirmAction.spoolId);
  1792. } else {
  1793. bulkResetUsageMutation.mutate(resetableSpoolIds);
  1794. }
  1795. setConfirmAction(null);
  1796. }}
  1797. onCancel={() => setConfirmAction(null)}
  1798. />
  1799. )}
  1800. {/* Column Config Modal */}
  1801. <ColumnConfigModal
  1802. isOpen={showColumnModal}
  1803. onClose={() => setShowColumnModal(false)}
  1804. columns={columnConfig}
  1805. defaultColumns={DEFAULT_COLUMNS}
  1806. onSave={handleColumnConfigSave}
  1807. />
  1808. <LabelTemplatePickerModal
  1809. isOpen={labelPickerSpoolIds !== null}
  1810. onClose={() => setLabelPickerSpoolIds(null)}
  1811. availableSpools={filteredSpools}
  1812. initialSelectedIds={labelPickerSpoolIds ?? []}
  1813. spoolmanMode={spoolmanMode}
  1814. />
  1815. </div>
  1816. );
  1817. }
  1818. /* Pagination bar (reused for cards view) */
  1819. function PaginationBar({
  1820. pageIndex, pageSize, totalRows, totalPages, onPageChange, onPageSizeChange, t,
  1821. }: {
  1822. pageIndex: number;
  1823. pageSize: number;
  1824. totalRows: number;
  1825. totalPages: number;
  1826. onPageChange: (page: number) => void;
  1827. onPageSizeChange: (size: number) => void;
  1828. t: (key: string) => string;
  1829. }) {
  1830. const isShowAll = pageSize === -1;
  1831. if (totalPages <= 1 && !isShowAll) return null;
  1832. const effectiveSize = isShowAll ? totalRows || 1 : pageSize;
  1833. return (
  1834. <div className="flex items-center justify-between pt-2 text-sm">
  1835. <span className="text-bambu-gray">
  1836. {isShowAll
  1837. ? `${totalRows} ${totalRows !== 1 ? t('inventory.spools') : t('inventory.spool')}`
  1838. : <>{t('inventory.showing')} {pageIndex * effectiveSize + 1} {t('inventory.to')}{' '}
  1839. {Math.min((pageIndex + 1) * effectiveSize, totalRows)}{' '}
  1840. {t('inventory.of')} {totalRows} {t('inventory.spools')}</>
  1841. }
  1842. </span>
  1843. <div className="flex items-center gap-2">
  1844. <span className="text-bambu-gray">{t('inventory.show')}</span>
  1845. <select
  1846. value={pageSize}
  1847. onChange={(e) => onPageSizeChange(Number(e.target.value))}
  1848. className="px-2 py-1 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded text-white text-sm focus:outline-none focus:border-bambu-green"
  1849. >
  1850. {[15, 30, 50, 100].map((n) => (
  1851. <option key={n} value={n}>{n}</option>
  1852. ))}
  1853. <option value={-1}>{t('inventory.all')}</option>
  1854. </select>
  1855. {!isShowAll && (
  1856. <>
  1857. <button
  1858. onClick={() => onPageChange(0)}
  1859. disabled={pageIndex === 0}
  1860. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1861. >
  1862. <ChevronsLeft className="w-4 h-4" />
  1863. </button>
  1864. <button
  1865. onClick={() => onPageChange(Math.max(0, pageIndex - 1))}
  1866. disabled={pageIndex === 0}
  1867. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1868. >
  1869. <ChevronLeft className="w-4 h-4" />
  1870. </button>
  1871. <span className="text-bambu-gray px-2 whitespace-nowrap">
  1872. {t('inventory.page')} {pageIndex + 1} {t('inventory.of')} {totalPages}
  1873. </span>
  1874. <button
  1875. onClick={() => onPageChange(Math.min(totalPages - 1, pageIndex + 1))}
  1876. disabled={pageIndex >= totalPages - 1}
  1877. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1878. >
  1879. <ChevronRight className="w-4 h-4" />
  1880. </button>
  1881. <button
  1882. onClick={() => onPageChange(totalPages - 1)}
  1883. disabled={pageIndex >= totalPages - 1}
  1884. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  1885. >
  1886. <ChevronsRight className="w-4 h-4" />
  1887. </button>
  1888. </>
  1889. )}
  1890. </div>
  1891. </div>
  1892. );
  1893. }
  1894. /* Spool card for cards view */
  1895. function SpoolCard({
  1896. spool, remaining, pct, onClick, onPrintLabel, onCopy, t,
  1897. }: {
  1898. spool: InventorySpool;
  1899. remaining: number;
  1900. pct: number;
  1901. onClick: () => void;
  1902. onPrintLabel?: () => void;
  1903. onCopy?: () => void;
  1904. t: (key: string, opts?: Record<string, unknown>) => string;
  1905. }) {
  1906. const bannerStyle = buildFilamentBackground({
  1907. rgba: spool.rgba,
  1908. extraColors: spool.extra_colors,
  1909. effectType: spool.effect_type,
  1910. subtype: spool.subtype,
  1911. effectSize: 'card',
  1912. });
  1913. return (
  1914. <div
  1915. className={`bg-bambu-dark-secondary rounded-lg overflow-hidden border border-bambu-dark-tertiary hover:border-bambu-green transition-colors cursor-pointer ${spool.archived_at ? 'opacity-50' : ''}`}
  1916. onClick={onClick}
  1917. >
  1918. <div className="h-14 flex items-center justify-center" style={bannerStyle}>
  1919. <span className="bg-white/90 text-gray-800 px-3 py-0.5 rounded-full text-sm font-medium">
  1920. {resolveSpoolColorName(spool.color_name, spool.rgba) || '-'}
  1921. </span>
  1922. {onCopy && (
  1923. <button
  1924. type="button"
  1925. onClick={(e) => { e.stopPropagation(); onCopy(); }}
  1926. className="p-1.5 bg-black/20 hover:bg-black/40 text-white rounded-full transition-colors"
  1927. title={t('inventory.copySpool')}
  1928. >
  1929. <Copy className="w-3.5 h-3.5" />
  1930. </button>
  1931. )}
  1932. </div>
  1933. <div className="p-4 space-y-3">
  1934. <div className="flex items-start justify-between gap-2">
  1935. <div>
  1936. <h3 className="font-semibold text-white">
  1937. {spool.material}{spool.subtype ? ` ${spool.subtype}` : ''}
  1938. </h3>
  1939. <p className="text-sm text-bambu-gray">{spool.brand || '-'}</p>
  1940. </div>
  1941. <div className="flex items-center gap-1">
  1942. {onPrintLabel && (
  1943. <button
  1944. onClick={(e) => { e.stopPropagation(); onPrintLabel(); }}
  1945. className="p-1 text-bambu-gray hover:text-white rounded transition-colors"
  1946. title={t('inventory.labels.printOne')}
  1947. aria-label={t('inventory.labels.printOne')}
  1948. >
  1949. <Printer className="w-4 h-4" />
  1950. </button>
  1951. )}
  1952. <span className="text-xs font-mono text-bambu-gray bg-bambu-dark-tertiary px-2 py-1 rounded">
  1953. #{spool.id}
  1954. </span>
  1955. </div>
  1956. </div>
  1957. <div>
  1958. <div className="flex justify-between text-xs text-bambu-gray mb-1">
  1959. <span>{t('inventory.remaining')}</span>
  1960. <span>{Math.round(pct)}%</span>
  1961. </div>
  1962. <div className="flex items-center gap-2">
  1963. <div className="flex-1 h-2 bg-bambu-dark-tertiary rounded-full overflow-hidden">
  1964. <div
  1965. className={`h-full rounded-full ${pct > 50 ? 'bg-bambu-green' : pct > 20 ? 'bg-yellow-500' : 'bg-red-500'}`}
  1966. style={{ width: `${Math.min(pct, 100)}%` }}
  1967. />
  1968. </div>
  1969. <span className="text-xs text-bambu-gray min-w-[40px] text-right">
  1970. {Math.round(remaining)}g
  1971. </span>
  1972. </div>
  1973. </div>
  1974. <div className="grid grid-cols-2 gap-2 text-xs">
  1975. <div>
  1976. <span className="text-bambu-gray/60">{t('inventory.labelWeight')}: </span>
  1977. <span className="text-bambu-gray">{formatWeight(spool.label_weight)}</span>
  1978. </div>
  1979. <div>
  1980. <span className="text-bambu-gray/60">{t('inventory.weightUsed')}: </span>
  1981. <span className="text-bambu-gray">
  1982. {spool.weight_used > 0 ? formatWeight(spool.weight_used) : '-'}
  1983. </span>
  1984. </div>
  1985. </div>
  1986. {spool.note && (
  1987. <div
  1988. className="text-xs text-bambu-gray/60 pt-2 border-t border-bambu-dark-tertiary truncate"
  1989. title={spool.note}
  1990. >
  1991. {spool.note}
  1992. </div>
  1993. )}
  1994. </div>
  1995. </div>
  1996. );
  1997. }
  1998. /* Single spool row for table view */
  1999. function SpoolTableRow({
  2000. spool, remaining, pct, onEdit, onCopy, onRestore, onArchive, onDelete, onPrintLabel, onResetUsage,
  2001. visibleColumns, assignmentMap, catalogMap, currencySymbol, dateFormat, t, onSyncWeight,
  2002. }: {
  2003. spool: InventorySpool;
  2004. remaining: number;
  2005. pct: number;
  2006. onEdit: () => void;
  2007. onCopy?: () => void;
  2008. onRestore: () => void;
  2009. onArchive: () => void;
  2010. onDelete: () => void;
  2011. onPrintLabel?: () => void;
  2012. onResetUsage?: () => void;
  2013. visibleColumns: string[];
  2014. assignmentMap: Record<number, LocationDisplay>;
  2015. catalogMap: Record<number, SpoolCatalogEntry>;
  2016. currencySymbol: string;
  2017. dateFormat: DateFormat;
  2018. t: TFn;
  2019. onSyncWeight?: (spool: InventorySpool) => void;
  2020. }) {
  2021. return (
  2022. <tr
  2023. className={`border-b border-bambu-dark-tertiary/50 hover:bg-bambu-dark-tertiary/30 transition-colors cursor-pointer ${
  2024. spool.archived_at ? 'opacity-50' : ''
  2025. }`}
  2026. onClick={onEdit}
  2027. >
  2028. {visibleColumns.map((colId) => (
  2029. <td key={colId} className="py-3 px-4">
  2030. {columnCells[colId]?.({ spool, remaining, pct, assignmentMap, catalogMap, currencySymbol, dateFormat, t, onSyncWeight })}
  2031. </td>
  2032. ))}
  2033. <td className="py-3 px-4">
  2034. <div className="flex items-center justify-end gap-1" onClick={(e) => e.stopPropagation()}>
  2035. <button onClick={onEdit} className="p-1.5 text-bambu-gray hover:text-white rounded transition-colors" title={t('common.edit')}>
  2036. <Edit2 className="w-4 h-4" />
  2037. </button>
  2038. {onCopy && (
  2039. <button onClick={onCopy} className="p-1.5 text-bambu-gray hover:text-bambu-green rounded transition-colors" title={t('inventory.copySpool')}>
  2040. <Copy className="w-4 h-4" />
  2041. </button>
  2042. )}
  2043. {onPrintLabel && (
  2044. <button onClick={onPrintLabel} className="p-1.5 text-bambu-gray hover:text-white rounded transition-colors" title={t('inventory.labels.printOne')}>
  2045. <Printer className="w-4 h-4" />
  2046. </button>
  2047. )}
  2048. {onResetUsage && spool.weight_used > 0 && (
  2049. // Eraser also shows on archived spools (#1390 follow-up):
  2050. // archived consumed weight now counts in "Total Consumed", so
  2051. // the user needs a way to zero an archived spool's tracking
  2052. // counter individually without having to un-archive it first.
  2053. <button onClick={onResetUsage} className="p-1.5 text-bambu-gray hover:text-orange-400 rounded transition-colors" title={t('inventory.resetUsageTooltip')}>
  2054. <Eraser className="w-4 h-4" />
  2055. </button>
  2056. )}
  2057. {spool.archived_at ? (
  2058. <button onClick={onRestore} className="p-1.5 text-bambu-gray hover:text-bambu-green rounded transition-colors" title={t('inventory.restore')}>
  2059. <RotateCcw className="w-4 h-4" />
  2060. </button>
  2061. ) : (
  2062. <button onClick={onArchive} className="p-1.5 text-bambu-gray hover:text-yellow-400 rounded transition-colors" title={t('inventory.archive')}>
  2063. <Archive className="w-4 h-4" />
  2064. </button>
  2065. )}
  2066. <button onClick={onDelete} className="p-1.5 text-bambu-gray hover:text-red-400 rounded transition-colors" title={t('common.delete')}>
  2067. <Trash2 className="w-4 h-4" />
  2068. </button>
  2069. </div>
  2070. </td>
  2071. </tr>
  2072. );
  2073. }
  2074. /* Grouped spool rows for table view */
  2075. function SpoolTableGroup({
  2076. spools, headerSpool, remaining, pct, isExpanded, onToggle,
  2077. onEdit, onCopy, onArchive, onDelete, onPrintLabel, onResetUsage,
  2078. visibleColumns, assignmentMap, catalogMap, currencySymbol, dateFormat, t, onSyncWeight,
  2079. }: {
  2080. spools: InventorySpool[];
  2081. // Aggregate of all members (summed quantities, shared identity) — rendered
  2082. // in the collapsed header row so it shows group totals (#1368).
  2083. headerSpool: InventorySpool;
  2084. remaining: number;
  2085. pct: number;
  2086. isExpanded: boolean;
  2087. onToggle: () => void;
  2088. onEdit: (spool: InventorySpool) => void;
  2089. onCopy?: (spool: InventorySpool) => void;
  2090. onArchive: (id: number) => void;
  2091. onDelete: (id: number) => void;
  2092. onPrintLabel?: (spoolId: number) => void;
  2093. onResetUsage?: (id: number) => void;
  2094. visibleColumns: string[];
  2095. assignmentMap: Record<number, LocationDisplay>;
  2096. catalogMap: Record<number, SpoolCatalogEntry>;
  2097. currencySymbol: string;
  2098. dateFormat: DateFormat;
  2099. t: TFn;
  2100. onSyncWeight?: (spool: InventorySpool) => void;
  2101. }) {
  2102. return (
  2103. <>
  2104. {/* Group header row */}
  2105. <tr
  2106. className="border-b border-bambu-dark-tertiary/50 hover:bg-bambu-dark-tertiary/30 transition-colors cursor-pointer bg-bambu-green/5"
  2107. onClick={onToggle}
  2108. >
  2109. {visibleColumns.map((colId, idx) => (
  2110. <td key={colId} className="py-3 px-4">
  2111. {idx === 0 ? (
  2112. <div className="flex items-center gap-2">
  2113. <ChevronDown className={`w-4 h-4 text-bambu-gray transition-transform ${isExpanded ? '' : '-rotate-90'}`} />
  2114. {columnCells[colId]?.({ spool: headerSpool, remaining, pct, assignmentMap, catalogMap, currencySymbol, dateFormat, t, onSyncWeight })}
  2115. </div>
  2116. ) : colId === 'id' ? (
  2117. <span className="text-xs font-medium bg-bambu-green/20 text-bambu-green px-2 py-0.5 rounded-full">
  2118. {t('inventory.groupedSpools', { count: spools.length })}
  2119. </span>
  2120. ) : (
  2121. columnCells[colId]?.({ spool: headerSpool, remaining, pct, assignmentMap, catalogMap, currencySymbol, dateFormat, t, onSyncWeight })
  2122. )}
  2123. </td>
  2124. ))}
  2125. <td className="py-3 px-4">
  2126. <span className="text-xs text-bambu-gray">
  2127. {spools.map((s) => `#${s.id}`).join(', ')}
  2128. </span>
  2129. </td>
  2130. </tr>
  2131. {/* Expanded individual rows */}
  2132. {isExpanded && spools.map((spool) => {
  2133. const r = Math.max(0, spool.label_weight - spool.weight_used);
  2134. const p = spool.label_weight > 0 ? (r / spool.label_weight) * 100 : 0;
  2135. return (
  2136. <SpoolTableRow
  2137. key={spool.id}
  2138. spool={spool}
  2139. remaining={r}
  2140. pct={p}
  2141. onEdit={() => onEdit(spool)}
  2142. onCopy={onCopy ? () => onCopy(spool) : undefined}
  2143. onRestore={() => {}}
  2144. onArchive={() => onArchive(spool.id)}
  2145. onDelete={() => onDelete(spool.id)}
  2146. onPrintLabel={onPrintLabel ? () => onPrintLabel(spool.id) : undefined}
  2147. onResetUsage={onResetUsage ? () => onResetUsage(spool.id) : undefined}
  2148. visibleColumns={visibleColumns}
  2149. assignmentMap={assignmentMap}
  2150. catalogMap={catalogMap}
  2151. currencySymbol={currencySymbol}
  2152. dateFormat={dateFormat}
  2153. t={t}
  2154. onSyncWeight={onSyncWeight}
  2155. />
  2156. );
  2157. })}
  2158. </>
  2159. );
  2160. }
  2161. /* Empty state matching SpoolBuddy's design */
  2162. function EmptyFilterState({
  2163. hasFilters,
  2164. onAddSpool,
  2165. t,
  2166. }: {
  2167. hasFilters: boolean;
  2168. onAddSpool: () => void;
  2169. t: (key: string) => string;
  2170. }) {
  2171. return (
  2172. <div className="flex flex-col items-center justify-center py-16 px-4">
  2173. <div className="relative mb-6">
  2174. <div className="absolute inset-0 -m-4 bg-bambu-green/5 rounded-full blur-2xl" />
  2175. <div className="relative flex items-center justify-center w-24 h-24 rounded-2xl bg-gradient-to-br from-bambu-dark-secondary to-bambu-dark-tertiary border border-bambu-dark-tertiary shadow-lg">
  2176. <div className="absolute -top-1 -right-1 w-3 h-3 rounded-full bg-bambu-green/30" />
  2177. <div className="absolute -bottom-2 -left-2 w-2 h-2 rounded-full bg-bambu-green/20" />
  2178. {hasFilters ? (
  2179. <Search className="w-10 h-10 text-bambu-gray/40" strokeWidth={1.5} />
  2180. ) : (
  2181. <div className="relative">
  2182. <div className="w-14 h-14 rounded-full border-4 border-bambu-gray/20 flex items-center justify-center">
  2183. <div className="w-6 h-6 rounded-full bg-bambu-gray/10 border-2 border-bambu-gray/20" />
  2184. </div>
  2185. <div className="absolute -bottom-1 -right-1 w-6 h-6 rounded-full bg-bambu-green flex items-center justify-center shadow-md">
  2186. <span className="text-white text-lg font-bold leading-none">+</span>
  2187. </div>
  2188. </div>
  2189. )}
  2190. </div>
  2191. </div>
  2192. <h3 className="text-lg font-semibold text-white mb-2 text-center">
  2193. {hasFilters ? t('inventory.noSpoolsMatch') : t('inventory.noSpools').split('.')[0]}
  2194. </h3>
  2195. <p className="text-sm text-bambu-gray text-center max-w-sm mb-6">
  2196. {hasFilters
  2197. ? t('inventory.noSpoolsMatchDesc')
  2198. : t('inventory.noSpools')
  2199. }
  2200. </p>
  2201. {!hasFilters && (
  2202. <Button onClick={onAddSpool}>
  2203. <Package className="w-4 h-4" />
  2204. {t('inventory.addSpool')}
  2205. </Button>
  2206. )}
  2207. </div>
  2208. );
  2209. }