workout-stepper.tsx 8.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269
  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 } 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. goToStep,
  34. nextStep,
  35. prevStep,
  36. toggleEquipment,
  37. clearEquipment,
  38. toggleMuscle,
  39. canProceedToStep2,
  40. canProceedToStep3,
  41. fetchExercises,
  42. exercisesOrder,
  43. setExercisesOrder,
  44. shuffleExercise,
  45. pickExercise,
  46. isShuffling,
  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 {
  70. isWorkoutActive,
  71. session,
  72. startWorkout,
  73. currentExercise,
  74. formatElapsedTime,
  75. isTimerRunning,
  76. toggleTimer,
  77. resetTimer,
  78. quitWorkout,
  79. } = useWorkoutSession();
  80. const canContinue = currentStep === 1 ? canProceedToStep2 : currentStep === 2 ? canProceedToStep3 : exercisesByMuscle.length > 0;
  81. const handleShuffleExercise = async (exerciseId: string, muscle: string) => {
  82. try {
  83. // Convertir le muscle string vers enum
  84. const muscleEnum = muscle as ExerciseAttributeValueEnum;
  85. await shuffleExercise(exerciseId, muscleEnum);
  86. } catch (error) {
  87. console.error("Error shuffling exercise:", error);
  88. alert("Error shuffling exercise. Please try again.");
  89. }
  90. };
  91. const handlePickExercise = async (exerciseId: string) => {
  92. try {
  93. await pickExercise(exerciseId);
  94. // Optionnel: afficher un toast de succès
  95. console.log("Exercise picked successfully!");
  96. } catch (error) {
  97. console.error("Error picking exercise:", error);
  98. alert("Error picking exercise. Please try again.");
  99. }
  100. };
  101. const handleDeleteExercise = (exerciseId: string, muscle: string) => {
  102. alert("TODO : Delete exercise");
  103. console.log("Delete exercise:", exerciseId, "for muscle:", muscle);
  104. };
  105. const handleAddExercise = () => {
  106. alert("TODO : Add exercise");
  107. console.log("Add exercise");
  108. };
  109. const orderedExercises = exercisesOrder.length
  110. ? exercisesOrder
  111. .map((id) => flatExercises.find((item) => item.id === id))
  112. .filter(Boolean)
  113. .map((item) => item!.exercise)
  114. : flatExercises.map((item) => item.exercise);
  115. const handleStartWorkout = () => {
  116. if (orderedExercises.length > 0) {
  117. startWorkout(orderedExercises, selectedEquipment, selectedMuscles);
  118. }
  119. };
  120. const [showCongrats, setShowCongrats] = useState(false);
  121. const goToProfile = () => {
  122. router.push("/profile");
  123. };
  124. const handleToggleEquipment = (equipment: ExerciseAttributeValueEnum) => {
  125. toggleEquipment(equipment);
  126. if (fromSession) setFromSession(null);
  127. };
  128. const handleClearEquipment = () => {
  129. clearEquipment();
  130. if (fromSession) setFromSession(null);
  131. };
  132. const handleToggleMuscle = (muscle: ExerciseAttributeValueEnum) => {
  133. toggleMuscle(muscle);
  134. if (fromSession) setFromSession(null);
  135. };
  136. if (showCongrats && !isWorkoutActive) {
  137. return (
  138. <div className="flex flex-col items-center justify-center py-16 h-full">
  139. <Image alt="Trophée" className="w-56 h-56" src={Trophy} />
  140. <h2 className="text-2xl font-bold mb-2">{t("workout_builder.session.congrats")}</h2>
  141. <p className="text-lg text-slate-600 mb-6">{t("workout_builder.session.congrats_subtitle")}</p>
  142. <Button onClick={goToProfile}>{t("commons.go_to_profile")}</Button>
  143. </div>
  144. );
  145. }
  146. if (isWorkoutActive && session) {
  147. return (
  148. <div className="w-full max-w-6xl mx-auto">
  149. {!showCongrats && (
  150. <WorkoutSessionHeader
  151. currentExerciseIndex={session.exercises.findIndex((exercise) => exercise.id === currentExercise?.id)}
  152. elapsedTime={formatElapsedTime()}
  153. isTimerRunning={isTimerRunning}
  154. onQuitWorkout={quitWorkout}
  155. onResetTimer={resetTimer}
  156. onToggleTimer={toggleTimer}
  157. />
  158. )}
  159. <WorkoutSessionSets isWorkoutActive={isWorkoutActive} onCongrats={() => setShowCongrats(true)} showCongrats={showCongrats} />
  160. </div>
  161. );
  162. }
  163. const STEPPER_STEPS: StepperStepProps[] = [
  164. {
  165. stepNumber: 1,
  166. title: t("workout_builder.steps.equipment.title"),
  167. description: t("workout_builder.steps.equipment.description"),
  168. isActive: false,
  169. isCompleted: false,
  170. },
  171. {
  172. stepNumber: 2,
  173. title: t("workout_builder.steps.muscles.title"),
  174. description: t("workout_builder.steps.muscles.description"),
  175. isActive: false,
  176. isCompleted: false,
  177. },
  178. {
  179. stepNumber: 3,
  180. title: t("workout_builder.steps.exercises.title"),
  181. description: t("workout_builder.steps.exercises.description"),
  182. isActive: false,
  183. isCompleted: false,
  184. },
  185. ];
  186. const steps = STEPPER_STEPS.map((step) => ({
  187. ...step,
  188. isActive: step.stepNumber === currentStep,
  189. isCompleted: step.stepNumber < currentStep,
  190. }));
  191. const renderStepContent = () => {
  192. switch (currentStep) {
  193. case 1:
  194. return (
  195. <EquipmentSelection
  196. onClearEquipment={handleClearEquipment}
  197. onToggleEquipment={handleToggleEquipment}
  198. selectedEquipment={selectedEquipment}
  199. />
  200. );
  201. case 2:
  202. return (
  203. <MuscleSelection onToggleMuscle={handleToggleMuscle} selectedEquipment={selectedEquipment} selectedMuscles={selectedMuscles} />
  204. );
  205. case 3:
  206. return (
  207. <ExercisesSelection
  208. error={exercisesError}
  209. exercisesByMuscle={exercisesByMuscle}
  210. isLoading={isLoadingExercises}
  211. isShuffling={isShuffling}
  212. onAdd={handleAddExercise}
  213. onDelete={handleDeleteExercise}
  214. onPick={handlePickExercise}
  215. onShuffle={handleShuffleExercise}
  216. />
  217. );
  218. default:
  219. return null;
  220. }
  221. };
  222. return (
  223. <div className="w-full max-w-6xl mx-auto h-full">
  224. <StepperHeader steps={steps} />
  225. <div className="px-2 sm:px-6">{renderStepContent()}</div>
  226. <WorkoutBuilderFooter
  227. canContinue={canContinue}
  228. currentStep={currentStep}
  229. onNext={nextStep}
  230. onPrevious={prevStep}
  231. onStartWorkout={handleStartWorkout}
  232. selectedEquipment={selectedEquipment}
  233. selectedMuscles={selectedMuscles}
  234. totalSteps={STEPPER_STEPS.length}
  235. />
  236. </div>
  237. );
  238. }