InventoryPage.tsx 44 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055
  1. import { useState, useMemo, type ReactNode } from 'react';
  2. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  3. import { useTranslation } from 'react-i18next';
  4. import {
  5. Plus, Loader2, Trash2, Archive, RotateCcw, Edit2, Package,
  6. Search, ChevronLeft, ChevronRight, ChevronsLeft, ChevronsRight,
  7. TrendingDown, Layers, Printer, AlertTriangle, X, Clock, LayoutGrid, TableProperties, Columns,
  8. } from 'lucide-react';
  9. import { api } from '../api/client';
  10. import type { InventorySpool, SpoolAssignment } from '../api/client';
  11. import { Button } from '../components/Button';
  12. import { SpoolFormModal } from '../components/SpoolFormModal';
  13. import { ColumnConfigModal, type ColumnConfig } from '../components/ColumnConfigModal';
  14. import { useToast } from '../contexts/ToastContext';
  15. type ArchiveFilter = 'active' | 'archived';
  16. type UsageFilter = 'all' | 'used' | 'new';
  17. type ViewMode = 'table' | 'cards';
  18. // Column definitions for the inventory table
  19. const COLUMN_CONFIG_KEY = 'bambuddy-inventory-columns';
  20. const DEFAULT_COLUMNS: ColumnConfig[] = [
  21. { id: 'id', label: '#', visible: true },
  22. { id: 'added_time', label: 'Added', visible: true },
  23. { id: 'encode_time', label: 'Encoded', visible: false },
  24. { id: 'last_used_time', label: 'Last Used', visible: false },
  25. { id: 'rgba', label: 'Color', visible: true },
  26. { id: 'material', label: 'Material', visible: true },
  27. { id: 'subtype', label: 'Subtype', visible: true },
  28. { id: 'color_name', label: 'Color Name', visible: false },
  29. { id: 'brand', label: 'Brand', visible: true },
  30. { id: 'slicer_filament', label: 'Slicer Filament', visible: false },
  31. { id: 'location', label: 'Location', visible: true },
  32. { id: 'label_weight', label: 'Label', visible: true },
  33. { id: 'net', label: 'Net', visible: true },
  34. { id: 'gross', label: 'Gross', visible: false },
  35. { id: 'added_full', label: 'Full', visible: false },
  36. { id: 'used', label: 'Used', visible: false },
  37. { id: 'printed_total', label: 'Printed Total', visible: false },
  38. { id: 'printed_since_weight', label: 'Printed Since Weight', visible: false },
  39. { id: 'note', label: 'Note', visible: false },
  40. { id: 'pa_k', label: 'PA(K)', visible: true },
  41. { id: 'tag_id', label: 'Tag ID', visible: false },
  42. { id: 'data_origin', label: 'Data Origin', visible: false },
  43. { id: 'tag_type', label: 'Linked Tag Type', visible: false },
  44. { id: 'remaining', label: 'Remaining', visible: true },
  45. ];
  46. function loadColumnConfig(): ColumnConfig[] {
  47. try {
  48. const stored = localStorage.getItem(COLUMN_CONFIG_KEY);
  49. if (stored) {
  50. const parsed = JSON.parse(stored) as ColumnConfig[];
  51. const defaultIds = new Set(DEFAULT_COLUMNS.map((c) => c.id));
  52. const storedIds = new Set(parsed.map((c) => c.id));
  53. // Keep stored columns that still exist in defaults
  54. const validStored = parsed.filter((c) => defaultIds.has(c.id));
  55. // Add any new default columns not in stored config
  56. const newColumns = DEFAULT_COLUMNS.filter((c) => !storedIds.has(c.id));
  57. return [...validStored, ...newColumns];
  58. }
  59. } catch {
  60. // Ignore errors
  61. }
  62. return DEFAULT_COLUMNS.map((c) => ({ ...c }));
  63. }
  64. function saveColumnConfig(config: ColumnConfig[]) {
  65. try {
  66. localStorage.setItem(COLUMN_CONFIG_KEY, JSON.stringify(config));
  67. } catch {
  68. // Ignore errors
  69. }
  70. }
  71. function formatWeight(g: number, useKg = false): string {
  72. if (useKg && g >= 1000) return `${(g / 1000).toFixed(1)}kg`;
  73. return `${Math.round(g)}g`;
  74. }
  75. // Material color mapping for pills
  76. const MATERIAL_COLORS: Record<string, string> = {
  77. PLA: 'bg-green-500/20 text-green-400',
  78. ABS: 'bg-red-500/20 text-red-400',
  79. PETG: 'bg-blue-500/20 text-blue-400',
  80. TPU: 'bg-purple-500/20 text-purple-400',
  81. ASA: 'bg-orange-500/20 text-orange-400',
  82. PA: 'bg-yellow-500/20 text-yellow-400',
  83. PC: 'bg-cyan-500/20 text-cyan-400',
  84. PET: 'bg-sky-500/20 text-sky-400',
  85. };
  86. type TFn = (key: string) => string;
  87. function formatDate(dateStr: string | null): string {
  88. if (!dateStr) return '-';
  89. const date = new Date(dateStr);
  90. return date.toLocaleDateString('en-GB', { day: '2-digit', month: '2-digit', year: '2-digit' });
  91. }
  92. type CellCtx = {
  93. spool: InventorySpool;
  94. remaining: number;
  95. pct: number;
  96. assignmentMap: Record<number, SpoolAssignment>;
  97. };
  98. // Column header labels (25 columns — matching SpoolBuddy exactly)
  99. const columnHeaders: Record<string, (t: TFn) => string> = {
  100. id: () => '#',
  101. added_time: () => 'Added',
  102. encode_time: () => 'Encoded',
  103. last_used_time: () => 'Last Used',
  104. rgba: (t) => t('inventory.color'),
  105. material: (t) => t('inventory.material'),
  106. subtype: (t) => t('inventory.subtype'),
  107. color_name: (t) => t('inventory.colorName'),
  108. brand: (t) => t('inventory.brand'),
  109. slicer_filament: (t) => t('inventory.slicerFilament'),
  110. location: () => 'Location',
  111. label_weight: (t) => t('inventory.labelWeight'),
  112. net: (t) => t('inventory.net'),
  113. gross: () => 'Gross',
  114. added_full: () => 'Full',
  115. used: (t) => t('inventory.weightUsed'),
  116. printed_total: () => 'Printed Total',
  117. printed_since_weight: () => 'Printed Since Weight',
  118. note: (t) => t('inventory.note'),
  119. pa_k: () => 'PA(K)',
  120. tag_id: () => 'Tag ID',
  121. data_origin: () => 'Data Origin',
  122. tag_type: () => 'Linked Tag Type',
  123. remaining: (t) => t('inventory.remaining'),
  124. };
  125. // Column cell renderers (25 columns — matching SpoolBuddy exactly)
  126. const columnCells: Record<string, (ctx: CellCtx) => ReactNode> = {
  127. id: ({ spool }) => (
  128. <span className="text-sm font-medium text-white">{spool.id}</span>
  129. ),
  130. added_time: ({ spool }) => (
  131. <span className="text-sm text-bambu-gray">{formatDate(spool.created_at)}</span>
  132. ),
  133. encode_time: ({ spool }) => (
  134. <span className="text-sm text-bambu-gray">{formatDate(spool.encode_time)}</span>
  135. ),
  136. last_used_time: ({ spool }) => (
  137. <span className="text-sm text-bambu-gray">{spool.last_used ? formatDate(spool.last_used) : 'Never'}</span>
  138. ),
  139. rgba: ({ spool }) => (
  140. <div className="flex items-center gap-2">
  141. <span
  142. className="w-5 h-5 rounded-full border border-white/20 flex-shrink-0"
  143. style={{ backgroundColor: spool.rgba ? `#${spool.rgba.substring(0, 6)}` : '#808080' }}
  144. />
  145. <span className="text-sm text-white">{spool.color_name || '-'}</span>
  146. </div>
  147. ),
  148. material: ({ spool }) => (
  149. <span className="text-sm text-white">{spool.material}</span>
  150. ),
  151. subtype: ({ spool }) => (
  152. <span className="text-sm text-bambu-gray">{spool.subtype || '-'}</span>
  153. ),
  154. color_name: ({ spool }) => (
  155. <span className="text-sm text-bambu-gray">{spool.color_name || '-'}</span>
  156. ),
  157. brand: ({ spool }) => (
  158. <span className="text-sm text-bambu-gray">{spool.brand || '-'}</span>
  159. ),
  160. slicer_filament: ({ spool }) => (
  161. <span className="text-sm text-bambu-gray" title={spool.slicer_filament || undefined}>
  162. {spool.slicer_filament_name || spool.slicer_filament || '-'}
  163. </span>
  164. ),
  165. location: ({ spool, assignmentMap }) => {
  166. const assignment = assignmentMap[spool.id];
  167. if (!assignment) return <span className="text-sm text-bambu-gray">-</span>;
  168. return (
  169. <span className="inline-flex items-center px-1.5 py-0.5 rounded text-xs font-medium bg-purple-500/20 text-purple-400">
  170. AMS {assignment.ams_id} T{assignment.tray_id}
  171. </span>
  172. );
  173. },
  174. label_weight: ({ spool }) => (
  175. <span className="text-sm text-white">{formatWeight(spool.label_weight)}</span>
  176. ),
  177. net: ({ remaining }) => (
  178. <span className="text-sm text-white">{formatWeight(remaining)}</span>
  179. ),
  180. gross: ({ spool, remaining }) => (
  181. <span className="text-sm text-bambu-gray">{formatWeight(remaining + spool.core_weight)}</span>
  182. ),
  183. added_full: ({ spool }) => (
  184. <span className="text-sm text-bambu-gray">{spool.added_full == null ? '-' : spool.added_full ? 'Yes' : 'No'}</span>
  185. ),
  186. used: ({ spool }) => (
  187. <span className="text-sm text-bambu-gray">{spool.weight_used > 0 ? formatWeight(spool.weight_used) : '-'}</span>
  188. ),
  189. printed_total: () => (
  190. <span className="text-sm text-bambu-gray/50">-</span>
  191. ),
  192. printed_since_weight: () => (
  193. <span className="text-sm text-bambu-gray/50">-</span>
  194. ),
  195. note: ({ spool }) => (
  196. <span className="text-sm text-bambu-gray max-w-[150px] truncate block" title={spool.note || undefined}>{spool.note || '-'}</span>
  197. ),
  198. pa_k: ({ spool }) => {
  199. const count = spool.k_profiles?.length ?? 0;
  200. if (count === 0) return <span className="text-sm text-bambu-gray">-</span>;
  201. return (
  202. <span className="inline-flex items-center px-1.5 py-0.5 rounded text-xs font-medium bg-bambu-green/20 text-bambu-green">
  203. K
  204. </span>
  205. );
  206. },
  207. tag_id: ({ spool }) => {
  208. const tag = spool.tag_uid || spool.tray_uuid;
  209. if (!tag) return <span className="text-sm text-bambu-gray/50">-</span>;
  210. return (
  211. <span className="text-sm text-bambu-gray font-mono" title={tag}>
  212. {tag.length > 12 ? `${tag.slice(0, 6)}...${tag.slice(-4)}` : tag}
  213. </span>
  214. );
  215. },
  216. data_origin: ({ spool }) => (
  217. <span className="text-sm text-bambu-gray">{spool.data_origin || '-'}</span>
  218. ),
  219. tag_type: ({ spool }) => (
  220. <span className="text-sm text-bambu-gray">{spool.tag_type || '-'}</span>
  221. ),
  222. remaining: ({ remaining, pct }) => (
  223. <div className="flex items-center gap-2">
  224. <div className="flex-1 h-2 bg-bambu-dark-tertiary rounded-full overflow-hidden">
  225. <div
  226. className={`h-full rounded-full ${pct > 50 ? 'bg-bambu-green' : pct > 20 ? 'bg-yellow-500' : 'bg-red-500'}`}
  227. style={{ width: `${Math.min(pct, 100)}%` }}
  228. />
  229. </div>
  230. <span className="text-xs text-bambu-gray min-w-[40px] text-right">{Math.round(remaining)}g</span>
  231. </div>
  232. ),
  233. };
  234. export default function InventoryPage() {
  235. const { t } = useTranslation();
  236. const queryClient = useQueryClient();
  237. const { showToast } = useToast();
  238. const [formModal, setFormModal] = useState<{ spool?: InventorySpool | null } | null>(null);
  239. // Filter state
  240. const [archiveFilter, setArchiveFilter] = useState<ArchiveFilter>('active');
  241. const [usageFilter, setUsageFilter] = useState<UsageFilter>('all');
  242. const [materialFilter, setMaterialFilter] = useState('');
  243. const [brandFilter, setBrandFilter] = useState('');
  244. const [search, setSearch] = useState('');
  245. const [viewMode, setViewMode] = useState<ViewMode>('table');
  246. const [columnConfig, setColumnConfig] = useState<ColumnConfig[]>(loadColumnConfig);
  247. const [showColumnModal, setShowColumnModal] = useState(false);
  248. // Pagination state
  249. const [pageIndex, setPageIndex] = useState(0);
  250. const [pageSize, setPageSize] = useState(15);
  251. const { data: spools, isLoading } = useQuery({
  252. queryKey: ['inventory-spools'],
  253. queryFn: () => api.getSpools(true), // Always fetch all, filter client-side
  254. });
  255. const { data: assignments } = useQuery({
  256. queryKey: ['spool-assignments'],
  257. queryFn: () => api.getAssignments(),
  258. });
  259. const deleteMutation = useMutation({
  260. mutationFn: (id: number) => api.deleteSpool(id),
  261. onSuccess: () => {
  262. queryClient.invalidateQueries({ queryKey: ['inventory-spools'] });
  263. showToast(t('inventory.spoolDeleted'), 'success');
  264. },
  265. });
  266. const archiveMutation = useMutation({
  267. mutationFn: (id: number) => api.archiveSpool(id),
  268. onSuccess: () => {
  269. queryClient.invalidateQueries({ queryKey: ['inventory-spools'] });
  270. showToast(t('inventory.spoolArchived'), 'success');
  271. },
  272. });
  273. const restoreMutation = useMutation({
  274. mutationFn: (id: number) => api.restoreSpool(id),
  275. onSuccess: () => {
  276. queryClient.invalidateQueries({ queryKey: ['inventory-spools'] });
  277. showToast(t('inventory.spoolRestored'), 'success');
  278. },
  279. });
  280. // Stats calculation (active spools only)
  281. const stats = useMemo(() => {
  282. if (!spools) return null;
  283. let totalWeight = 0;
  284. let totalConsumed = 0;
  285. let lowStock = 0;
  286. let activeCount = 0;
  287. const byMaterial: Record<string, { count: number; weight: number }> = {};
  288. for (const s of spools) {
  289. if (s.archived_at) continue;
  290. activeCount++;
  291. const remaining = Math.max(0, s.label_weight - s.weight_used);
  292. totalWeight += remaining;
  293. totalConsumed += s.weight_used;
  294. const pct = s.label_weight > 0 ? (remaining / s.label_weight) * 100 : 0;
  295. if (pct < 20) lowStock++;
  296. const mat = s.material || 'Unknown';
  297. if (!byMaterial[mat]) byMaterial[mat] = { count: 0, weight: 0 };
  298. byMaterial[mat].count++;
  299. byMaterial[mat].weight += remaining;
  300. }
  301. return { totalWeight, totalConsumed, lowStock, byMaterial, totalSpools: activeCount };
  302. }, [spools]);
  303. const inPrinterCount = assignments?.length ?? 0;
  304. // Map spool_id -> assignment for location column
  305. const assignmentMap = useMemo(() => {
  306. const map: Record<number, SpoolAssignment> = {};
  307. for (const a of assignments || []) {
  308. map[a.spool_id] = a;
  309. }
  310. return map;
  311. }, [assignments]);
  312. // Top materials by weight for stat card pills
  313. const topMaterials = useMemo(() => {
  314. if (!stats) return [];
  315. return Object.entries(stats.byMaterial)
  316. .sort((a, b) => b[1].weight - a[1].weight)
  317. .slice(0, 4);
  318. }, [stats]);
  319. // Filtering pipeline
  320. const filteredSpools = useMemo(() => {
  321. let filtered = spools || [];
  322. // Archive filter
  323. if (archiveFilter === 'active') {
  324. filtered = filtered.filter((s) => !s.archived_at);
  325. } else {
  326. filtered = filtered.filter((s) => !!s.archived_at);
  327. }
  328. // Usage filter
  329. if (usageFilter === 'used') {
  330. filtered = filtered.filter((s) => s.weight_used > 0);
  331. } else if (usageFilter === 'new') {
  332. filtered = filtered.filter((s) => s.weight_used === 0);
  333. }
  334. // Material dropdown
  335. if (materialFilter) {
  336. filtered = filtered.filter((s) => s.material === materialFilter);
  337. }
  338. // Brand dropdown
  339. if (brandFilter) {
  340. filtered = filtered.filter((s) => s.brand === brandFilter);
  341. }
  342. // Global search
  343. if (search) {
  344. const q = search.toLowerCase();
  345. filtered = filtered.filter((s) =>
  346. s.brand?.toLowerCase().includes(q) ||
  347. s.material.toLowerCase().includes(q) ||
  348. s.color_name?.toLowerCase().includes(q) ||
  349. s.subtype?.toLowerCase().includes(q) ||
  350. s.note?.toLowerCase().includes(q) ||
  351. s.slicer_filament_name?.toLowerCase().includes(q)
  352. );
  353. }
  354. return filtered;
  355. }, [spools, archiveFilter, usageFilter, materialFilter, brandFilter, search]);
  356. // Pagination
  357. const totalPages = Math.max(1, Math.ceil(filteredSpools.length / pageSize));
  358. const safePageIndex = Math.min(pageIndex, totalPages - 1);
  359. const pagedSpools = filteredSpools.slice(safePageIndex * pageSize, (safePageIndex + 1) * pageSize);
  360. // Reset page on filter changes
  361. const resetPage = () => setPageIndex(0);
  362. // Unique values for filter dropdowns
  363. const uniqueMaterials = [...new Set(spools?.map((s) => s.material) || [])].sort();
  364. const uniqueBrands = [...new Set(spools?.map((s) => s.brand).filter(Boolean) || [])].sort() as string[];
  365. // Check if any filters are non-default
  366. const hasActiveFilters = archiveFilter !== 'active' || usageFilter !== 'all' || !!materialFilter || !!brandFilter || !!search;
  367. const handleColumnConfigSave = (config: ColumnConfig[]) => {
  368. setColumnConfig(config);
  369. saveColumnConfig(config);
  370. };
  371. // Visible column IDs in order
  372. const visibleColumns = useMemo(
  373. () => columnConfig.filter((c) => c.visible).map((c) => c.id),
  374. [columnConfig]
  375. );
  376. const clearAllFilters = () => {
  377. setArchiveFilter('active');
  378. setUsageFilter('all');
  379. setMaterialFilter('');
  380. setBrandFilter('');
  381. setSearch('');
  382. resetPage();
  383. };
  384. return (
  385. <div className="p-4 md:p-6 space-y-6">
  386. {/* Header */}
  387. <div className="flex items-center justify-between">
  388. <div>
  389. <div className="flex items-center gap-3">
  390. <Package className="w-6 h-6 text-bambu-green" />
  391. <h1 className="text-2xl font-bold text-white">{t('inventory.title')}</h1>
  392. </div>
  393. <p className="text-sm text-bambu-gray mt-1 ml-9">{t('inventory.noSpools').split('.')[0] ? '' : ''}</p>
  394. </div>
  395. <Button onClick={() => setFormModal({ spool: null })}>
  396. <Plus className="w-4 h-4" />
  397. {t('inventory.addSpool')}
  398. </Button>
  399. </div>
  400. {/* Stats Bar */}
  401. {stats && !isLoading && (
  402. <div className="grid grid-cols-2 md:grid-cols-3 lg:grid-cols-5 gap-3">
  403. {/* Total Inventory */}
  404. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  405. <div className="flex items-center gap-2 mb-1">
  406. <Package className="w-4 h-4 text-bambu-green" />
  407. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.totalInventory')}</span>
  408. </div>
  409. <div className="text-xl font-bold text-white">{formatWeight(stats.totalWeight, true)}</div>
  410. <div className="text-xs text-bambu-gray mt-1">{stats.totalSpools} {stats.totalSpools !== 1 ? t('inventory.spools') : t('inventory.spool')}</div>
  411. </div>
  412. {/* Total Consumed */}
  413. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  414. <div className="flex items-center gap-2 mb-1">
  415. <TrendingDown className="w-4 h-4 text-blue-400" />
  416. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.totalConsumed')}</span>
  417. </div>
  418. <div className="text-xl font-bold text-white">{formatWeight(stats.totalConsumed, true)}</div>
  419. <div className="text-xs text-bambu-gray mt-1">{t('inventory.sinceTracking')}</div>
  420. </div>
  421. {/* By Material */}
  422. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  423. <div className="flex items-center gap-2 mb-1">
  424. <Layers className="w-4 h-4 text-green-400" />
  425. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.byMaterial')}</span>
  426. </div>
  427. <div className="flex flex-wrap gap-1.5 mt-1">
  428. {topMaterials.map(([mat, data]) => (
  429. <span
  430. key={mat}
  431. 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'}`}
  432. >
  433. {mat} <span className="opacity-70">{formatWeight(data.weight, true)}</span>
  434. </span>
  435. ))}
  436. </div>
  437. </div>
  438. {/* In Printer */}
  439. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  440. <div className="flex items-center gap-2 mb-1">
  441. <Printer className="w-4 h-4 text-purple-400" />
  442. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.inPrinter')}</span>
  443. </div>
  444. <div className="text-xl font-bold text-white">{inPrinterCount}</div>
  445. <div className="text-xs text-bambu-gray mt-1">{t('inventory.loadedInAms')}</div>
  446. </div>
  447. {/* Low Stock */}
  448. <div className="bg-bambu-dark-secondary rounded-lg p-4">
  449. <div className="flex items-center gap-2 mb-1">
  450. <AlertTriangle className="w-4 h-4 text-yellow-400" />
  451. <span className="text-xs text-bambu-gray font-medium uppercase tracking-wide">{t('inventory.lowStock')}</span>
  452. </div>
  453. <div className={`text-xl font-bold ${stats.lowStock > 0 ? 'text-yellow-400' : 'text-white'}`}>{stats.lowStock}</div>
  454. <div className="text-xs text-bambu-gray mt-1">{t('inventory.lowStockThreshold')}</div>
  455. </div>
  456. </div>
  457. )}
  458. {/* Toolbar: Search + View toggle */}
  459. <div className="flex flex-col sm:flex-row gap-3 items-start sm:items-center justify-between">
  460. <div className="relative flex-1 max-w-md">
  461. <Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-bambu-gray/50" />
  462. <input
  463. type="text"
  464. value={search}
  465. onChange={(e) => { setSearch(e.target.value); resetPage(); }}
  466. placeholder={t('inventory.search')}
  467. 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"
  468. />
  469. {search && (
  470. <button
  471. onClick={() => { setSearch(''); resetPage(); }}
  472. className="absolute right-3 top-1/2 -translate-y-1/2 text-bambu-gray hover:text-white"
  473. >
  474. <X className="w-4 h-4" />
  475. </button>
  476. )}
  477. </div>
  478. <div className="flex items-center gap-2">
  479. {/* Columns button */}
  480. <button
  481. onClick={() => setShowColumnModal(true)}
  482. 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"
  483. title={t('inventory.configureColumns')}
  484. >
  485. <Columns className="w-4 h-4" />
  486. <span className="hidden sm:inline">{t('inventory.columns')}</span>
  487. </button>
  488. {/* Table / Cards toggle */}
  489. <div className="flex bg-bambu-dark-primary border border-bambu-dark-tertiary rounded-lg overflow-hidden">
  490. <button
  491. onClick={() => setViewMode('table')}
  492. className={`flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium transition-colors ${
  493. viewMode === 'table'
  494. ? 'bg-bambu-green text-white'
  495. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  496. }`}
  497. >
  498. <TableProperties className="w-4 h-4" />
  499. <span className="hidden sm:inline">{t('inventory.table')}</span>
  500. </button>
  501. <button
  502. onClick={() => setViewMode('cards')}
  503. className={`flex items-center gap-1.5 px-3 py-1.5 text-sm font-medium transition-colors ${
  504. viewMode === 'cards'
  505. ? 'bg-bambu-green text-white'
  506. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  507. }`}
  508. >
  509. <LayoutGrid className="w-4 h-4" />
  510. <span className="hidden sm:inline">{t('inventory.cards')}</span>
  511. </button>
  512. </div>
  513. </div>
  514. </div>
  515. {/* Filter chips row */}
  516. <div className="flex flex-wrap items-center gap-2">
  517. {/* Active / Archived chips */}
  518. <div className="flex items-center rounded-lg border border-bambu-dark-tertiary overflow-hidden">
  519. <button
  520. onClick={() => { setArchiveFilter('active'); resetPage(); }}
  521. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  522. archiveFilter === 'active'
  523. ? 'bg-bambu-green/20 text-bambu-green'
  524. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  525. }`}
  526. >
  527. <Package className="w-3.5 h-3.5" />
  528. {t('inventory.active')}
  529. </button>
  530. <button
  531. onClick={() => { setArchiveFilter('archived'); resetPage(); }}
  532. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  533. archiveFilter === 'archived'
  534. ? 'bg-bambu-green/20 text-bambu-green'
  535. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  536. }`}
  537. >
  538. <Archive className="w-3.5 h-3.5" />
  539. {t('inventory.archived')}
  540. </button>
  541. </div>
  542. <div className="w-px h-5 bg-bambu-dark-tertiary" />
  543. {/* All / Used / New chips */}
  544. <div className="flex items-center rounded-lg border border-bambu-dark-tertiary overflow-hidden">
  545. <button
  546. onClick={() => { setUsageFilter('all'); resetPage(); }}
  547. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  548. usageFilter === 'all'
  549. ? 'bg-bambu-green/20 text-bambu-green'
  550. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  551. }`}
  552. >
  553. {t('inventory.all')}
  554. </button>
  555. <button
  556. onClick={() => { setUsageFilter('used'); resetPage(); }}
  557. className={`flex items-center gap-1.5 px-3 py-1.5 text-xs font-medium transition-colors ${
  558. usageFilter === 'used'
  559. ? 'bg-bambu-green/20 text-bambu-green'
  560. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  561. }`}
  562. >
  563. <Clock className="w-3.5 h-3.5" />
  564. {t('inventory.used')}
  565. </button>
  566. <button
  567. onClick={() => { setUsageFilter('new'); resetPage(); }}
  568. className={`px-3 py-1.5 text-xs font-medium transition-colors ${
  569. usageFilter === 'new'
  570. ? 'bg-bambu-green/20 text-bambu-green'
  571. : 'text-bambu-gray hover:bg-bambu-dark-tertiary'
  572. }`}
  573. >
  574. {t('inventory.new')}
  575. </button>
  576. </div>
  577. <div className="w-px h-5 bg-bambu-dark-tertiary" />
  578. {/* Material dropdown chip */}
  579. <select
  580. value={materialFilter}
  581. onChange={(e) => { setMaterialFilter(e.target.value); resetPage(); }}
  582. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  583. materialFilter
  584. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  585. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  586. }`}
  587. >
  588. <option value="">{t('inventory.material')}</option>
  589. {uniqueMaterials.map((m) => (
  590. <option key={m} value={m}>{m}</option>
  591. ))}
  592. </select>
  593. {/* Brand dropdown chip */}
  594. <select
  595. value={brandFilter}
  596. onChange={(e) => { setBrandFilter(e.target.value); resetPage(); }}
  597. className={`px-3 py-1.5 rounded-lg border text-xs font-medium transition-colors cursor-pointer focus:outline-none ${
  598. brandFilter
  599. ? 'bg-bambu-green/20 text-bambu-green border-bambu-green/30'
  600. : 'bg-transparent text-bambu-gray border-bambu-dark-tertiary hover:bg-bambu-dark-tertiary'
  601. }`}
  602. >
  603. <option value="">{t('inventory.brand')}</option>
  604. {uniqueBrands.map((b) => (
  605. <option key={b} value={b}>{b}</option>
  606. ))}
  607. </select>
  608. {/* Clear filters */}
  609. {hasActiveFilters && (
  610. <>
  611. <div className="w-px h-5 bg-bambu-dark-tertiary" />
  612. <button
  613. onClick={clearAllFilters}
  614. className="flex items-center gap-1 text-xs text-bambu-gray hover:text-bambu-green transition-colors"
  615. >
  616. <X className="w-3.5 h-3.5" />
  617. {t('inventory.clearFilters')}
  618. </button>
  619. </>
  620. )}
  621. {/* Results count */}
  622. <span className="ml-auto text-xs text-bambu-gray">
  623. {filteredSpools.length} {filteredSpools.length !== 1 ? t('inventory.spools') : t('inventory.spool')}
  624. </span>
  625. </div>
  626. {/* Content */}
  627. {isLoading ? (
  628. <div className="flex justify-center py-16">
  629. <Loader2 className="w-8 h-8 text-bambu-green animate-spin" />
  630. </div>
  631. ) : viewMode === 'cards' ? (
  632. /* Cards view */
  633. pagedSpools.length > 0 ? (
  634. <>
  635. <div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 xl:grid-cols-4 gap-4">
  636. {pagedSpools.map((spool) => {
  637. const remaining = Math.max(0, spool.label_weight - spool.weight_used);
  638. const pct = spool.label_weight > 0 ? (remaining / spool.label_weight) * 100 : 0;
  639. const colorStyle = spool.rgba ? `#${spool.rgba.substring(0, 6)}` : '#808080';
  640. return (
  641. <div
  642. key={spool.id}
  643. 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' : ''}`}
  644. onClick={() => setFormModal({ spool })}
  645. >
  646. {/* Color header */}
  647. <div className="h-14 flex items-center justify-center" style={{ backgroundColor: colorStyle }}>
  648. <span className="bg-white/90 text-gray-800 px-3 py-0.5 rounded-full text-sm font-medium">
  649. {spool.color_name || '-'}
  650. </span>
  651. </div>
  652. {/* Content */}
  653. <div className="p-4 space-y-3">
  654. <div className="flex items-start justify-between gap-2">
  655. <div>
  656. <h3 className="font-semibold text-white">{spool.material}{spool.subtype ? ` ${spool.subtype}` : ''}</h3>
  657. <p className="text-sm text-bambu-gray">{spool.brand || '-'}</p>
  658. </div>
  659. <span className="text-xs font-mono text-bambu-gray bg-bambu-dark-tertiary px-2 py-1 rounded">#{spool.id}</span>
  660. </div>
  661. {/* Progress */}
  662. <div>
  663. <div className="flex justify-between text-xs text-bambu-gray mb-1">
  664. <span>{t('inventory.remaining')}</span>
  665. <span>{Math.round(pct)}%</span>
  666. </div>
  667. <div className="flex items-center gap-2">
  668. <div className="flex-1 h-2 bg-bambu-dark-tertiary rounded-full overflow-hidden">
  669. <div
  670. className={`h-full rounded-full ${pct > 50 ? 'bg-bambu-green' : pct > 20 ? 'bg-yellow-500' : 'bg-red-500'}`}
  671. style={{ width: `${Math.min(pct, 100)}%` }}
  672. />
  673. </div>
  674. <span className="text-xs text-bambu-gray min-w-[40px] text-right">{Math.round(remaining)}g</span>
  675. </div>
  676. </div>
  677. {/* Weight info */}
  678. <div className="grid grid-cols-2 gap-2 text-xs">
  679. <div>
  680. <span className="text-bambu-gray/60">{t('inventory.labelWeight')}: </span>
  681. <span className="text-bambu-gray">{formatWeight(spool.label_weight)}</span>
  682. </div>
  683. <div>
  684. <span className="text-bambu-gray/60">{t('inventory.weightUsed')}: </span>
  685. <span className="text-bambu-gray">{spool.weight_used > 0 ? formatWeight(spool.weight_used) : '-'}</span>
  686. </div>
  687. </div>
  688. {/* Note */}
  689. {spool.note && (
  690. <div className="text-xs text-bambu-gray/60 pt-2 border-t border-bambu-dark-tertiary truncate" title={spool.note}>
  691. {spool.note}
  692. </div>
  693. )}
  694. </div>
  695. </div>
  696. );
  697. })}
  698. </div>
  699. {/* Pagination for cards */}
  700. <PaginationBar
  701. pageIndex={safePageIndex}
  702. pageSize={pageSize}
  703. totalRows={filteredSpools.length}
  704. totalPages={totalPages}
  705. onPageChange={setPageIndex}
  706. onPageSizeChange={(size) => { setPageSize(size); resetPage(); }}
  707. t={t}
  708. />
  709. </>
  710. ) : (
  711. <EmptyFilterState
  712. hasFilters={hasActiveFilters}
  713. onAddSpool={() => setFormModal({ spool: null })}
  714. t={t}
  715. />
  716. )
  717. ) : (
  718. /* Table view */
  719. pagedSpools.length > 0 ? (
  720. <div className="bg-bambu-dark-secondary rounded-lg overflow-hidden border border-bambu-dark-tertiary">
  721. <div className="overflow-x-auto">
  722. <table className="w-full">
  723. <thead>
  724. <tr className="border-b border-bambu-dark-tertiary bg-bambu-dark-tertiary/30">
  725. {visibleColumns.map((colId) => (
  726. <th
  727. key={colId}
  728. className={`text-left py-3 px-4 text-xs font-medium text-bambu-gray uppercase tracking-wide ${colId === 'remaining' ? 'min-w-[150px]' : ''}`}
  729. >
  730. {columnHeaders[colId]?.(t) ?? colId}
  731. </th>
  732. ))}
  733. <th className="text-right py-3 px-4 text-xs font-medium text-bambu-gray uppercase tracking-wide">{t('common.actions')}</th>
  734. </tr>
  735. </thead>
  736. <tbody>
  737. {pagedSpools.map((spool) => {
  738. const remaining = Math.max(0, spool.label_weight - spool.weight_used);
  739. const pct = spool.label_weight > 0 ? (remaining / spool.label_weight) * 100 : 0;
  740. return (
  741. <tr
  742. key={spool.id}
  743. className={`border-b border-bambu-dark-tertiary/50 hover:bg-bambu-dark-tertiary/30 transition-colors cursor-pointer ${
  744. spool.archived_at ? 'opacity-50' : ''
  745. }`}
  746. onClick={() => setFormModal({ spool })}
  747. >
  748. {visibleColumns.map((colId) => (
  749. <td key={colId} className="py-3 px-4">
  750. {columnCells[colId]?.({ spool, remaining, pct, assignmentMap })}
  751. </td>
  752. ))}
  753. <td className="py-3 px-4">
  754. <div className="flex items-center justify-end gap-1" onClick={(e) => e.stopPropagation()}>
  755. <button
  756. onClick={() => setFormModal({ spool })}
  757. className="p-1.5 text-bambu-gray hover:text-white rounded transition-colors"
  758. title={t('inventory.editSpool')}
  759. >
  760. <Edit2 className="w-4 h-4" />
  761. </button>
  762. {spool.archived_at ? (
  763. <button
  764. onClick={() => restoreMutation.mutate(spool.id)}
  765. className="p-1.5 text-bambu-gray hover:text-bambu-green rounded transition-colors"
  766. title={t('inventory.restore')}
  767. >
  768. <RotateCcw className="w-4 h-4" />
  769. </button>
  770. ) : (
  771. <button
  772. onClick={() => archiveMutation.mutate(spool.id)}
  773. className="p-1.5 text-bambu-gray hover:text-yellow-400 rounded transition-colors"
  774. title={t('inventory.archive')}
  775. >
  776. <Archive className="w-4 h-4" />
  777. </button>
  778. )}
  779. <button
  780. onClick={() => {
  781. if (confirm(t('inventory.deleteConfirm'))) {
  782. deleteMutation.mutate(spool.id);
  783. }
  784. }}
  785. className="p-1.5 text-bambu-gray hover:text-red-400 rounded transition-colors"
  786. title={t('common.delete')}
  787. >
  788. <Trash2 className="w-4 h-4" />
  789. </button>
  790. </div>
  791. </td>
  792. </tr>
  793. );
  794. })}
  795. </tbody>
  796. </table>
  797. </div>
  798. {/* Pagination inside card footer */}
  799. <div className="flex items-center justify-between px-4 py-3 bg-bambu-dark-tertiary/50 border-t border-bambu-dark-tertiary text-sm">
  800. <span className="text-bambu-gray">
  801. {t('inventory.showing')} {safePageIndex * pageSize + 1} {t('inventory.to')}{' '}
  802. {Math.min((safePageIndex + 1) * pageSize, filteredSpools.length)}{' '}
  803. {t('inventory.of')} {filteredSpools.length} {t('inventory.spools')}
  804. </span>
  805. <div className="flex items-center gap-2">
  806. <span className="text-bambu-gray">{t('inventory.show')}</span>
  807. <select
  808. value={pageSize}
  809. onChange={(e) => { setPageSize(Number(e.target.value)); resetPage(); }}
  810. 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"
  811. >
  812. {[15, 30, 50, 100].map((n) => (
  813. <option key={n} value={n}>{n}</option>
  814. ))}
  815. </select>
  816. <button
  817. onClick={() => setPageIndex(0)}
  818. disabled={safePageIndex === 0}
  819. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  820. title="First page"
  821. >
  822. <ChevronsLeft className="w-4 h-4" />
  823. </button>
  824. <button
  825. onClick={() => setPageIndex((p) => Math.max(0, p - 1))}
  826. disabled={safePageIndex === 0}
  827. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  828. >
  829. <ChevronLeft className="w-4 h-4" />
  830. </button>
  831. <span className="text-bambu-gray px-2 whitespace-nowrap">
  832. {t('inventory.page')} {safePageIndex + 1} {t('inventory.of')} {totalPages}
  833. </span>
  834. <button
  835. onClick={() => setPageIndex((p) => Math.min(totalPages - 1, p + 1))}
  836. disabled={safePageIndex >= totalPages - 1}
  837. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  838. >
  839. <ChevronRight className="w-4 h-4" />
  840. </button>
  841. <button
  842. onClick={() => setPageIndex(totalPages - 1)}
  843. disabled={safePageIndex >= totalPages - 1}
  844. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  845. title="Last page"
  846. >
  847. <ChevronsRight className="w-4 h-4" />
  848. </button>
  849. </div>
  850. </div>
  851. </div>
  852. ) : (
  853. <EmptyFilterState
  854. hasFilters={hasActiveFilters}
  855. onAddSpool={() => setFormModal({ spool: null })}
  856. t={t}
  857. />
  858. )
  859. )}
  860. {/* Spool Form Modal */}
  861. {formModal !== null && (
  862. <SpoolFormModal
  863. isOpen={true}
  864. onClose={() => setFormModal(null)}
  865. spool={formModal.spool}
  866. />
  867. )}
  868. {/* Column Config Modal */}
  869. <ColumnConfigModal
  870. isOpen={showColumnModal}
  871. onClose={() => setShowColumnModal(false)}
  872. columns={columnConfig}
  873. defaultColumns={DEFAULT_COLUMNS}
  874. onSave={handleColumnConfigSave}
  875. />
  876. </div>
  877. );
  878. }
  879. /* Pagination bar (reused for cards view) */
  880. function PaginationBar({
  881. pageIndex, pageSize, totalRows, totalPages, onPageChange, onPageSizeChange, t,
  882. }: {
  883. pageIndex: number;
  884. pageSize: number;
  885. totalRows: number;
  886. totalPages: number;
  887. onPageChange: (page: number) => void;
  888. onPageSizeChange: (size: number) => void;
  889. t: (key: string) => string;
  890. }) {
  891. if (totalPages <= 1) return null;
  892. return (
  893. <div className="flex items-center justify-between pt-2 text-sm">
  894. <span className="text-bambu-gray">
  895. {t('inventory.showing')} {pageIndex * pageSize + 1} {t('inventory.to')}{' '}
  896. {Math.min((pageIndex + 1) * pageSize, totalRows)}{' '}
  897. {t('inventory.of')} {totalRows} {t('inventory.spools')}
  898. </span>
  899. <div className="flex items-center gap-2">
  900. <span className="text-bambu-gray">{t('inventory.show')}</span>
  901. <select
  902. value={pageSize}
  903. onChange={(e) => onPageSizeChange(Number(e.target.value))}
  904. 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"
  905. >
  906. {[15, 30, 50, 100].map((n) => (
  907. <option key={n} value={n}>{n}</option>
  908. ))}
  909. </select>
  910. <button
  911. onClick={() => onPageChange(0)}
  912. disabled={pageIndex === 0}
  913. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  914. >
  915. <ChevronsLeft className="w-4 h-4" />
  916. </button>
  917. <button
  918. onClick={() => onPageChange(Math.max(0, pageIndex - 1))}
  919. disabled={pageIndex === 0}
  920. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  921. >
  922. <ChevronLeft className="w-4 h-4" />
  923. </button>
  924. <span className="text-bambu-gray px-2 whitespace-nowrap">
  925. {t('inventory.page')} {pageIndex + 1} {t('inventory.of')} {totalPages}
  926. </span>
  927. <button
  928. onClick={() => onPageChange(Math.min(totalPages - 1, pageIndex + 1))}
  929. disabled={pageIndex >= totalPages - 1}
  930. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  931. >
  932. <ChevronRight className="w-4 h-4" />
  933. </button>
  934. <button
  935. onClick={() => onPageChange(totalPages - 1)}
  936. disabled={pageIndex >= totalPages - 1}
  937. className="p-1.5 rounded text-bambu-gray hover:text-white disabled:opacity-30 disabled:cursor-not-allowed transition-colors"
  938. >
  939. <ChevronsRight className="w-4 h-4" />
  940. </button>
  941. </div>
  942. </div>
  943. );
  944. }
  945. /* Empty state matching SpoolBuddy's design */
  946. function EmptyFilterState({
  947. hasFilters,
  948. onAddSpool,
  949. t,
  950. }: {
  951. hasFilters: boolean;
  952. onAddSpool: () => void;
  953. t: (key: string) => string;
  954. }) {
  955. return (
  956. <div className="flex flex-col items-center justify-center py-16 px-4">
  957. <div className="relative mb-6">
  958. <div className="absolute inset-0 -m-4 bg-bambu-green/5 rounded-full blur-2xl" />
  959. <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">
  960. <div className="absolute -top-1 -right-1 w-3 h-3 rounded-full bg-bambu-green/30" />
  961. <div className="absolute -bottom-2 -left-2 w-2 h-2 rounded-full bg-bambu-green/20" />
  962. {hasFilters ? (
  963. <Search className="w-10 h-10 text-bambu-gray/40" strokeWidth={1.5} />
  964. ) : (
  965. <div className="relative">
  966. <div className="w-14 h-14 rounded-full border-4 border-bambu-gray/20 flex items-center justify-center">
  967. <div className="w-6 h-6 rounded-full bg-bambu-gray/10 border-2 border-bambu-gray/20" />
  968. </div>
  969. <div className="absolute -bottom-1 -right-1 w-6 h-6 rounded-full bg-bambu-green flex items-center justify-center shadow-md">
  970. <span className="text-white text-lg font-bold leading-none">+</span>
  971. </div>
  972. </div>
  973. )}
  974. </div>
  975. </div>
  976. <h3 className="text-lg font-semibold text-white mb-2 text-center">
  977. {hasFilters ? t('inventory.noSpoolsMatch') : t('inventory.noSpools').split('.')[0]}
  978. </h3>
  979. <p className="text-sm text-bambu-gray text-center max-w-sm mb-6">
  980. {hasFilters
  981. ? t('inventory.noSpoolsMatchDesc')
  982. : t('inventory.noSpools')
  983. }
  984. </p>
  985. {!hasFilters && (
  986. <Button onClick={onAddSpool}>
  987. <Package className="w-4 h-4" />
  988. {t('inventory.addSpool')}
  989. </Button>
  990. )}
  991. </div>
  992. );
  993. }