workout-stepper.tsx 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. "use client";
  2. import { useState, useEffect } from "react";
  3. import { useQueryState } from "nuqs";
  4. import { useRouter } from "next/navigation";
  5. import Image from "next/image";
  6. import { ExerciseAttributeValueEnum } from "@prisma/client";
  7. import { useI18n } from "locales/client";
  8. import Trophy from "@public/images/trophy.png";
  9. import { WorkoutSessionSets } from "@/features/workout-session/ui/workout-session-sets";
  10. import { WorkoutSessionHeader } from "@/features/workout-session/ui/workout-session-header";
  11. import { WorkoutBuilderFooter } from "@/features/workout-builder/ui/workout-stepper-footer";
  12. import { Button } from "@/components/ui/button";
  13. import { StepperStepProps } from "../types";
  14. import { useWorkoutStepper } from "../model/use-workout-stepper";
  15. import { useWorkoutSession } from "../../workout-session/model/use-workout-session";
  16. import { StepperHeader } from "./stepper-header";
  17. import { MuscleSelection } from "./muscle-selection";
  18. import { ExercisesSelection } from "./exercises-selection";
  19. import { EquipmentSelection } from "./equipment-selection";
  20. import type { ExerciseWithAttributes, WorkoutBuilderStep } from "../types";
  21. export function WorkoutStepper() {
  22. const { loadSessionFromLocal } = useWorkoutSession();
  23. const t = useI18n();
  24. const router = useRouter();
  25. const [fromSession, setFromSession] = useQueryState("fromSession");
  26. const {
  27. currentStep,
  28. selectedEquipment,
  29. selectedMuscles,
  30. exercisesByMuscle,
  31. isLoadingExercises,
  32. exercisesError,
  33. nextStep,
  34. prevStep,
  35. toggleEquipment,
  36. clearEquipment,
  37. toggleMuscle,
  38. canProceedToStep2,
  39. canProceedToStep3,
  40. fetchExercises,
  41. exercisesOrder,
  42. shuffleExercise,
  43. pickExercise,
  44. isShuffling,
  45. goToStep,
  46. deleteExercise,
  47. } = useWorkoutStepper();
  48. useEffect(() => {
  49. loadSessionFromLocal();
  50. }, []);
  51. const [flatExercises, setFlatExercises] = useState<{ id: string; muscle: string; exercise: ExerciseWithAttributes }[]>([]);
  52. useEffect(() => {
  53. if (exercisesByMuscle.length > 0) {
  54. const flat = exercisesByMuscle.flatMap((group) =>
  55. group.exercises.map((exercise: ExerciseWithAttributes) => ({
  56. id: exercise.id,
  57. muscle: group.muscle,
  58. exercise,
  59. })),
  60. );
  61. setFlatExercises(flat);
  62. }
  63. }, [exercisesByMuscle]);
  64. useEffect(() => {
  65. if (currentStep === 3 && !fromSession) {
  66. fetchExercises();
  67. }
  68. }, [currentStep, selectedEquipment, selectedMuscles, fromSession]);
  69. const { isWorkoutActive, session, startWorkout, formatElapsedTime, isTimerRunning, toggleTimer, resetTimer, quitWorkout } =
  70. useWorkoutSession();
  71. const canContinue = currentStep === 1 ? canProceedToStep2 : currentStep === 2 ? canProceedToStep3 : exercisesByMuscle.length > 0;
  72. const handleShuffleExercise = async (exerciseId: string, muscle: string) => {
  73. try {
  74. // Convertir le muscle string vers enum
  75. const muscleEnum = muscle as ExerciseAttributeValueEnum;
  76. await shuffleExercise(exerciseId, muscleEnum);
  77. } catch (error) {
  78. console.error("Error shuffling exercise:", error);
  79. alert("Error shuffling exercise. Please try again.");
  80. }
  81. };
  82. const handlePickExercise = async (exerciseId: string) => {
  83. try {
  84. await pickExercise(exerciseId);
  85. // Optionnel: afficher un toast de succès
  86. console.log("Exercise picked successfully!");
  87. } catch (error) {
  88. console.error("Error picking exercise:", error);
  89. alert("Error picking exercise. Please try again.");
  90. }
  91. };
  92. const handleDeleteExercise = (exerciseId: string) => {
  93. deleteExercise(exerciseId);
  94. };
  95. const handleAddExercise = () => {
  96. alert("TODO : Add exercise 🥶");
  97. console.log("Add exercise");
  98. };
  99. const orderedExercises = exercisesOrder.length
  100. ? exercisesOrder
  101. .map((id) => flatExercises.find((item) => item.id === id))
  102. .filter(Boolean)
  103. .map((item) => item!.exercise)
  104. : flatExercises.map((item) => item.exercise);
  105. const handleStartWorkout = () => {
  106. if (orderedExercises.length > 0) {
  107. startWorkout(orderedExercises, selectedEquipment, selectedMuscles);
  108. }
  109. };
  110. const [showCongrats, setShowCongrats] = useState(false);
  111. const goToProfile = () => {
  112. router.push("/profile");
  113. };
  114. const handleToggleEquipment = (equipment: ExerciseAttributeValueEnum) => {
  115. toggleEquipment(equipment);
  116. if (fromSession) setFromSession(null);
  117. };
  118. const handleClearEquipment = () => {
  119. clearEquipment();
  120. if (fromSession) setFromSession(null);
  121. };
  122. const handleToggleMuscle = (muscle: ExerciseAttributeValueEnum) => {
  123. toggleMuscle(muscle);
  124. if (fromSession) setFromSession(null);
  125. };
  126. const handleStepClick = (stepNumber: number) => {
  127. if (stepNumber < currentStep) {
  128. goToStep(stepNumber as WorkoutBuilderStep);
  129. }
  130. };
  131. if (showCongrats && !isWorkoutActive) {
  132. return (
  133. <div className="flex flex-col items-center justify-center py-16 h-full">
  134. <Image alt="Trophée" className="w-56 h-56" src={Trophy} />
  135. <h2 className="text-2xl font-bold mb-2">{t("workout_builder.session.congrats")}</h2>
  136. <p className="text-lg text-slate-600 mb-6">{t("workout_builder.session.congrats_subtitle")}</p>
  137. <Button onClick={goToProfile}>{t("commons.go_to_profile")}</Button>
  138. </div>
  139. );
  140. }
  141. if (isWorkoutActive && session) {
  142. return (
  143. <div className="w-full max-w-6xl mx-auto">
  144. {!showCongrats && (
  145. <WorkoutSessionHeader
  146. elapsedTime={formatElapsedTime()}
  147. isTimerRunning={isTimerRunning}
  148. onQuitWorkout={quitWorkout}
  149. onResetTimer={resetTimer}
  150. onToggleTimer={toggleTimer}
  151. />
  152. )}
  153. <WorkoutSessionSets isWorkoutActive={isWorkoutActive} onCongrats={() => setShowCongrats(true)} showCongrats={showCongrats} />
  154. </div>
  155. );
  156. }
  157. const STEPPER_STEPS: StepperStepProps[] = [
  158. {
  159. stepNumber: 1,
  160. title: t("workout_builder.steps.equipment.title"),
  161. description: t("workout_builder.steps.equipment.description"),
  162. isActive: false,
  163. isCompleted: false,
  164. },
  165. {
  166. stepNumber: 2,
  167. title: t("workout_builder.steps.muscles.title"),
  168. description: t("workout_builder.steps.muscles.description"),
  169. isActive: false,
  170. isCompleted: false,
  171. },
  172. {
  173. stepNumber: 3,
  174. title: t("workout_builder.steps.exercises.title"),
  175. description: t("workout_builder.steps.exercises.description"),
  176. isActive: false,
  177. isCompleted: false,
  178. },
  179. ];
  180. const steps = STEPPER_STEPS.map((step) => ({
  181. ...step,
  182. isActive: step.stepNumber === currentStep,
  183. isCompleted: step.stepNumber < currentStep,
  184. }));
  185. const renderStepContent = () => {
  186. switch (currentStep) {
  187. case 1:
  188. return (
  189. <EquipmentSelection
  190. onClearEquipment={handleClearEquipment}
  191. onToggleEquipment={handleToggleEquipment}
  192. selectedEquipment={selectedEquipment}
  193. />
  194. );
  195. case 2:
  196. return (
  197. <MuscleSelection onToggleMuscle={handleToggleMuscle} selectedEquipment={selectedEquipment} selectedMuscles={selectedMuscles} />
  198. );
  199. case 3:
  200. return (
  201. <ExercisesSelection
  202. error={exercisesError}
  203. exercisesByMuscle={exercisesByMuscle}
  204. isLoading={isLoadingExercises}
  205. isShuffling={isShuffling}
  206. onAdd={handleAddExercise}
  207. onDelete={handleDeleteExercise}
  208. onPick={handlePickExercise}
  209. onShuffle={handleShuffleExercise}
  210. />
  211. );
  212. default:
  213. return null;
  214. }
  215. };
  216. return (
  217. <div className="w-full max-w-6xl mx-auto h-full">
  218. <StepperHeader currentStep={currentStep} onStepClick={handleStepClick} steps={steps} />
  219. <div className="px-2 sm:px-6">{renderStepContent()}</div>
  220. <WorkoutBuilderFooter
  221. canContinue={canContinue}
  222. currentStep={currentStep}
  223. onNext={nextStep}
  224. onPrevious={prevStep}
  225. onStartWorkout={handleStartWorkout}
  226. selectedEquipment={selectedEquipment}
  227. selectedMuscles={selectedMuscles}
  228. totalSteps={STEPPER_STEPS.length}
  229. />
  230. </div>
  231. );
  232. }