SpoolFormModal.tsx 38 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962
  1. import { useState, useEffect, useMemo } from 'react';
  2. import { useQuery, useMutation, useQueryClient } from '@tanstack/react-query';
  3. import { useTranslation } from 'react-i18next';
  4. import { X, Loader2, Save, Beaker, Palette, Zap, Tag, Unlink } from 'lucide-react';
  5. import { api, ApiError } from '../api/client';
  6. import type { InventorySpool, SlicerSetting, SpoolCatalogEntry, LocalPreset, BuiltinFilament, SpoolmanBulkCreateResult, SpoolKProfileInput, SpoolmanFilamentEntry } from '../api/client';
  7. import { Button } from './Button';
  8. import { useToast } from '../contexts/ToastContext';
  9. import type { SpoolFormData, PrinterWithCalibrations, ColorPreset } from './spool-form/types';
  10. import { defaultFormData, validateForm, SPOOLMAN_LINKED_FIELDS } from './spool-form/types';
  11. import { buildFilamentOptions, extractBrandsFromPresets, findPresetOption, loadRecentColors, parsePresetName, saveRecentColor } from './spool-form/utils';
  12. import { MATERIALS } from './spool-form/constants';
  13. import { FilamentSection } from './spool-form/FilamentSection';
  14. import { ColorSection } from './spool-form/ColorSection';
  15. import { AdditionalSection } from './spool-form/AdditionalSection';
  16. import { SpoolmanFilamentPicker } from './spool-form/SpoolmanFilamentPicker';
  17. import { PAProfileSection } from './spool-form/PAProfileSection';
  18. import { SpoolUsageHistory } from './SpoolUsageHistory';
  19. type TabId = 'filament' | 'pa-profile';
  20. const CLEAR_TAG_PAYLOAD = { tag_uid: null, tray_uuid: null, tag_type: null, data_origin: null };
  21. export type SpoolFormMode = 'create' | 'edit' | 'copy';
  22. interface SpoolFormModalProps {
  23. isOpen: boolean;
  24. onClose: () => void;
  25. spool?: InventorySpool | null;
  26. mode: SpoolFormMode;
  27. printersWithCalibrations?: PrinterWithCalibrations[];
  28. currencySymbol: string;
  29. onSpoolsCreated?: (spools: InventorySpool[]) => void;
  30. /** When true, CRUD operations target the Spoolman inventory proxy endpoints. */
  31. spoolmanMode?: boolean;
  32. /** Query key to invalidate after mutations (differs for Spoolman vs local). */
  33. spoolsQueryKey?: string[];
  34. }
  35. export function SpoolFormModal({
  36. isOpen,
  37. onClose,
  38. spool,
  39. mode,
  40. printersWithCalibrations = [],
  41. currencySymbol,
  42. onSpoolsCreated,
  43. spoolmanMode = false,
  44. spoolsQueryKey = ['inventory-spools'],
  45. }: SpoolFormModalProps) {
  46. const { t } = useTranslation();
  47. const queryClient = useQueryClient();
  48. const { showToast } = useToast();
  49. const isEditing = mode === 'edit';
  50. const isCopying = mode === 'copy';
  51. // Form state
  52. const [formData, setFormData] = useState<SpoolFormData>(defaultFormData);
  53. const [errors, setErrors] = useState<Partial<Record<keyof SpoolFormData, string>>>({});
  54. const [activeTab, setActiveTab] = useState<TabId>('filament');
  55. const [weightTouched, setWeightTouched] = useState(false);
  56. const [storageLocationTouched, setStorageLocationTouched] = useState(false);
  57. const [quickAdd, setQuickAdd] = useState(false);
  58. const [quantity, setQuantity] = useState(1);
  59. // Cloud presets
  60. const [cloudAuthenticated, setCloudAuthenticated] = useState(false);
  61. const [loadingCloudPresets, setLoadingCloudPresets] = useState(false);
  62. const [cloudPresets, setCloudPresets] = useState<SlicerSetting[]>([]);
  63. const [presetInputValue, setPresetInputValue] = useState('');
  64. // Spool catalog
  65. const [spoolCatalog, setSpoolCatalog] = useState<SpoolCatalogEntry[]>([]);
  66. // Local presets (OrcaSlicer imports)
  67. const [localPresets, setLocalPresets] = useState<LocalPreset[]>([]);
  68. // Built-in filaments (static fallback)
  69. const [builtinFilaments, setBuiltinFilaments] = useState<BuiltinFilament[]>([]);
  70. // Color catalog
  71. const [colorCatalog, setColorCatalog] = useState<{
  72. manufacturer: string;
  73. color_name: string;
  74. hex_color: string;
  75. material: string | null;
  76. // #1340: gradient + effect carried from the catalog entry through to the
  77. // color picker so they're applied alongside hex + name on selection.
  78. extra_colors?: string | null;
  79. effect_type?: string | null;
  80. }[]>([]);
  81. // Color state
  82. const [recentColors, setRecentColors] = useState<ColorPreset[]>([]);
  83. // PA Profile state
  84. const [fetchedCalibrations, setFetchedCalibrations] = useState<PrinterWithCalibrations[]>([]);
  85. const [selectedProfiles, setSelectedProfiles] = useState<Set<string>>(new Set());
  86. const [expandedPrinters, setExpandedPrinters] = useState<Set<string>>(new Set());
  87. // Use prop if provided, otherwise use self-fetched data
  88. const resolvedCalibrations = printersWithCalibrations.length > 0
  89. ? printersWithCalibrations
  90. : fetchedCalibrations;
  91. // Count selected PA profiles for tab badge
  92. const selectedProfileCount = selectedProfiles.size;
  93. // Fetch Spoolman filament catalog when in Spoolman mode
  94. // retry:false — Spoolman may be intentionally disabled (400); don't flood the server
  95. const { data: spoolmanFilaments = [], isLoading: isLoadingFilaments, error: filamentsError } = useQuery<SpoolmanFilamentEntry[], Error>({
  96. queryKey: ['spoolman-inventory-filaments'],
  97. queryFn: () => api.getSpoolmanInventoryFilaments(),
  98. enabled: spoolmanMode && isOpen,
  99. staleTime: 60_000,
  100. retry: false,
  101. });
  102. // Load recent colors on mount
  103. useEffect(() => {
  104. setRecentColors(loadRecentColors());
  105. }, []);
  106. // Fetch cloud presets and catalog when modal opens
  107. useEffect(() => {
  108. if (isOpen) {
  109. const fetchData = async () => {
  110. setLoadingCloudPresets(true);
  111. try {
  112. const status = await api.getCloudStatus();
  113. setCloudAuthenticated(status.is_authenticated);
  114. if (status.is_authenticated) {
  115. const presets = await api.getFilamentPresets();
  116. setCloudPresets(presets);
  117. }
  118. } catch (e) {
  119. console.error('Failed to fetch cloud presets:', e);
  120. setCloudAuthenticated(false);
  121. } finally {
  122. setLoadingCloudPresets(false);
  123. }
  124. };
  125. fetchData();
  126. if (!spoolmanMode) {
  127. api.getSpoolCatalog().then(setSpoolCatalog).catch(console.error);
  128. }
  129. api.getColorCatalog().then(setColorCatalog).catch(console.error);
  130. api.getLocalPresets().then(r => setLocalPresets(r.filament)).catch(console.error);
  131. api.getBuiltinFilaments().then(setBuiltinFilaments).catch(console.error);
  132. // Fetch printer calibrations if not provided via props
  133. if (printersWithCalibrations.length === 0) {
  134. (async () => {
  135. try {
  136. const printers = await api.getPrinters();
  137. const statuses = await Promise.all(
  138. printers.map(p => api.getPrinterStatus(p.id).catch(() => null)),
  139. );
  140. const results: PrinterWithCalibrations[] = [];
  141. for (let i = 0; i < printers.length; i++) {
  142. const printer = printers[i];
  143. const status = statuses[i];
  144. const connected = status?.connected ?? false;
  145. let calibrations: PrinterWithCalibrations['calibrations'] = [];
  146. if (connected) {
  147. try {
  148. const kRes = await api.getKProfiles(printer.id);
  149. calibrations = kRes.profiles.map(p => ({
  150. cali_idx: p.slot_id,
  151. filament_id: p.filament_id,
  152. setting_id: p.setting_id || '',
  153. name: p.name,
  154. k_value: parseFloat(p.k_value) || 0,
  155. n_coef: parseFloat(p.n_coef) || 0,
  156. extruder_id: p.extruder_id,
  157. nozzle_diameter: p.nozzle_diameter,
  158. }));
  159. } catch {
  160. // Printer may not support K-profiles
  161. }
  162. }
  163. results.push({ printer: { ...printer, connected }, calibrations });
  164. }
  165. setFetchedCalibrations(results);
  166. } catch (e) {
  167. console.error('Failed to fetch printer calibrations:', e);
  168. }
  169. })();
  170. }
  171. }
  172. // The effect intentionally depends only on `isOpen` (and the prop-side
  173. // calibration count) — re-running on every spoolmanMode toggle would
  174. // race the in-flight async fetches with unmount/teardown and emit
  175. // "test environment was torn down" errors in vitest. spoolmanMode only
  176. // gates a single fetch (getSpoolCatalog) which is cheap enough to skip
  177. // when the modal opens in Spoolman mode.
  178. // eslint-disable-next-line react-hooks/exhaustive-deps
  179. }, [isOpen, printersWithCalibrations.length]);
  180. // Build filament options: cloud → local → fallback
  181. const filamentOptions = useMemo(
  182. () => buildFilamentOptions(cloudPresets, new Set(), localPresets, builtinFilaments),
  183. [cloudPresets, localPresets, builtinFilaments],
  184. );
  185. // Extract brands from presets
  186. const baseAvailableBrands = useMemo(() => {
  187. const presetBrands = extractBrandsFromPresets(cloudPresets, localPresets);
  188. const catalogBrands = colorCatalog
  189. .map(entry => entry.manufacturer?.trim())
  190. .filter((brand): brand is string => !!brand);
  191. const brandSet = new Set<string>([...presetBrands, ...catalogBrands]);
  192. return Array.from(brandSet).sort((a, b) => a.localeCompare(b));
  193. }, [cloudPresets, localPresets, colorCatalog]);
  194. const baseAvailableMaterials = useMemo(() => {
  195. const catalogMaterials = colorCatalog
  196. .map(entry => entry.material?.trim())
  197. .filter((material): material is string => !!material);
  198. const materialSet = new Set<string>([...MATERIALS, ...catalogMaterials]);
  199. return Array.from(materialSet).sort((a, b) => a.localeCompare(b));
  200. }, [colorCatalog]);
  201. const brandMaterialPairs = useMemo(() => {
  202. const pairs: Array<{ brand: string; material: string }> = [];
  203. for (const entry of colorCatalog) {
  204. const brand = entry.manufacturer?.trim();
  205. const material = entry.material?.trim();
  206. if (brand && material) pairs.push({ brand, material });
  207. }
  208. for (const preset of cloudPresets) {
  209. const parsed = parsePresetName(preset.name);
  210. if (parsed.brand && parsed.material) {
  211. pairs.push({ brand: parsed.brand, material: parsed.material });
  212. }
  213. }
  214. for (const preset of localPresets) {
  215. const parsed = parsePresetName(preset.name);
  216. const brand = preset.filament_vendor?.trim() || parsed.brand;
  217. const material = parsed.material;
  218. if (brand && material) {
  219. pairs.push({ brand, material });
  220. }
  221. }
  222. return pairs;
  223. }, [cloudPresets, colorCatalog, localPresets]);
  224. const brandToMaterials = useMemo(() => {
  225. const map = new Map<string, Set<string>>();
  226. for (const pair of brandMaterialPairs) {
  227. const brandKey = pair.brand.toLowerCase();
  228. const materialKey = pair.material.toLowerCase();
  229. if (!map.has(brandKey)) map.set(brandKey, new Set());
  230. map.get(brandKey)!.add(materialKey);
  231. }
  232. return map;
  233. }, [brandMaterialPairs]);
  234. const materialToBrands = useMemo(() => {
  235. const map = new Map<string, Set<string>>();
  236. for (const pair of brandMaterialPairs) {
  237. const brandKey = pair.brand.toLowerCase();
  238. const materialKey = pair.material.toLowerCase();
  239. if (!map.has(materialKey)) map.set(materialKey, new Set());
  240. map.get(materialKey)!.add(brandKey);
  241. }
  242. return map;
  243. }, [brandMaterialPairs]);
  244. const availableBrands = useMemo(() => {
  245. if (!formData.material) return baseAvailableBrands;
  246. const materialKey = formData.material.toLowerCase();
  247. const brandKeys = materialToBrands.get(materialKey);
  248. if (!brandKeys || brandKeys.size === 0) return baseAvailableBrands;
  249. return baseAvailableBrands.filter(brand => brandKeys.has(brand.toLowerCase()));
  250. }, [baseAvailableBrands, formData.material, materialToBrands]);
  251. const availableMaterials = useMemo(() => {
  252. if (!formData.brand) return baseAvailableMaterials;
  253. const brandKey = formData.brand.toLowerCase();
  254. const materialKeys = brandToMaterials.get(brandKey);
  255. if (!materialKeys || materialKeys.size === 0) return baseAvailableMaterials;
  256. return baseAvailableMaterials.filter(material => materialKeys.has(material.toLowerCase()));
  257. }, [baseAvailableMaterials, formData.brand, brandToMaterials]);
  258. // Find selected preset option
  259. const selectedPresetOption = useMemo(
  260. () => findPresetOption(formData.slicer_filament, filamentOptions),
  261. [formData.slicer_filament, filamentOptions],
  262. );
  263. // Reset form when modal opens/closes or spool changes
  264. useEffect(() => {
  265. if (isOpen) {
  266. if (spool) {
  267. // Legacy rows may carry a malformed rgba (e.g. the 7-char 'FFFFFFF'
  268. // from #1055 before the create/update pattern was enforced). The
  269. // backend SpoolUpdate schema rejects non-8-char hex on PATCH, so
  270. // re-submitting a malformed value would 422 every edit on that spool
  271. // — even edits that don't touch color. Normalize on load: any value
  272. // that isn't exactly 8 hex chars falls back to the default, so the
  273. // user can save unrelated fields (weight, material, note) without
  274. // first being forced to fix a color they may not even be aware is
  275. // broken. Saving also purges the bad value from the DB.
  276. const validRgba = spool.rgba && /^[0-9A-Fa-f]{8}$/.test(spool.rgba) ? spool.rgba : '808080FF';
  277. setFormData({
  278. material: spool.material || '',
  279. subtype: spool.subtype || '',
  280. brand: spool.brand || '',
  281. // #1319: leave color_name blank when the backend reports it was
  282. // synthesised from subtype — otherwise the form would round-trip
  283. // the synth value to Spoolman on save as if the user had set it,
  284. // which is what produced the "color reverts to subtype" symptom.
  285. color_name: spool.color_name_is_synthesized ? '' : (spool.color_name || ''),
  286. rgba: validRgba,
  287. extra_colors: spool.extra_colors || '',
  288. effect_type: spool.effect_type || '',
  289. label_weight: spool.label_weight || 1000,
  290. core_weight: spool.core_weight || 250,
  291. core_weight_catalog_id: spool.core_weight_catalog_id ?? null,
  292. weight_used: isCopying ? 0 : spool.weight_used || 0,
  293. slicer_filament: spool.slicer_filament || '',
  294. note: spool.note || '',
  295. cost_per_kg: spool.cost_per_kg ?? null,
  296. category: spool.category || '',
  297. low_stock_threshold_pct: spool.low_stock_threshold_pct ?? null,
  298. storage_location: spool.storage_location || '',
  299. spoolman_filament_id: null,
  300. });
  301. setPresetInputValue(spool.slicer_filament_name || spool.slicer_filament || '');
  302. // Load K-profiles for this spool
  303. if (spool.k_profiles && spool.k_profiles.length > 0) {
  304. const profileKeys = new Set<string>();
  305. for (const p of spool.k_profiles) {
  306. if (p.cali_idx !== null && p.cali_idx !== undefined) {
  307. profileKeys.add(`${p.printer_id}:${p.cali_idx}:${p.extruder ?? 'null'}`);
  308. }
  309. }
  310. setSelectedProfiles(profileKeys);
  311. } else {
  312. setSelectedProfiles(new Set());
  313. }
  314. } else {
  315. setFormData(defaultFormData);
  316. setPresetInputValue('');
  317. setSelectedProfiles(new Set());
  318. setQuickAdd(false);
  319. setQuantity(1);
  320. }
  321. setErrors({});
  322. setActiveTab('filament');
  323. setWeightTouched(false);
  324. setStorageLocationTouched(false);
  325. }
  326. }, [isOpen, spool, mode, isCopying]);
  327. // Expand all printers in PA profile section when calibrations are available
  328. useEffect(() => {
  329. if (isOpen && resolvedCalibrations.length > 0) {
  330. setExpandedPrinters(new Set(resolvedCalibrations.map(p => String(p.printer.id))));
  331. }
  332. }, [isOpen, resolvedCalibrations]);
  333. // Update field helper
  334. const updateField = <K extends keyof SpoolFormData>(key: K, value: SpoolFormData[K]) => {
  335. const isLinkedField = SPOOLMAN_LINKED_FIELDS.has(key);
  336. if (spoolmanMode && isLinkedField && formData.spoolman_filament_id !== null) {
  337. showToast(t('inventory.spoolmanFilamentUnlinked'), 'info');
  338. }
  339. setFormData(prev => ({
  340. ...prev,
  341. [key]: value,
  342. ...(spoolmanMode && isLinkedField && prev.spoolman_filament_id !== null
  343. ? { spoolman_filament_id: null }
  344. : {}),
  345. }));
  346. if (key === 'weight_used') setWeightTouched(true);
  347. if (key === 'storage_location') setStorageLocationTouched(true);
  348. if (errors[key]) {
  349. setErrors(prev => ({ ...prev, [key]: undefined }));
  350. }
  351. };
  352. // Prefill form from a Spoolman filament catalog entry
  353. // subtype extraction mirrors _spoolman_helpers.py logic
  354. const handleFilamentSelect = (filament: SpoolmanFilamentEntry) => {
  355. const material = filament.material || '';
  356. const name = filament.name || '';
  357. const subtype = material && name.startsWith(material) ? name.slice(material.length).trim() : name;
  358. const rawHex = (filament.color_hex ?? '').replace('#', '').toUpperCase();
  359. // Guard against short/malformed hex values — must be exactly 6 hex chars
  360. const colorHex = /^[0-9A-F]{6}$/.test(rawHex) ? rawHex : '808080';
  361. setFormData(prev => ({
  362. ...prev,
  363. spoolman_filament_id: filament.id,
  364. material,
  365. subtype,
  366. brand: filament.vendor?.name || '',
  367. rgba: `${colorHex}FF`,
  368. color_name: filament.color_name || '',
  369. label_weight: filament.weight ?? prev.label_weight,
  370. }));
  371. showToast(t('inventory.spoolmanFilamentSelected'), 'success');
  372. };
  373. // Handle color selection
  374. const handleColorUsed = (color: ColorPreset) => {
  375. setRecentColors(prev => saveRecentColor(color, prev));
  376. };
  377. // Mutations – dispatch to Spoolman proxy or local inventory based on mode
  378. const createMutation = useMutation({
  379. mutationFn: (data: Record<string, unknown>) =>
  380. spoolmanMode
  381. ? api.createSpoolmanInventorySpool(data as Parameters<typeof api.createSpoolmanInventorySpool>[0])
  382. : api.createSpool(data as Parameters<typeof api.createSpool>[0]),
  383. onSuccess: async (newSpool) => {
  384. if (newSpool?.id) {
  385. const ok = await saveKProfiles(newSpool.id);
  386. if (!ok) return;
  387. }
  388. await queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  389. if (onSpoolsCreated) onSpoolsCreated([newSpool]);
  390. showToast(t('inventory.spoolCreated'), 'success');
  391. onClose();
  392. },
  393. onError: (error: Error) => {
  394. if (error instanceof ApiError && error.status === 503) {
  395. showToast(t('inventory.spoolmanUnreachable'), 'error');
  396. } else {
  397. showToast(t('inventory.saveFailed'), 'error');
  398. }
  399. },
  400. });
  401. const bulkCreateMutation = useMutation<
  402. SpoolmanBulkCreateResult | InventorySpool[],
  403. Error,
  404. { data: Record<string, unknown>; qty: number }
  405. >({
  406. mutationFn: ({ data, qty }) =>
  407. spoolmanMode
  408. ? api.bulkCreateSpoolmanInventorySpools(data as Parameters<typeof api.bulkCreateSpoolmanInventorySpools>[0], qty)
  409. : api.bulkCreateSpools(data as Parameters<typeof api.bulkCreateSpools>[0], qty),
  410. onSuccess: async (result) => {
  411. // Spoolman bulk-create returns SpoolmanBulkCreateResult (207); local returns InventorySpool[].
  412. // Cast via unknown to satisfy strict TypeScript — the runtime shape is guaranteed by
  413. // the duck-type check ('created' in result) before any property access.
  414. const spoolmanResult = (spoolmanMode && 'created' in result)
  415. ? (result as unknown as SpoolmanBulkCreateResult)
  416. : null;
  417. const createdSpools: InventorySpool[] = spoolmanResult
  418. ? spoolmanResult.created
  419. : (result as InventorySpool[]);
  420. if (selectedProfiles.size > 0) {
  421. for (const s of createdSpools) {
  422. await saveKProfiles(s.id);
  423. }
  424. }
  425. await queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  426. if (onSpoolsCreated) onSpoolsCreated(createdSpools);
  427. if (spoolmanResult && spoolmanResult.failed_count > 0) {
  428. showToast(
  429. t('inventory.spoolsPartiallyCreated', {
  430. created: createdSpools.length,
  431. total: spoolmanResult.requested_count,
  432. }),
  433. 'warning',
  434. );
  435. } else {
  436. showToast(t('inventory.spoolsCreated', { count: createdSpools.length }), 'success');
  437. }
  438. onClose();
  439. },
  440. onError: (error: Error) => {
  441. if (error instanceof ApiError && error.status === 503) {
  442. showToast(t('inventory.spoolmanUnreachable'), 'error');
  443. } else {
  444. showToast(t('inventory.saveFailed'), 'error');
  445. }
  446. },
  447. });
  448. const updateMutation = useMutation({
  449. mutationFn: (data: Record<string, unknown>) =>
  450. spoolmanMode
  451. ? api.updateSpoolmanInventorySpool(spool!.id, data as Parameters<typeof api.updateSpoolmanInventorySpool>[1])
  452. : api.updateSpool(spool!.id, data as Parameters<typeof api.updateSpool>[1]),
  453. onSuccess: async () => {
  454. if (spool?.id) {
  455. const ok = await saveKProfiles(spool.id);
  456. if (!ok) return;
  457. }
  458. await queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  459. showToast(t('inventory.spoolUpdated'), 'success');
  460. onClose();
  461. },
  462. onError: (error: Error) => {
  463. if (error instanceof ApiError && error.status === 503) {
  464. showToast(t('inventory.spoolmanUnreachable'), 'error');
  465. } else {
  466. showToast(t('inventory.saveFailed'), 'error');
  467. }
  468. },
  469. });
  470. const deleteTagMutation = useMutation({
  471. mutationFn: () => {
  472. if (spoolmanMode) {
  473. return api.updateSpoolmanInventorySpool(spool!.id, CLEAR_TAG_PAYLOAD as Parameters<typeof api.updateSpoolmanInventorySpool>[1]);
  474. }
  475. return api.updateSpool(spool!.id, CLEAR_TAG_PAYLOAD as Parameters<typeof api.updateSpool>[1]);
  476. },
  477. onSuccess: async () => {
  478. await queryClient.invalidateQueries({ queryKey: spoolsQueryKey });
  479. showToast(t('inventory.rfidCleared', 'RFID tag cleared'), 'success');
  480. onClose();
  481. },
  482. onError: (error: Error) => {
  483. if (error instanceof ApiError && error.status === 503) {
  484. showToast(t('inventory.spoolmanUnreachable'), 'error');
  485. } else {
  486. showToast(t('inventory.tagClearFailed'), 'error');
  487. }
  488. },
  489. });
  490. // Fetch assignment for this spool (to show Unassign button). In Spoolman mode
  491. // the slot assignment lives in the spoolman_slot_assignments table keyed by
  492. // spoolman_spool_id, not in the legacy spool_assignments table — #1336 was the
  493. // resulting "Unassign button is always disabled" report.
  494. const { data: assignments } = useQuery({
  495. queryKey: ['spool-assignments'],
  496. queryFn: () => api.getAssignments(),
  497. enabled: isOpen && isEditing && !spoolmanMode,
  498. });
  499. const { data: spoolmanSlotAssignments } = useQuery({
  500. queryKey: ['spoolman-slot-assignments-all'],
  501. queryFn: () => api.getSpoolmanSlotAssignments(),
  502. enabled: isOpen && isEditing && spoolmanMode,
  503. });
  504. const spoolAssignment = (() => {
  505. if (!spool) return undefined;
  506. if (spoolmanMode) {
  507. return spoolmanSlotAssignments?.find(a => a.spoolman_spool_id === spool.id);
  508. }
  509. return assignments?.find(a => a.spool_id === spool.id);
  510. })();
  511. // Read inventory + settings caches (already populated by InventoryPage) to
  512. // drive the category autocomplete and low-stock-threshold placeholder. #729
  513. const { data: allSpools } = useQuery({
  514. queryKey: ['inventory-spools'],
  515. queryFn: () => api.getSpools(true),
  516. enabled: isOpen,
  517. });
  518. const { data: settingsForForm } = useQuery({
  519. queryKey: ['settings'],
  520. queryFn: api.getSettings,
  521. enabled: isOpen,
  522. });
  523. const availableCategories = (() => {
  524. const set = new Set<string>();
  525. for (const s of allSpools ?? []) {
  526. const c = s.category?.trim();
  527. if (c) set.add(c);
  528. }
  529. return Array.from(set).sort((a, b) => a.localeCompare(b));
  530. })();
  531. const globalLowStockThreshold = settingsForForm?.low_stock_threshold ?? 20;
  532. const unassignMutation = useMutation({
  533. mutationFn: async () => {
  534. if (!spoolAssignment) throw new Error('No assignment');
  535. if (spoolmanMode) {
  536. if (!spool) throw new Error('No spool');
  537. await api.unassignSpoolmanSlot(spool.id);
  538. return;
  539. }
  540. await api.unassignSpool(spoolAssignment.printer_id, spoolAssignment.ams_id, spoolAssignment.tray_id);
  541. },
  542. onSuccess: async () => {
  543. if (spoolmanMode) {
  544. await queryClient.invalidateQueries({ queryKey: ['spoolman-slot-assignments-all'] });
  545. await queryClient.invalidateQueries({ queryKey: ['spoolman-slot-assignments'] });
  546. } else {
  547. await queryClient.invalidateQueries({ queryKey: ['spool-assignments'] });
  548. }
  549. showToast(t('inventory.unassignSuccess', 'Spool unassigned'), 'success');
  550. onClose();
  551. },
  552. onError: (error: Error) => {
  553. showToast(error.message, 'error');
  554. },
  555. });
  556. // Save K-profiles for selected calibrations. Returns false if any error occurred.
  557. const saveKProfiles = async (spoolId: number): Promise<boolean> => {
  558. const saveApi = spoolmanMode ? api.saveSpoolmanKProfiles : api.saveSpoolKProfiles;
  559. if (selectedProfiles.size === 0) {
  560. try {
  561. await saveApi(spoolId, []);
  562. return true;
  563. } catch (e) {
  564. console.error('Failed to save K-profiles:', e);
  565. showToast(t('inventory.kProfileSaveFailed'), 'warning');
  566. return false;
  567. }
  568. }
  569. const profiles: SpoolKProfileInput[] = [];
  570. let dropped = 0;
  571. for (const key of selectedProfiles) {
  572. const [printerIdStr, caliIdxStr, extruderStr] = key.split(':');
  573. const printerId = parseInt(printerIdStr);
  574. const caliIdx = parseInt(caliIdxStr);
  575. const extruder = extruderStr === 'null' ? 0 : parseInt(extruderStr);
  576. const pc = resolvedCalibrations.find(p => p.printer.id === printerId);
  577. if (pc) {
  578. const cal = pc.calibrations.find(c => c.cali_idx === caliIdx);
  579. if (cal) {
  580. profiles.push({
  581. printer_id: printerId,
  582. extruder,
  583. nozzle_diameter: cal.nozzle_diameter || '0.4',
  584. k_value: cal.k_value,
  585. name: cal.name || null,
  586. cali_idx: cal.cali_idx,
  587. setting_id: cal.setting_id || null,
  588. });
  589. } else {
  590. dropped++;
  591. }
  592. } else {
  593. dropped++;
  594. }
  595. }
  596. if (dropped > 0) {
  597. console.error(`saveKProfiles: ${dropped} profile key(s) could not be resolved`, Array.from(selectedProfiles));
  598. showToast(t('inventory.kProfileSaveFailed'), 'warning');
  599. return false;
  600. }
  601. if (profiles.length > 0) {
  602. try {
  603. await saveApi(spoolId, profiles);
  604. return true;
  605. } catch (e) {
  606. console.error('Failed to save K-profiles:', e);
  607. showToast(t('inventory.kProfileSaveFailed'), 'warning');
  608. return false;
  609. }
  610. }
  611. return true;
  612. };
  613. // Close on Escape key
  614. useEffect(() => {
  615. if (!isOpen) return;
  616. const handleKeyDown = (e: KeyboardEvent) => {
  617. if (e.key === 'Escape') onClose();
  618. };
  619. document.addEventListener('keydown', handleKeyDown);
  620. return () => document.removeEventListener('keydown', handleKeyDown);
  621. }, [isOpen, onClose]);
  622. if (!isOpen) return null;
  623. const handleSubmit = () => {
  624. const validation = validateForm(formData, quickAdd, spoolmanMode);
  625. if (!validation.isValid) {
  626. setErrors(validation.errors);
  627. if (validation.errors.slicer_filament || validation.errors.material || validation.errors.brand || validation.errors.subtype) {
  628. setActiveTab('filament');
  629. }
  630. return;
  631. }
  632. // Find preset name from selected option
  633. const presetName = selectedPresetOption?.displayName || presetInputValue || null;
  634. const data: Record<string, unknown> = {
  635. material: formData.material || null,
  636. subtype: formData.subtype || null,
  637. brand: formData.brand || null,
  638. color_name: formData.color_name || null,
  639. rgba: formData.rgba || null,
  640. extra_colors: formData.extra_colors || null,
  641. effect_type: formData.effect_type || null,
  642. label_weight: formData.label_weight,
  643. ...(spoolmanMode ? {} : { core_weight: formData.core_weight, core_weight_catalog_id: formData.core_weight_catalog_id }),
  644. slicer_filament: formData.slicer_filament || null,
  645. slicer_filament_name: presetName,
  646. nozzle_temp_min: null,
  647. nozzle_temp_max: null,
  648. note: formData.note || null,
  649. cost_per_kg: formData.cost_per_kg,
  650. category: formData.category.trim() || null,
  651. low_stock_threshold_pct: formData.low_stock_threshold_pct,
  652. ...(spoolmanMode ? { spoolman_filament_id: formData.spoolman_filament_id } : {}),
  653. };
  654. // Only send weight_used when creating or when explicitly changed by the user.
  655. // This prevents stale cached values from overwriting usage-tracker data.
  656. if (!isEditing || weightTouched) {
  657. data.weight_used = formData.weight_used;
  658. }
  659. // Only send storage_location when creating or when explicitly changed by the user.
  660. // This prevents the modal round-trip from overwriting the Spoolman location field
  661. // with a stale cached value when the user saves without touching this field.
  662. if (!isEditing || storageLocationTouched) {
  663. data.storage_location = formData.storage_location || null;
  664. }
  665. if (isEditing) {
  666. updateMutation.mutate(data);
  667. } else if (quantity > 1) {
  668. bulkCreateMutation.mutate({ data, qty: quantity });
  669. } else {
  670. createMutation.mutate(data);
  671. }
  672. };
  673. const isPending = createMutation.isPending || bulkCreateMutation.isPending || updateMutation.isPending || deleteTagMutation.isPending || unassignMutation.isPending;
  674. return (
  675. <div className="fixed inset-0 z-50 flex items-center justify-center">
  676. <div
  677. className="absolute inset-0 bg-black/60 backdrop-blur-sm"
  678. onClick={onClose}
  679. />
  680. <div className="relative w-full max-w-xl mx-4 bg-bambu-dark-secondary border border-bambu-dark-tertiary rounded-xl shadow-2xl max-h-[90vh] flex flex-col">
  681. {/* Header */}
  682. <div className="flex items-center justify-between p-4 border-b border-bambu-dark-tertiary flex-shrink-0">
  683. <h2 className="text-lg font-semibold text-white">
  684. {isEditing ? t('inventory.editSpool') : isCopying ? t('inventory.copySpool') : t('inventory.addSpool')}
  685. </h2>
  686. <button
  687. onClick={onClose}
  688. className="p-1 text-bambu-gray hover:text-white rounded transition-colors"
  689. >
  690. <X className="w-5 h-5" />
  691. </button>
  692. </div>
  693. {/* Quick Add toggle — only in create mode (not edit, not copy).
  694. In copy mode the modal title is the singular "Copy Spool", so the
  695. quantity-driven bulkCreateMutation path would silently produce N
  696. copies under a misleading title — keep this toggle out of that
  697. mode entirely. */}
  698. {mode === 'create' && (
  699. <div className="flex items-center justify-between px-4 py-2 border-b border-bambu-dark-tertiary flex-shrink-0">
  700. <div className="flex items-center gap-2">
  701. <Zap className="w-4 h-4 text-amber-400" />
  702. <span className="text-sm text-white">{t('inventory.quickAdd')}</span>
  703. </div>
  704. <button
  705. type="button"
  706. onClick={() => {
  707. setQuickAdd(!quickAdd);
  708. if (!quickAdd) setActiveTab('filament');
  709. }}
  710. className={`relative inline-flex h-5 w-9 items-center rounded-full transition-colors ${
  711. quickAdd ? 'bg-bambu-green' : 'bg-bambu-dark-tertiary'
  712. }`}
  713. >
  714. <span
  715. className={`inline-block h-3.5 w-3.5 rounded-full bg-white transition-transform ${
  716. quickAdd ? 'translate-x-4' : 'translate-x-0.5'
  717. }`}
  718. />
  719. </button>
  720. </div>
  721. )}
  722. {/* Tabs */}
  723. <div className="flex border-b border-bambu-dark-tertiary flex-shrink-0">
  724. <button
  725. onClick={() => setActiveTab('filament')}
  726. className={`flex-1 px-4 py-2.5 text-sm font-medium flex items-center justify-center gap-2 transition-colors ${
  727. activeTab === 'filament'
  728. ? 'text-bambu-green border-b-2 border-bambu-green'
  729. : 'text-bambu-gray hover:text-white'
  730. }`}
  731. >
  732. <Palette className="w-4 h-4" />
  733. {t('inventory.filamentInfoTab')}
  734. </button>
  735. {!quickAdd && (
  736. <button
  737. onClick={() => setActiveTab('pa-profile')}
  738. className={`flex-1 px-4 py-2.5 text-sm font-medium flex items-center justify-center gap-2 transition-colors ${
  739. activeTab === 'pa-profile'
  740. ? 'text-bambu-green border-b-2 border-bambu-green'
  741. : 'text-bambu-gray hover:text-white'
  742. }`}
  743. >
  744. <Beaker className="w-4 h-4" />
  745. {t('inventory.paProfileTab')}
  746. {selectedProfileCount > 0 && (
  747. <span className="text-xs px-1.5 py-0.5 rounded-full bg-bambu-green/20 text-bambu-green">
  748. {selectedProfileCount}
  749. </span>
  750. )}
  751. </button>
  752. )}
  753. </div>
  754. {/* Content */}
  755. <div className="p-4 overflow-y-auto flex-1" style={{ scrollbarGutter: 'stable' }}>
  756. {activeTab === 'filament' ? (
  757. <div className="space-y-6">
  758. {/* Spoolman Filament Catalog Picker — only when creating a spool in Spoolman mode */}
  759. {spoolmanMode && !isEditing && (
  760. <div>
  761. {filamentsError ? (
  762. <p className="text-sm text-red-400 px-1">{t('inventory.spoolmanCatalogLoadFailed')}</p>
  763. ) : (
  764. <SpoolmanFilamentPicker
  765. filaments={spoolmanFilaments}
  766. isLoading={isLoadingFilaments}
  767. selectedId={formData.spoolman_filament_id}
  768. onSelect={handleFilamentSelect}
  769. />
  770. )}
  771. </div>
  772. )}
  773. {/* Filament Info Section */}
  774. <div>
  775. <h3 className="text-sm font-semibold text-bambu-gray uppercase tracking-wide mb-3">
  776. {t('inventory.filamentInfo')}
  777. </h3>
  778. <FilamentSection
  779. formData={formData}
  780. updateField={updateField}
  781. cloudAuthenticated={cloudAuthenticated}
  782. loadingCloudPresets={loadingCloudPresets}
  783. presetInputValue={presetInputValue}
  784. setPresetInputValue={setPresetInputValue}
  785. selectedPresetOption={selectedPresetOption}
  786. filamentOptions={filamentOptions}
  787. availableBrands={availableBrands}
  788. availableMaterials={availableMaterials}
  789. quickAdd={quickAdd}
  790. quantity={quantity}
  791. onQuantityChange={setQuantity}
  792. errors={errors}
  793. />
  794. </div>
  795. {/* Color Section */}
  796. <div>
  797. <h3 className="text-sm font-semibold text-bambu-gray uppercase tracking-wide mb-3">
  798. {t('inventory.color')}
  799. </h3>
  800. <ColorSection
  801. formData={formData}
  802. updateField={updateField}
  803. recentColors={recentColors}
  804. onColorUsed={handleColorUsed}
  805. catalogColors={colorCatalog}
  806. />
  807. </div>
  808. {/* Additional Section */}
  809. <div>
  810. <h3 className="text-sm font-semibold text-bambu-gray uppercase tracking-wide mb-3">
  811. {t('inventory.additional')}
  812. </h3>
  813. <AdditionalSection
  814. formData={formData}
  815. updateField={updateField}
  816. spoolCatalog={spoolCatalog}
  817. currencySymbol={currencySymbol}
  818. availableCategories={availableCategories}
  819. globalLowStockThreshold={globalLowStockThreshold}
  820. spoolmanMode={spoolmanMode}
  821. />
  822. </div>
  823. {/* Usage History (only when editing internal inventory; Spoolman tracks its own) */}
  824. {isEditing && spool && !spoolmanMode && (
  825. <div>
  826. <SpoolUsageHistory spoolId={spool.id} />
  827. </div>
  828. )}
  829. </div>
  830. ) : (
  831. <PAProfileSection
  832. formData={formData}
  833. updateField={updateField}
  834. printersWithCalibrations={resolvedCalibrations}
  835. selectedProfiles={selectedProfiles}
  836. setSelectedProfiles={setSelectedProfiles}
  837. expandedPrinters={expandedPrinters}
  838. setExpandedPrinters={setExpandedPrinters}
  839. />
  840. )}
  841. </div>
  842. {/* Footer */}
  843. <div className="flex gap-2 p-4 border-t border-bambu-dark-tertiary flex-shrink-0">
  844. {isEditing && (
  845. <div className="flex gap-2 mr-auto">
  846. <Button
  847. variant="secondary"
  848. onClick={() => deleteTagMutation.mutate()}
  849. disabled={isPending || !spool?.tag_uid}
  850. >
  851. <Tag className="w-4 h-4" />
  852. {t('inventory.clearRfid', 'Clear RFID Tag')}
  853. </Button>
  854. <Button
  855. variant="secondary"
  856. onClick={() => unassignMutation.mutate()}
  857. disabled={isPending || !spoolAssignment}
  858. >
  859. <Unlink className="w-4 h-4" />
  860. {t('inventory.unassignSpool', 'Unassign')}
  861. </Button>
  862. </div>
  863. )}
  864. <div className="flex gap-2 ml-auto">
  865. <Button variant="secondary" onClick={onClose}>
  866. {t('common.cancel')}
  867. </Button>
  868. <Button
  869. onClick={handleSubmit}
  870. disabled={isPending}
  871. >
  872. {isPending ? (
  873. <>
  874. <Loader2 className="w-4 h-4 animate-spin" />
  875. {t('common.saving')}
  876. </>
  877. ) : (
  878. <>
  879. <Save className="w-4 h-4" />
  880. {isEditing ? t('common.save') : isCopying ? t('inventory.copySpool') : t('inventory.addSpool')}
  881. </>
  882. )}
  883. </Button>
  884. </div>
  885. </div>
  886. </div>
  887. </div>
  888. );
  889. }