import { useState } from "react"; import { LuPlus } from "react-icons/lu"; import { KeySequence } from "@/hooks/stores"; import { Button } from "@/components/Button"; import { InputFieldWithLabel, FieldError } from "@/components/InputField"; import Fieldset from "@/components/Fieldset"; import { MacroStepCard } from "@/components/MacroStepCard"; import { DEFAULT_DELAY, MAX_STEPS_PER_MACRO, MAX_KEYS_PER_STEP, } from "@/constants/macros"; import FieldLabel from "@/components/FieldLabel"; interface ValidationErrors { name?: string; steps?: Record< number, { keys?: string; modifiers?: string; delay?: string; } >; } interface MacroFormProps { initialData: Partial; onSubmit: (macro: Partial) => Promise; onCancel: () => void; isSubmitting?: boolean; submitText?: string; } export function MacroForm({ initialData, onSubmit, onCancel, isSubmitting = false, submitText = "Save Macro", }: MacroFormProps) { const [macro, setMacro] = useState>(initialData); const [keyQueries, setKeyQueries] = useState>({}); const [errors, setErrors] = useState({}); const [errorMessage, setErrorMessage] = useState(null); const showTemporaryError = (message: string) => { setErrorMessage(message); setTimeout(() => setErrorMessage(null), 3000); }; const validateForm = (): boolean => { const newErrors: ValidationErrors = {}; // Name validation if (!macro.name?.trim()) { newErrors.name = "Name is required"; } else if (macro.name.trim().length > 50) { newErrors.name = "Name must be less than 50 characters"; } if (!macro.steps?.length) { newErrors.steps = { 0: { keys: "At least one step is required" } }; } else { const hasKeyOrModifier = macro.steps.some( step => (step.keys?.length || 0) > 0 || (step.modifiers?.length || 0) > 0, ); if (!hasKeyOrModifier) { newErrors.steps = { 0: { keys: "At least one step must have keys or modifiers" }, }; } } setErrors(newErrors); return Object.keys(newErrors).length === 0; }; const handleSubmit = async () => { if (!validateForm()) { showTemporaryError("Please fix the validation errors"); return; } try { await onSubmit(macro); } catch (error) { if (error instanceof Error) { showTemporaryError(error.message); } else { showTemporaryError("An error occurred while saving"); } } }; const handleKeySelect = ( stepIndex: number, option: { value: string | null; keys?: string[] }, ) => { const newSteps = [...(macro.steps || [])]; if (!newSteps[stepIndex]) return; if (option.keys) { newSteps[stepIndex].keys = option.keys; } else if (option.value) { if (!newSteps[stepIndex].keys) { newSteps[stepIndex].keys = []; } const keysArray = Array.isArray(newSteps[stepIndex].keys) ? newSteps[stepIndex].keys : []; if (keysArray.length >= MAX_KEYS_PER_STEP) { showTemporaryError(`Maximum of ${MAX_KEYS_PER_STEP} keys per step allowed`); return; } newSteps[stepIndex].keys = [...keysArray, option.value]; } setMacro({ ...macro, steps: newSteps }); if (errors.steps?.[stepIndex]?.keys) { const newErrors = { ...errors }; delete newErrors.steps?.[stepIndex].keys; if (Object.keys(newErrors.steps?.[stepIndex] || {}).length === 0) { delete newErrors.steps?.[stepIndex]; } if (Object.keys(newErrors.steps || {}).length === 0) { delete newErrors.steps; } setErrors(newErrors); } }; const handleKeyQueryChange = (stepIndex: number, query: string) => { setKeyQueries(prev => ({ ...prev, [stepIndex]: query })); }; const handleModifierChange = (stepIndex: number, modifiers: string[]) => { const newSteps = [...(macro.steps || [])]; newSteps[stepIndex].modifiers = modifiers; setMacro({ ...macro, steps: newSteps }); // Clear step errors when modifiers are added if (errors.steps?.[stepIndex]?.keys && modifiers.length > 0) { const newErrors = { ...errors }; delete newErrors.steps?.[stepIndex].keys; if (Object.keys(newErrors.steps?.[stepIndex] || {}).length === 0) { delete newErrors.steps?.[stepIndex]; } if (Object.keys(newErrors.steps || {}).length === 0) { delete newErrors.steps; } setErrors(newErrors); } }; const handleDelayChange = (stepIndex: number, delay: number) => { const newSteps = [...(macro.steps || [])]; newSteps[stepIndex].delay = delay; setMacro({ ...macro, steps: newSteps }); }; const handleStepMove = (stepIndex: number, direction: "up" | "down") => { const newSteps = [...(macro.steps || [])]; const newIndex = direction === "up" ? stepIndex - 1 : stepIndex + 1; [newSteps[stepIndex], newSteps[newIndex]] = [newSteps[newIndex], newSteps[stepIndex]]; setMacro({ ...macro, steps: newSteps }); }; const isMaxStepsReached = (macro.steps?.length || 0) >= MAX_STEPS_PER_MACRO; return ( <>
{ setMacro(prev => ({ ...prev, name: e.target.value })); if (errors.name) { const newErrors = { ...errors }; delete newErrors.name; setErrors(newErrors); } }} />
{macro.steps?.length || 0}/{MAX_STEPS_PER_MACRO} steps
{errors.steps && errors.steps[0]?.keys && (
)}
{(macro.steps || []).map((step, stepIndex) => ( 1 ? () => { const newSteps = [...(macro.steps || [])]; newSteps.splice(stepIndex, 1); setMacro(prev => ({ ...prev, steps: newSteps })); } : undefined } onMoveUp={() => handleStepMove(stepIndex, "up")} onMoveDown={() => handleStepMove(stepIndex, "down")} onKeySelect={option => handleKeySelect(stepIndex, option)} onKeyQueryChange={query => handleKeyQueryChange(stepIndex, query)} keyQuery={keyQueries[stepIndex] || ""} onModifierChange={modifiers => handleModifierChange(stepIndex, modifiers) } onDelayChange={delay => handleDelayChange(stepIndex, delay)} isLastStep={stepIndex === (macro.steps?.length || 0) - 1} /> ))}
{errorMessage && (
)}
); }