import { useState, useEffect, useMemo } from 'react' import { X, Save, AlertCircle } from 'lucide-react' import ColorSwatch from './ColorSwatch' import { createFilament, updateFilament, fetchMaterialBases, fetchMaterialFinishes, fetchMaterialModifiers } from '../services/filamentService' import type { FilamentSpool, MaterialBase, MaterialFinish, MaterialModifier } from '../types/filament' import { useQuery } from '@tanstack/react-query' interface FilamentFormProps { mode: 'create' | 'edit' initialData?: FilamentSpool | null onClose: () => void onSuccess: () => void } interface FormErrors { [key: string]: string } export default function FilamentForm({ mode, initialData, onClose, onSuccess }: FilamentFormProps) { const [name, setName] = useState('') const [materialBaseId, setMaterialBaseId] = useState('') const [materialFinishId, setMaterialFinishId] = useState('1') const [materialModifierId, setMaterialModifierId] = useState('') const [colorHex, setColorHex] = useState('#3B82F6') const [brand, setBrand] = useState('') const [diameterMm, setDiameterMm] = useState('1.75') const [initialGrams, setInitialGrams] = useState('') const [remainingGrams, setRemainingGrams] = useState('') const [costUsd, setCostUsd] = useState('') const [lowStockThreshold, setLowStockThreshold] = useState('50') const [notes, setNotes] = useState('') const [barcode, setBarcode] = useState('') const [errors, setErrors] = useState({}) const [submitting, setSubmitting] = useState(false) const [submitError, setSubmitError] = useState(null) const { data: materials } = useQuery({ queryKey: ['materials'], queryFn: fetchMaterialBases, staleTime: Infinity, }) const { data: finishes } = useQuery({ queryKey: ['finishes'], queryFn: fetchMaterialFinishes, staleTime: Infinity, }) const { data: modifiers } = useQuery({ queryKey: ['modifiers'], queryFn: fetchMaterialModifiers, staleTime: Infinity, }) useEffect(() => { if (mode === 'edit' && initialData) { setName(initialData.name ?? '') setMaterialBaseId(String(initialData.material_base_id ?? '')) setMaterialFinishId(String(initialData.material_finish_id ?? '1')) setMaterialModifierId(initialData.material_modifier_id ? String(initialData.material_modifier_id) : '') setColorHex(initialData.color_hex ?? '#3B82F6') setBrand(initialData.brand ?? '') setDiameterMm(String(initialData.diameter_mm ?? 1.75)) setInitialGrams(String(initialData.initial_grams ?? '')) setRemainingGrams(String(initialData.remaining_grams ?? '')) setCostUsd(initialData.cost_usd != null ? String(initialData.cost_usd) : '') setLowStockThreshold(String(initialData.low_stock_threshold_grams ?? 50)) setNotes(initialData.notes ?? '') setBarcode(initialData.barcode ?? '') } }, [mode, initialData]) const colorHexValid = useMemo(() => { return /^#[0-9A-Fa-f]{6}$/.test(colorHex) }, [colorHex]) function validate(): FormErrors { const e: FormErrors = {} if (!name.trim()) e.name = 'Name is required' if (!materialBaseId) e.material_base_id = 'Material Base is required' if (!materialFinishId) e.material_finish_id = 'Material Finish is required' if (!colorHexValid) e.color_hex = 'Enter a valid hex color (e.g., #FF0000)' if (!initialGrams || Number(initialGrams) <= 0) e.initial_grams = 'Must be > 0' if (remainingGrams === '' || Number(remainingGrams) < 0) e.remaining_grams = 'Must be >= 0' if (Number(remainingGrams) > Number(initialGrams)) e.remaining_grams = 'Cannot exceed Initial Grams' if (costUsd && Number(costUsd) < 0) e.cost_usd = 'Must be >= 0' if (!diameterMm || Number(diameterMm) <= 0) e.diameter_mm = 'Must be > 0' if (!lowStockThreshold || Number(lowStockThreshold) < 0) e.low_stock_threshold_grams = 'Must be >= 0' return e } async function handleSubmit(e: React.FormEvent) { e.preventDefault() setSubmitError(null) const validation = validate() if (Object.keys(validation).length > 0) { setErrors(validation) return } setErrors({}) setSubmitting(true) const payload: Record = { name: name.trim(), material_base_id: Number(materialBaseId), material_finish_id: Number(materialFinishId), color_hex: colorHex, initial_grams: Number(initialGrams), remaining_grams: Number(remainingGrams), diameter_mm: Number(diameterMm), low_stock_threshold_grams: Number(lowStockThreshold), } if (materialModifierId) payload.material_modifier_id = Number(materialModifierId) if (brand.trim()) payload.brand = brand.trim() if (costUsd) payload.cost_usd = Number(costUsd) if (notes.trim()) payload.notes = notes.trim() if (barcode.trim()) payload.barcode = barcode.trim() try { if (mode === 'edit' && initialData) { await updateFilament(initialData.id, payload) } else { await createFilament(payload) } onSuccess() onClose() } catch (err: any) { setSubmitError(err?.response?.data?.error || 'Failed to save spool. Please try again.') } finally { setSubmitting(false) } } return (
{/* Header */}

{mode === 'edit' ? 'Edit Filament Spool' : 'Add Filament Spool'}

{/* Error banner */} {submitError && (
{submitError}
)} {/* Form */}
{/* Row 1: Name + Brand */}
setName(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 ${errors.name ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} placeholder="e.g. Sunlu PLA Silk Red" /> {errors.name &&

{errors.name}

}
setBrand(e.target.value)} className="w-full rounded-lg bg-slate-900 border border-slate-600 px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 focus:ring-emerald-500" placeholder="e.g. Hatchbox" />
{/* Row 2: Material Base + Finish + Modifier */}
{errors.material_base_id &&

{errors.material_base_id}

}
{errors.material_finish_id &&

{errors.material_finish_id}

}
{/* Row 3: Color + Diameter */}
setColorHex(e.target.value)} className="h-10 w-14 rounded border border-slate-600 bg-slate-900 cursor-pointer" />
setColorHex(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 font-mono uppercase focus:outline-none focus:ring-2 ${errors.color_hex ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} placeholder="#FF0000" maxLength={7} />
{errors.color_hex &&

{errors.color_hex}

}
setDiameterMm(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 ${errors.diameter_mm ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} /> {errors.diameter_mm &&

{errors.diameter_mm}

}
{/* Row 4: Grams */}
setInitialGrams(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 ${errors.initial_grams ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} /> {errors.initial_grams &&

{errors.initial_grams}

}
setRemainingGrams(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 ${errors.remaining_grams ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} /> {errors.remaining_grams &&

{errors.remaining_grams}

}
setCostUsd(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 ${errors.cost_usd ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} />
{/* Row 5: Threshold + Barcode */}
setLowStockThreshold(e.target.value)} className={`w-full rounded-lg bg-slate-900 border px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 ${errors.low_stock_threshold_grams ? 'border-red-500 focus:ring-red-500' : 'border-slate-600 focus:ring-emerald-500'}`} /> {errors.low_stock_threshold_grams &&

{errors.low_stock_threshold_grams}

}
setBarcode(e.target.value)} className="w-full rounded-lg bg-slate-900 border border-slate-600 px-3 py-2.5 text-sm text-slate-100 focus:outline-none focus:ring-2 focus:ring-emerald-500" placeholder="e.g. 123456789012" />
{/* Notes */}