import React, { useState } from 'react'; import { toTitleCase } from '../utils/text'; import { Dumbbell, User, Flame, Timer as TimerIcon, ArrowUp, ArrowRight, Footprints, Ruler, Percent } from 'lucide-react'; import { ExerciseDef, ExerciseType, Language } from '../types'; import { t } from '../services/i18n'; import { generateId } from '../utils/uuid'; import FilledInput from './FilledInput'; import { Modal } from './ui/Modal'; import { Button } from './ui/Button'; interface ExerciseModalProps { isOpen: boolean; onClose: () => void; onSave: (exercise: ExerciseDef) => Promise | void; lang: Language; existingExercises?: ExerciseDef[]; } const ExerciseModal: React.FC = ({ isOpen, onClose, onSave, lang, existingExercises = [] }) => { const [newName, setNewName] = useState(''); const [newType, setNewType] = useState(ExerciseType.STRENGTH); const [newBwPercentage, setNewBwPercentage] = useState('100'); const [isUnilateral, setIsUnilateral] = useState(false); const [error, setError] = useState(''); const exerciseTypeLabels: Record = { [ExerciseType.STRENGTH]: t('type_strength', lang), [ExerciseType.BODYWEIGHT]: t('type_bodyweight', lang), [ExerciseType.CARDIO]: t('type_cardio', lang), [ExerciseType.STATIC]: t('type_static', lang), [ExerciseType.HIGH_JUMP]: t('type_height', lang), [ExerciseType.LONG_JUMP]: t('type_dist', lang), [ExerciseType.PLYOMETRIC]: t('type_jump', lang), }; const handleCreateExercise = async () => { if (!newName.trim()) return; // Check for duplicate name (case-insensitive) const trimmedName = newName.trim(); const isDuplicate = existingExercises.some( ex => ex.name.toLowerCase() === trimmedName.toLowerCase() ); if (isDuplicate) { setError(t('exercise_name_exists', lang) || 'An exercise with this name already exists'); return; } const newEx: ExerciseDef = { id: generateId(), name: trimmedName, type: newType, isUnilateral, ...(newType === ExerciseType.BODYWEIGHT && { bodyWeightPercentage: parseFloat(newBwPercentage) || 100 }) }; await onSave(newEx); setNewName(''); setNewType(ExerciseType.STRENGTH); setNewBwPercentage('100'); setIsUnilateral(false); setError(''); onClose(); // Modal controls its own open state usually, but here checking prop }; return ( { console.log('ExerciseModal onClose'); onClose(); }} title={t('create_exercise', lang)} maxWidth="sm" > {console.log('ExerciseModal Rendering. isOpen:', isOpen)}
{ setNewName(e.target.value); setError(''); // Clear error when user types }} type="text" autoFocus autocapitalize="words" onBlur={() => setNewName(toTitleCase(newName))} /> {error && (

{error}

)}
{[ { id: ExerciseType.STRENGTH, label: exerciseTypeLabels[ExerciseType.STRENGTH], icon: Dumbbell }, { id: ExerciseType.BODYWEIGHT, label: exerciseTypeLabels[ExerciseType.BODYWEIGHT], icon: User }, { id: ExerciseType.CARDIO, label: exerciseTypeLabels[ExerciseType.CARDIO], icon: Flame }, { id: ExerciseType.STATIC, label: exerciseTypeLabels[ExerciseType.STATIC], icon: TimerIcon }, { id: ExerciseType.HIGH_JUMP, label: exerciseTypeLabels[ExerciseType.HIGH_JUMP], icon: ArrowUp }, { id: ExerciseType.LONG_JUMP, label: exerciseTypeLabels[ExerciseType.LONG_JUMP], icon: Ruler }, { id: ExerciseType.PLYOMETRIC, label: exerciseTypeLabels[ExerciseType.PLYOMETRIC], icon: Footprints }, ].map((type) => ( ))}
{newType === ExerciseType.BODYWEIGHT && ( setNewBwPercentage(e.target.value)} icon={} /> )}
setIsUnilateral(e.target.checked)} className="w-5 h-5 rounded border-2 border-outline bg-surface-container-high checked:bg-primary checked:border-primary cursor-pointer" />
); }; export default ExerciseModal;