@wger-project/react-components 25.10.16
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +661 -0
- package/README.md +58 -0
- package/build/OpenSans-Bold.woff +0 -0
- package/build/OpenSans-Light.woff +0 -0
- package/build/assets/index.css +1 -0
- package/build/chunks/browser-ponyfill-DL_vVusK.js +3 -0
- package/build/chunks/browser-ponyfill-DL_vVusK.js.map +1 -0
- package/build/favicon.ico +0 -0
- package/build/index.html +60 -0
- package/build/locales/README.md +10 -0
- package/build/locales/ar/translation.json +240 -0
- package/build/locales/ca/translation.json +35 -0
- package/build/locales/cs/translation.json +257 -0
- package/build/locales/de/translation.json +338 -0
- package/build/locales/el/translation.json +97 -0
- package/build/locales/en/translation.json +332 -0
- package/build/locales/es/translation.json +335 -0
- package/build/locales/fa/translation.json +127 -0
- package/build/locales/fr/translation.json +341 -0
- package/build/locales/he/translation.json +146 -0
- package/build/locales/hr/translation.json +249 -0
- package/build/locales/it/translation.json +338 -0
- package/build/locales/ko/translation.json +331 -0
- package/build/locales/nb_NO/translation.json +50 -0
- package/build/locales/nl/translation.json +242 -0
- package/build/locales/pl/translation.json +261 -0
- package/build/locales/pt/translation.json +342 -0
- package/build/locales/pt_BR/translation.json +259 -0
- package/build/locales/pt_PT/translation.json +334 -0
- package/build/locales/ru/translation.json +116 -0
- package/build/locales/sk/translation.json +10 -0
- package/build/locales/sl/translation.json +326 -0
- package/build/locales/sr/translation.json +240 -0
- package/build/locales/sv/translation.json +165 -0
- package/build/locales/ta/translation.json +331 -0
- package/build/locales/th/translation.json +27 -0
- package/build/locales/tr/translation.json +238 -0
- package/build/locales/uk/translation.json +339 -0
- package/build/locales/uz/translation.json +56 -0
- package/build/locales/vi/translation.json +70 -0
- package/build/locales/zh_Hans/translation.json +336 -0
- package/build/locales/zh_Hant/translation.json +246 -0
- package/build/logo192.png +0 -0
- package/build/logo512.png +0 -0
- package/build/main.js +542 -0
- package/build/main.js.map +1 -0
- package/build/manifest.json +25 -0
- package/build/muscles/SOURCES +2 -0
- package/build/muscles/main/muscle-1.svg +72 -0
- package/build/muscles/main/muscle-10.svg +73 -0
- package/build/muscles/main/muscle-11.svg +72 -0
- package/build/muscles/main/muscle-12.svg +72 -0
- package/build/muscles/main/muscle-13.svg +92 -0
- package/build/muscles/main/muscle-14.svg +73 -0
- package/build/muscles/main/muscle-15.svg +72 -0
- package/build/muscles/main/muscle-16.svg +73 -0
- package/build/muscles/main/muscle-2.svg +72 -0
- package/build/muscles/main/muscle-3.svg +72 -0
- package/build/muscles/main/muscle-4.svg +72 -0
- package/build/muscles/main/muscle-5.svg +72 -0
- package/build/muscles/main/muscle-6.svg +73 -0
- package/build/muscles/main/muscle-7.svg +72 -0
- package/build/muscles/main/muscle-8.svg +72 -0
- package/build/muscles/main/muscle-9.svg +72 -0
- package/build/muscles/muscular_system_back.svg +118 -0
- package/build/muscles/muscular_system_front.svg +99 -0
- package/build/muscles/secondary/muscle-1.svg +72 -0
- package/build/muscles/secondary/muscle-10.svg +73 -0
- package/build/muscles/secondary/muscle-11.svg +72 -0
- package/build/muscles/secondary/muscle-12.svg +72 -0
- package/build/muscles/secondary/muscle-13.svg +92 -0
- package/build/muscles/secondary/muscle-14.svg +73 -0
- package/build/muscles/secondary/muscle-15.svg +72 -0
- package/build/muscles/secondary/muscle-16.svg +73 -0
- package/build/muscles/secondary/muscle-2.svg +72 -0
- package/build/muscles/secondary/muscle-3.svg +72 -0
- package/build/muscles/secondary/muscle-4.svg +72 -0
- package/build/muscles/secondary/muscle-5.svg +72 -0
- package/build/muscles/secondary/muscle-6.svg +73 -0
- package/build/muscles/secondary/muscle-7.svg +72 -0
- package/build/muscles/secondary/muscle-8.svg +72 -0
- package/build/muscles/secondary/muscle-9.svg +72 -0
- package/build/robots.txt +3 -0
- package/index.html +59 -0
- package/package.json +125 -0
- package/src/.sass-cache/166f82406a0edab65b7f8fd8256819ee2f34bebf/App.module.scssc +0 -0
- package/src/@types/i18next.d.ts +13 -0
- package/src/App.tsx +25 -0
- package/src/assets/images/discord-logo.svg +1 -0
- package/src/assets/images/get-it-on-google-play.svg +1 -0
- package/src/assets/images/logo.png +0 -0
- package/src/assets/images/twitter-logo-50.png +0 -0
- package/src/assets/images/twitter.svg +5 -0
- package/src/components/BodyWeight/Form/WeightForm.test.tsx +100 -0
- package/src/components/BodyWeight/Form/WeightForm.tsx +102 -0
- package/src/components/BodyWeight/Table/ActionButton/ActionButton.test.tsx +65 -0
- package/src/components/BodyWeight/Table/ActionButton/ActionButton.tsx +67 -0
- package/src/components/BodyWeight/Table/Fab/Fab.tsx +35 -0
- package/src/components/BodyWeight/Table/index.test.tsx +33 -0
- package/src/components/BodyWeight/Table/index.tsx +98 -0
- package/src/components/BodyWeight/Table/table.module.css +5 -0
- package/src/components/BodyWeight/Table/table.module.css.map +1 -0
- package/src/components/BodyWeight/Table/table.module.scss +0 -0
- package/src/components/BodyWeight/Table/table.mosule.css +2 -0
- package/src/components/BodyWeight/Table/table.mosule.css.map +1 -0
- package/src/components/BodyWeight/TableDashboard/TableDashboard.test.tsx +24 -0
- package/src/components/BodyWeight/TableDashboard/TableDashboard.tsx +62 -0
- package/src/components/BodyWeight/WeightChart/index.test.tsx +66 -0
- package/src/components/BodyWeight/WeightChart/index.tsx +105 -0
- package/src/components/BodyWeight/body_weight.module.css +8 -0
- package/src/components/BodyWeight/body_weight.module.css.map +1 -0
- package/src/components/BodyWeight/body_weight.module.scss +8 -0
- package/src/components/BodyWeight/index.test.tsx +92 -0
- package/src/components/BodyWeight/index.tsx +40 -0
- package/src/components/BodyWeight/model.ts +40 -0
- package/src/components/BodyWeight/queries/index.ts +49 -0
- package/src/components/BodyWeight/utils.test.ts +44 -0
- package/src/components/BodyWeight/utils.ts +22 -0
- package/src/components/BodyWeight/widgets/FilterButtons.test.tsx +57 -0
- package/src/components/BodyWeight/widgets/FilterButtons.tsx +69 -0
- package/src/components/BodyWeight/widgets/fab.tsx +33 -0
- package/src/components/Calendar/Components/CalendarComponent.test.tsx +141 -0
- package/src/components/Calendar/Components/CalendarComponent.tsx +250 -0
- package/src/components/Calendar/Components/CalendarDay.tsx +105 -0
- package/src/components/Calendar/Components/CalendarDayGrid.tsx +53 -0
- package/src/components/Calendar/Components/CalendarHeader.tsx +39 -0
- package/src/components/Calendar/Components/Entries.test.tsx +81 -0
- package/src/components/Calendar/Components/Entries.tsx +163 -0
- package/src/components/Calendar/Helpers/CalendarMeasurement.tsx +10 -0
- package/src/components/Carousel/carousel.module.css +43 -0
- package/src/components/Carousel/carousel.module.css.map +1 -0
- package/src/components/Carousel/carousel.module.scss +46 -0
- package/src/components/Carousel/index.tsx +66 -0
- package/src/components/Common/forms/LicenseAuthor.tsx +19 -0
- package/src/components/Common/forms/LicenseAuthorUrl.tsx +20 -0
- package/src/components/Common/forms/LicenseDerivativeSourceUrl.tsx +20 -0
- package/src/components/Common/forms/LicenseObjectUrl.tsx +20 -0
- package/src/components/Common/forms/LicenseTitle.tsx +19 -0
- package/src/components/Common/forms/WgerTextField.tsx +27 -0
- package/src/components/Core/LoadingWidget/LoadingWidget.tsx +22 -0
- package/src/components/Core/Modals/DeleteConfirmationModal.test.tsx +57 -0
- package/src/components/Core/Modals/DeleteConfirmationModal.tsx +74 -0
- package/src/components/Core/Modals/WgerModal.test.tsx +45 -0
- package/src/components/Core/Modals/WgerModal.tsx +46 -0
- package/src/components/Core/Notifications/index.tsx +57 -0
- package/src/components/Core/Notifications/notifications.module.css +8 -0
- package/src/components/Core/Notifications/notifications.module.css.map +1 -0
- package/src/components/Core/Notifications/notifications.module.scss +6 -0
- package/src/components/Core/Tooltips/LightToolTip.tsx +14 -0
- package/src/components/Core/Widgets/Container.tsx +98 -0
- package/src/components/Core/Widgets/FormError.tsx +64 -0
- package/src/components/Core/Widgets/OverviewEmpty.tsx +26 -0
- package/src/components/Core/Widgets/RenderLoadingQuery.tsx +28 -0
- package/src/components/Core/queries/index.ts +2 -0
- package/src/components/Core/queries/languageCheck.ts +31 -0
- package/src/components/Dashboard/Dashboard.tsx +22 -0
- package/src/components/Dashboard/EmptyCard.tsx +56 -0
- package/src/components/Dashboard/NutritionCard.test.tsx +89 -0
- package/src/components/Dashboard/NutritionCard.tsx +178 -0
- package/src/components/Dashboard/RoutineCard.test.tsx +66 -0
- package/src/components/Dashboard/RoutineCard.tsx +107 -0
- package/src/components/Dashboard/WeightCard.test.tsx +81 -0
- package/src/components/Dashboard/WeightCard.tsx +72 -0
- package/src/components/Exercises/Add/AddExerciseStepper.test.tsx +26 -0
- package/src/components/Exercises/Add/AddExerciseStepper.tsx +97 -0
- package/src/components/Exercises/Add/NotEnoughRights.tsx +37 -0
- package/src/components/Exercises/Add/Step1Basics.test.tsx +133 -0
- package/src/components/Exercises/Add/Step1Basics.tsx +179 -0
- package/src/components/Exercises/Add/Step2Variations.test.tsx +149 -0
- package/src/components/Exercises/Add/Step2Variations.tsx +237 -0
- package/src/components/Exercises/Add/Step3Description.test.tsx +66 -0
- package/src/components/Exercises/Add/Step3Description.tsx +96 -0
- package/src/components/Exercises/Add/Step4Translation.test.tsx +119 -0
- package/src/components/Exercises/Add/Step4Translations.tsx +183 -0
- package/src/components/Exercises/Add/Step5Images.test.tsx +28 -0
- package/src/components/Exercises/Add/Step5Images.tsx +269 -0
- package/src/components/Exercises/Add/Step6Overview.test.tsx +128 -0
- package/src/components/Exercises/Add/Step6Overview.tsx +262 -0
- package/src/components/Exercises/Detail/ExerciseDetailEdit.test.tsx +248 -0
- package/src/components/Exercises/Detail/ExerciseDetailEdit.tsx +340 -0
- package/src/components/Exercises/Detail/ExerciseDetailView.test.tsx +96 -0
- package/src/components/Exercises/Detail/ExerciseDetailView.tsx +242 -0
- package/src/components/Exercises/Detail/ExerciseDetails.test.tsx +88 -0
- package/src/components/Exercises/Detail/ExerciseDetails.tsx +119 -0
- package/src/components/Exercises/Detail/ExerciseImageAvatar.tsx +20 -0
- package/src/components/Exercises/Detail/ExerciseImagePlaceholder.tsx +21 -0
- package/src/components/Exercises/Detail/Head/ExerciseDeleteDialog.test.tsx +125 -0
- package/src/components/Exercises/Detail/Head/ExerciseDeleteDialog.tsx +185 -0
- package/src/components/Exercises/Detail/Head/head.module.css +73 -0
- package/src/components/Exercises/Detail/Head/head.module.css.map +7 -0
- package/src/components/Exercises/Detail/Head/head.module.scss +111 -0
- package/src/components/Exercises/Detail/Head/index.tsx +175 -0
- package/src/components/Exercises/Detail/OverviewCard.test.tsx +98 -0
- package/src/components/Exercises/Detail/OverviewCard.tsx +68 -0
- package/src/components/Exercises/Detail/SideGallery/index.tsx +65 -0
- package/src/components/Exercises/Detail/SideGallery/side_gallery.module.css +26 -0
- package/src/components/Exercises/Detail/SideGallery/side_gallery.module.css.map +7 -0
- package/src/components/Exercises/Detail/SideGallery/side_gallery.module.scss +34 -0
- package/src/components/Exercises/ExerciseOverview.test.tsx +183 -0
- package/src/components/Exercises/ExerciseOverview.tsx +293 -0
- package/src/components/Exercises/Filter/CategoryFilter.test.tsx +57 -0
- package/src/components/Exercises/Filter/CategoryFilter.tsx +105 -0
- package/src/components/Exercises/Filter/EquipmentFilter.test.tsx +55 -0
- package/src/components/Exercises/Filter/EquipmentFilter.tsx +105 -0
- package/src/components/Exercises/Filter/ExerciseFiltersContext.tsx +16 -0
- package/src/components/Exercises/Filter/FilterDrawer.tsx +38 -0
- package/src/components/Exercises/Filter/MuscleFilter.test.tsx +50 -0
- package/src/components/Exercises/Filter/MuscleFilter.tsx +130 -0
- package/src/components/Exercises/Filter/NameAutcompleter.tsx +146 -0
- package/src/components/Exercises/Filter/NameAutocompleter.test.tsx +76 -0
- package/src/components/Exercises/Overview/ExerciseGrid.tsx +44 -0
- package/src/components/Exercises/Overview/ExerciseGridLoadingSkeleton.test.tsx +10 -0
- package/src/components/Exercises/Overview/ExerciseGridLoadingSkeleton.tsx +26 -0
- package/src/components/Exercises/forms/Category.test.tsx +49 -0
- package/src/components/Exercises/forms/Category.tsx +41 -0
- package/src/components/Exercises/forms/Equipment.test.tsx +84 -0
- package/src/components/Exercises/forms/Equipment.tsx +38 -0
- package/src/components/Exercises/forms/ExerciseAliases.tsx +37 -0
- package/src/components/Exercises/forms/ExerciseDescription.tsx +57 -0
- package/src/components/Exercises/forms/ExerciseEquipmentSelect.tsx +29 -0
- package/src/components/Exercises/forms/ExerciseName.tsx +19 -0
- package/src/components/Exercises/forms/ExerciseNotes.tsx +72 -0
- package/src/components/Exercises/forms/ExerciseSelect.tsx +28 -0
- package/src/components/Exercises/forms/ImageCard.tsx +101 -0
- package/src/components/Exercises/forms/ImageStyle.tsx +74 -0
- package/src/components/Exercises/forms/Muscle.test.tsx +74 -0
- package/src/components/Exercises/forms/Muscle.tsx +41 -0
- package/src/components/Exercises/forms/VideoCard.tsx +109 -0
- package/src/components/Exercises/forms/yupValidators.ts +46 -0
- package/src/components/Exercises/models/alias.ts +29 -0
- package/src/components/Exercises/models/category.ts +35 -0
- package/src/components/Exercises/models/equipment.ts +33 -0
- package/src/components/Exercises/models/exercise.test.ts +74 -0
- package/src/components/Exercises/models/exercise.ts +159 -0
- package/src/components/Exercises/models/image.ts +41 -0
- package/src/components/Exercises/models/language.ts +23 -0
- package/src/components/Exercises/models/muscle.test.ts +17 -0
- package/src/components/Exercises/models/muscle.ts +39 -0
- package/src/components/Exercises/models/note.ts +29 -0
- package/src/components/Exercises/models/translation.test.ts +68 -0
- package/src/components/Exercises/models/translation.ts +83 -0
- package/src/components/Exercises/models/video.ts +30 -0
- package/src/components/Exercises/queries/exercises.ts +29 -0
- package/src/components/Exercises/queries/index.ts +133 -0
- package/src/components/Exercises/queries/variations.ts +15 -0
- package/src/components/Header/SubMenus/BodyWeightSubMenu.tsx +27 -0
- package/src/components/Header/SubMenus/MeasurementsSubMenu.tsx +28 -0
- package/src/components/Header/SubMenus/NutritionSubMenu.tsx +28 -0
- package/src/components/Header/SubMenus/TrainingSubMenu.tsx +43 -0
- package/src/components/Header/SubMenus/WorkoutSubMenu.tsx +28 -0
- package/src/components/Header/index.tsx +27 -0
- package/src/components/Measurements/Screens/MeasurementCategoryDetail.test.tsx +67 -0
- package/src/components/Measurements/Screens/MeasurementCategoryDetail.tsx +37 -0
- package/src/components/Measurements/Screens/MeasurementCategoryOverview.test.tsx +59 -0
- package/src/components/Measurements/Screens/MeasurementCategoryOverview.tsx +64 -0
- package/src/components/Measurements/models/Category.ts +38 -0
- package/src/components/Measurements/models/Entry.ts +37 -0
- package/src/components/Measurements/queries/index.ts +121 -0
- package/src/components/Measurements/widgets/CategoryDetailDataGrid.tsx +209 -0
- package/src/components/Measurements/widgets/CategoryDetailDropdown.tsx +83 -0
- package/src/components/Measurements/widgets/CategoryForm.test.tsx +96 -0
- package/src/components/Measurements/widgets/CategoryForm.tsx +88 -0
- package/src/components/Measurements/widgets/EntryForm.test.tsx +120 -0
- package/src/components/Measurements/widgets/EntryForm.tsx +139 -0
- package/src/components/Measurements/widgets/MeasurementChart.tsx +52 -0
- package/src/components/Measurements/widgets/fab.tsx +65 -0
- package/src/components/Muscles/MuscleOverview.tsx +37 -0
- package/src/components/Nutrition/components/BmiCalculator.test.tsx +55 -0
- package/src/components/Nutrition/components/BmiCalculator.tsx +203 -0
- package/src/components/Nutrition/components/IngredientSearch.tsx +24 -0
- package/src/components/Nutrition/components/NutritionDiaryOverview.test.tsx +51 -0
- package/src/components/Nutrition/components/NutritionDiaryOverview.tsx +47 -0
- package/src/components/Nutrition/components/PlanDetail.test.tsx +58 -0
- package/src/components/Nutrition/components/PlanDetail.tsx +100 -0
- package/src/components/Nutrition/components/PlanOverview.test.tsx +37 -0
- package/src/components/Nutrition/components/PlansOverview.tsx +62 -0
- package/src/components/Nutrition/helpers/nutritionalValues.test.ts +108 -0
- package/src/components/Nutrition/helpers/nutritionalValues.ts +128 -0
- package/src/components/Nutrition/models/Ingredient.ts +84 -0
- package/src/components/Nutrition/models/IngredientImage.ts +57 -0
- package/src/components/Nutrition/models/IngredientImageThumbnails.ts +59 -0
- package/src/components/Nutrition/models/diaryEntry.ts +129 -0
- package/src/components/Nutrition/models/ingredient.test.ts +37 -0
- package/src/components/Nutrition/models/meal.test.ts +54 -0
- package/src/components/Nutrition/models/meal.ts +128 -0
- package/src/components/Nutrition/models/mealItem.test.ts +23 -0
- package/src/components/Nutrition/models/mealItem.ts +134 -0
- package/src/components/Nutrition/models/nutritionalPlan.test.ts +169 -0
- package/src/components/Nutrition/models/nutritionalPlan.ts +268 -0
- package/src/components/Nutrition/models/weightUnit.ts +24 -0
- package/src/components/Nutrition/queries/diary.ts +59 -0
- package/src/components/Nutrition/queries/index.ts +20 -0
- package/src/components/Nutrition/queries/ingredient.ts +12 -0
- package/src/components/Nutrition/queries/meal.ts +41 -0
- package/src/components/Nutrition/queries/mealItem.ts +43 -0
- package/src/components/Nutrition/queries/plan.ts +93 -0
- package/src/components/Nutrition/widgets/DiaryDetail.tsx +39 -0
- package/src/components/Nutrition/widgets/DiaryOverview.tsx +51 -0
- package/src/components/Nutrition/widgets/Fab.tsx +65 -0
- package/src/components/Nutrition/widgets/IngredientAutcompleter.tsx +140 -0
- package/src/components/Nutrition/widgets/IngredientAutocompleter.test.tsx +60 -0
- package/src/components/Nutrition/widgets/IngredientDetailTable.tsx +91 -0
- package/src/components/Nutrition/widgets/LoggedPlannedNutritionalValuesTable.tsx +138 -0
- package/src/components/Nutrition/widgets/MealDetail.tsx +170 -0
- package/src/components/Nutrition/widgets/MealDetailDropdown.tsx +117 -0
- package/src/components/Nutrition/widgets/NutritionalValuesTable.tsx +103 -0
- package/src/components/Nutrition/widgets/PlanDetailDropdown.tsx +82 -0
- package/src/components/Nutrition/widgets/PlanSidebar.tsx +44 -0
- package/src/components/Nutrition/widgets/charts/LinearPlannedLoggedChart.tsx +27 -0
- package/src/components/Nutrition/widgets/charts/MacrosPieChart.tsx +66 -0
- package/src/components/Nutrition/widgets/charts/NutritionDiaryChart.tsx +95 -0
- package/src/components/Nutrition/widgets/charts/NutritionalValuesDashboardChart.tsx +84 -0
- package/src/components/Nutrition/widgets/charts/NutritionalValuesPlannedLoggedChart.tsx +54 -0
- package/src/components/Nutrition/widgets/forms/MealForm.test.tsx +85 -0
- package/src/components/Nutrition/widgets/forms/MealForm.tsx +105 -0
- package/src/components/Nutrition/widgets/forms/MealItemForm.test.tsx +106 -0
- package/src/components/Nutrition/widgets/forms/MealItemForm.tsx +118 -0
- package/src/components/Nutrition/widgets/forms/NutritionDiaryEntryForm.test.tsx +165 -0
- package/src/components/Nutrition/widgets/forms/NutritionDiaryEntryForm.tsx +161 -0
- package/src/components/Nutrition/widgets/forms/PlanForm.test.tsx +101 -0
- package/src/components/Nutrition/widgets/forms/PlanForm.tsx +361 -0
- package/src/components/User/models/profile.ts +81 -0
- package/src/components/User/queries/contribute.test.ts +132 -0
- package/src/components/User/queries/contribute.ts +33 -0
- package/src/components/User/queries/permission.ts +11 -0
- package/src/components/User/queries/profile.ts +22 -0
- package/src/components/WorkoutRoutines/Detail/RoutineAdd.tsx +12 -0
- package/src/components/WorkoutRoutines/Detail/RoutineDetail.test.tsx +71 -0
- package/src/components/WorkoutRoutines/Detail/RoutineDetail.tsx +87 -0
- package/src/components/WorkoutRoutines/Detail/RoutineDetailsTable.test.tsx +83 -0
- package/src/components/WorkoutRoutines/Detail/RoutineDetailsTable.tsx +371 -0
- package/src/components/WorkoutRoutines/Detail/RoutineEdit.test.tsx +51 -0
- package/src/components/WorkoutRoutines/Detail/RoutineEdit.tsx +95 -0
- package/src/components/WorkoutRoutines/Detail/SessionAdd.test.tsx +43 -0
- package/src/components/WorkoutRoutines/Detail/SessionAdd.tsx +50 -0
- package/src/components/WorkoutRoutines/Detail/SlotProgressionEdit.test.tsx +41 -0
- package/src/components/WorkoutRoutines/Detail/SlotProgressionEdit.tsx +145 -0
- package/src/components/WorkoutRoutines/Detail/TemplateDetail.test.tsx +44 -0
- package/src/components/WorkoutRoutines/Detail/TemplateDetail.tsx +84 -0
- package/src/components/WorkoutRoutines/Detail/WorkoutLogs.test.tsx +64 -0
- package/src/components/WorkoutRoutines/Detail/WorkoutLogs.tsx +98 -0
- package/src/components/WorkoutRoutines/Detail/WorkoutStats.test.tsx +65 -0
- package/src/components/WorkoutRoutines/Detail/WorkoutStats.tsx +244 -0
- package/src/components/WorkoutRoutines/Overview/Fab.tsx +91 -0
- package/src/components/WorkoutRoutines/Overview/PrivateTemplateOverview.test.tsx +36 -0
- package/src/components/WorkoutRoutines/Overview/PrivateTemplateOverview.tsx +41 -0
- package/src/components/WorkoutRoutines/Overview/PublicTemplateOverview.test.tsx +36 -0
- package/src/components/WorkoutRoutines/Overview/PublicTemplateOverview.tsx +40 -0
- package/src/components/WorkoutRoutines/Overview/RoutineOverview.test.tsx +38 -0
- package/src/components/WorkoutRoutines/Overview/RoutineOverview.tsx +76 -0
- package/src/components/WorkoutRoutines/models/BaseConfig.ts +118 -0
- package/src/components/WorkoutRoutines/models/Day.test.ts +15 -0
- package/src/components/WorkoutRoutines/models/Day.ts +117 -0
- package/src/components/WorkoutRoutines/models/LogStats.test.ts +44 -0
- package/src/components/WorkoutRoutines/models/LogStats.ts +110 -0
- package/src/components/WorkoutRoutines/models/RepetitionUnit.ts +22 -0
- package/src/components/WorkoutRoutines/models/Routine.test.ts +113 -0
- package/src/components/WorkoutRoutines/models/Routine.ts +218 -0
- package/src/components/WorkoutRoutines/models/RoutineDayData.ts +38 -0
- package/src/components/WorkoutRoutines/models/RoutineLogData.ts +22 -0
- package/src/components/WorkoutRoutines/models/SetConfigData.ts +121 -0
- package/src/components/WorkoutRoutines/models/Slot.test.ts +19 -0
- package/src/components/WorkoutRoutines/models/Slot.ts +83 -0
- package/src/components/WorkoutRoutines/models/SlotData.ts +30 -0
- package/src/components/WorkoutRoutines/models/SlotEntry.test.ts +31 -0
- package/src/components/WorkoutRoutines/models/SlotEntry.ts +245 -0
- package/src/components/WorkoutRoutines/models/WeightUnit.ts +22 -0
- package/src/components/WorkoutRoutines/models/WorkoutLog.ts +159 -0
- package/src/components/WorkoutRoutines/models/WorkoutSession.ts +124 -0
- package/src/components/WorkoutRoutines/queries/configs.ts +393 -0
- package/src/components/WorkoutRoutines/queries/days.ts +42 -0
- package/src/components/WorkoutRoutines/queries/index.ts +64 -0
- package/src/components/WorkoutRoutines/queries/logs.ts +49 -0
- package/src/components/WorkoutRoutines/queries/routines.ts +121 -0
- package/src/components/WorkoutRoutines/queries/sessions.ts +41 -0
- package/src/components/WorkoutRoutines/queries/slot_entries.ts +33 -0
- package/src/components/WorkoutRoutines/queries/slots.ts +40 -0
- package/src/components/WorkoutRoutines/queries/units.ts +15 -0
- package/src/components/WorkoutRoutines/widgets/DayDetails.test.tsx +69 -0
- package/src/components/WorkoutRoutines/widgets/DayDetails.tsx +441 -0
- package/src/components/WorkoutRoutines/widgets/LogWidgets.test.tsx +91 -0
- package/src/components/WorkoutRoutines/widgets/LogWidgets.tsx +334 -0
- package/src/components/WorkoutRoutines/widgets/RoutineDetailDropdown.test.tsx +36 -0
- package/src/components/WorkoutRoutines/widgets/RoutineDetailDropdown.tsx +184 -0
- package/src/components/WorkoutRoutines/widgets/RoutineDetailsCard.test.tsx +42 -0
- package/src/components/WorkoutRoutines/widgets/RoutineDetailsCard.tsx +202 -0
- package/src/components/WorkoutRoutines/widgets/RoutineStatistics.test.tsx +364 -0
- package/src/components/WorkoutRoutines/widgets/RoutineStatistics.tsx +249 -0
- package/src/components/WorkoutRoutines/widgets/SlotDetails.test.tsx +72 -0
- package/src/components/WorkoutRoutines/widgets/SlotDetails.tsx +233 -0
- package/src/components/WorkoutRoutines/widgets/forms/BaseConfigForm.test.tsx +161 -0
- package/src/components/WorkoutRoutines/widgets/forms/BaseConfigForm.tsx +508 -0
- package/src/components/WorkoutRoutines/widgets/forms/DayForm.test.tsx +86 -0
- package/src/components/WorkoutRoutines/widgets/forms/DayForm.tsx +202 -0
- package/src/components/WorkoutRoutines/widgets/forms/ProgressionForm.test.tsx +176 -0
- package/src/components/WorkoutRoutines/widgets/forms/ProgressionForm.tsx +494 -0
- package/src/components/WorkoutRoutines/widgets/forms/RoutineForm.test.tsx +94 -0
- package/src/components/WorkoutRoutines/widgets/forms/RoutineForm.tsx +309 -0
- package/src/components/WorkoutRoutines/widgets/forms/RoutineTemplateForm.test.tsx +70 -0
- package/src/components/WorkoutRoutines/widgets/forms/RoutineTemplateForm.tsx +68 -0
- package/src/components/WorkoutRoutines/widgets/forms/SessionForm.test.tsx +164 -0
- package/src/components/WorkoutRoutines/widgets/forms/SessionForm.tsx +263 -0
- package/src/components/WorkoutRoutines/widgets/forms/SessionLogsForm.test.tsx +126 -0
- package/src/components/WorkoutRoutines/widgets/forms/SessionLogsForm.tsx +330 -0
- package/src/components/WorkoutRoutines/widgets/forms/SlotEntryForm.test.tsx +227 -0
- package/src/components/WorkoutRoutines/widgets/forms/SlotEntryForm.tsx +206 -0
- package/src/components/WorkoutRoutines/widgets/forms/SlotForm.test.tsx +44 -0
- package/src/components/WorkoutRoutines/widgets/forms/SlotForm.tsx +36 -0
- package/src/components/index.ts +3 -0
- package/src/config.ts +12 -0
- package/src/i18n.ts +63 -0
- package/src/i18n.tsx +52 -0
- package/src/index.css +13 -0
- package/src/index.tsx +184 -0
- package/src/locales/README.md +7 -0
- package/src/pages/About/index.tsx +9 -0
- package/src/pages/AddExercise/index.tsx +15 -0
- package/src/pages/AddWeight/index.tsx +9 -0
- package/src/pages/ApiPage/index.tsx +9 -0
- package/src/pages/Calendar/index.tsx +10 -0
- package/src/pages/CaloriesCalculator/index.tsx +9 -0
- package/src/pages/Equipments/index.tsx +9 -0
- package/src/pages/ExerciseDetails/index.tsx +6 -0
- package/src/pages/Ingrdedients/index.tsx +9 -0
- package/src/pages/Login/index.tsx +9 -0
- package/src/pages/Preferences/index.tsx +9 -0
- package/src/pages/PublicTemplate/index.tsx +9 -0
- package/src/pages/TemplatePage/index.tsx +9 -0
- package/src/pages/WeightOverview/index.tsx +6 -0
- package/src/pages/index.ts +14 -0
- package/src/permissions/index.ts +13 -0
- package/src/reportWebVitals.ts +15 -0
- package/src/routes.tsx +137 -0
- package/src/services/alias.test.ts +47 -0
- package/src/services/alias.ts +34 -0
- package/src/services/base_config.test.ts +48 -0
- package/src/services/base_config.ts +86 -0
- package/src/services/category.test.ts +44 -0
- package/src/services/category.ts +21 -0
- package/src/services/config.test.ts +213 -0
- package/src/services/config.ts +50 -0
- package/src/services/day.test.ts +118 -0
- package/src/services/day.ts +44 -0
- package/src/services/equipment.test.ts +46 -0
- package/src/services/equipment.ts +20 -0
- package/src/services/exercise.test.ts +98 -0
- package/src/services/exercise.ts +182 -0
- package/src/services/exerciseTranslation.test.ts +98 -0
- package/src/services/exerciseTranslation.ts +112 -0
- package/src/services/image.test.ts +77 -0
- package/src/services/image.ts +54 -0
- package/src/services/index.ts +104 -0
- package/src/services/ingredient.test.ts +81 -0
- package/src/services/ingredient.ts +70 -0
- package/src/services/ingredientweightunit.ts +29 -0
- package/src/services/language.test.ts +86 -0
- package/src/services/language.ts +35 -0
- package/src/services/languageCheck.ts +33 -0
- package/src/services/meal.ts +67 -0
- package/src/services/mealItem.ts +32 -0
- package/src/services/measurements.test.ts +106 -0
- package/src/services/measurements.ts +173 -0
- package/src/services/muscles.test.ts +46 -0
- package/src/services/muscles.ts +20 -0
- package/src/services/note.ts +39 -0
- package/src/services/nutritionalDiary.test.ts +35 -0
- package/src/services/nutritionalDiary.ts +70 -0
- package/src/services/nutritionalPlan.test.ts +74 -0
- package/src/services/nutritionalPlan.ts +104 -0
- package/src/services/permission.ts +26 -0
- package/src/services/permissions.test.ts +34 -0
- package/src/services/profile.test.ts +40 -0
- package/src/services/profile.ts +49 -0
- package/src/services/responseType.ts +35 -0
- package/src/services/routine.test.ts +195 -0
- package/src/services/routine.ts +250 -0
- package/src/services/session.test.ts +157 -0
- package/src/services/session.ts +99 -0
- package/src/services/slot.test.ts +68 -0
- package/src/services/slot.ts +43 -0
- package/src/services/slot_entry.ts +43 -0
- package/src/services/variation.test.ts +24 -0
- package/src/services/variation.ts +18 -0
- package/src/services/video.test.ts +65 -0
- package/src/services/video.ts +47 -0
- package/src/services/weight.test.ts +75 -0
- package/src/services/weight.ts +58 -0
- package/src/services/workoutLogs.test.ts +148 -0
- package/src/services/workoutLogs.ts +89 -0
- package/src/services/workoutUnits.ts +29 -0
- package/src/setupTests.ts +40 -0
- package/src/state/exerciseSubmissionReducer.ts +170 -0
- package/src/state/exerciseSubmissionState.tsx +75 -0
- package/src/state/index.ts +15 -0
- package/src/state/notificationReducer.ts +29 -0
- package/src/state/notificationState.tsx +32 -0
- package/src/state/stateTypes.ts +24 -0
- package/src/tests/api/ingredientInfoEndpoint.ts +72 -0
- package/src/tests/exerciseTestdata.ts +159 -0
- package/src/tests/exercises/searchResponse.ts +23 -0
- package/src/tests/ingredientTestdata.ts +76 -0
- package/src/tests/measurementsTestData.ts +40 -0
- package/src/tests/nutritionDiaryTestdata.ts +167 -0
- package/src/tests/nutritionTestdata.ts +200 -0
- package/src/tests/queryClient/index.ts +7 -0
- package/src/tests/responseApi.ts +197 -0
- package/src/tests/setup.ts +10 -0
- package/src/tests/slotEntryApiResponse.ts +144 -0
- package/src/tests/userTestdata.ts +43 -0
- package/src/tests/weight/testData.ts +7 -0
- package/src/tests/workoutLogsRoutinesTestData.ts +90 -0
- package/src/tests/workoutRoutinesTestData.ts +451 -0
- package/src/tests/workoutStatisticsTestData.ts +219 -0
- package/src/theme.ts +73 -0
- package/src/types.ts +144 -0
- package/src/utils/Adapter.ts +10 -0
- package/src/utils/colors.test.ts +38 -0
- package/src/utils/colors.ts +17 -0
- package/src/utils/consts.ts +160 -0
- package/src/utils/date.test.ts +56 -0
- package/src/utils/date.ts +145 -0
- package/src/utils/forms.test.ts +41 -0
- package/src/utils/forms.ts +54 -0
- package/src/utils/numbers.test.ts +55 -0
- package/src/utils/numbers.ts +42 -0
- package/src/utils/requests.test.ts +61 -0
- package/src/utils/requests.ts +22 -0
- package/src/utils/strings.test.ts +62 -0
- package/src/utils/strings.ts +28 -0
- package/src/utils/url.test.ts +127 -0
- package/src/utils/url.ts +237 -0
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import { Exercise } from "components/Exercises/models/exercise";
|
|
3
|
+
import { WorkoutLog, WorkoutLogAdapter } from "components/WorkoutRoutines/models/WorkoutLog";
|
|
4
|
+
import {
|
|
5
|
+
AddSessionParams,
|
|
6
|
+
EditSessionParams,
|
|
7
|
+
WorkoutSession,
|
|
8
|
+
WorkoutSessionAdapter
|
|
9
|
+
} from "components/WorkoutRoutines/models/WorkoutSession";
|
|
10
|
+
import { getExercise } from "services";
|
|
11
|
+
import { API_MAX_PAGE_SIZE, ApiPath } from "utils/consts";
|
|
12
|
+
import { fetchPaginated } from "utils/requests";
|
|
13
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
14
|
+
|
|
15
|
+
export type SessionQueryOptions = {
|
|
16
|
+
filtersetQuerySessions?: object,
|
|
17
|
+
filtersetQueryLogs?: object,
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
21
|
+
export const searchSession = async (queryParams: Record<string, any>): Promise<WorkoutSession | null> => {
|
|
22
|
+
const response = await axios.get(
|
|
23
|
+
makeUrl(ApiPath.SESSION, { query: queryParams }),
|
|
24
|
+
{ headers: makeHeader() }
|
|
25
|
+
);
|
|
26
|
+
|
|
27
|
+
if (response.data.count === 1) {
|
|
28
|
+
return new WorkoutSessionAdapter().fromJson(response.data.results[0]);
|
|
29
|
+
}
|
|
30
|
+
|
|
31
|
+
return null;
|
|
32
|
+
};
|
|
33
|
+
|
|
34
|
+
export const getSessions = async (options?: SessionQueryOptions): Promise<WorkoutSession[]> => {
|
|
35
|
+
|
|
36
|
+
const { filtersetQuerySessions = {}, filtersetQueryLogs = {} } = options || {};
|
|
37
|
+
|
|
38
|
+
const sessionAdapter = new WorkoutSessionAdapter();
|
|
39
|
+
const logAdapter = new WorkoutLogAdapter();
|
|
40
|
+
const result: WorkoutSession[] = [];
|
|
41
|
+
const logs: WorkoutLog[] = [];
|
|
42
|
+
const exercises: Record<number, Exercise> = {};
|
|
43
|
+
|
|
44
|
+
// Fetch all logs first
|
|
45
|
+
for await (const logPage of fetchPaginated(
|
|
46
|
+
makeUrl(
|
|
47
|
+
ApiPath.WORKOUT_LOG,
|
|
48
|
+
{ query: { limit: API_MAX_PAGE_SIZE, ...filtersetQueryLogs } }
|
|
49
|
+
),
|
|
50
|
+
makeHeader()
|
|
51
|
+
)) {
|
|
52
|
+
for (const logData of logPage) {
|
|
53
|
+
logs.push(logAdapter.fromJson(logData));
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
// Fetch all exercises and associate them with logs if it's not known
|
|
58
|
+
for (const log of logs) {
|
|
59
|
+
if (!exercises[log.exerciseId]) {
|
|
60
|
+
exercises[log.exerciseId] = await getExercise(log.exerciseId);
|
|
61
|
+
}
|
|
62
|
+
log.exerciseObj = exercises[log.exerciseId];
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
for await (const sessionPage of fetchPaginated(
|
|
66
|
+
makeUrl(
|
|
67
|
+
ApiPath.SESSION,
|
|
68
|
+
{ query: { limit: API_MAX_PAGE_SIZE, ...filtersetQuerySessions } }
|
|
69
|
+
), makeHeader()
|
|
70
|
+
)) {
|
|
71
|
+
for (const sessionData of sessionPage) {
|
|
72
|
+
const session = sessionAdapter.fromJson(sessionData);
|
|
73
|
+
session.logs = logs.filter(log => log.sessionId === session.id);
|
|
74
|
+
result.push(session);
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
return result;
|
|
79
|
+
};
|
|
80
|
+
|
|
81
|
+
export const addSession = async (data: AddSessionParams): Promise<WorkoutSession> => {
|
|
82
|
+
const response = await axios.post(
|
|
83
|
+
makeUrl(ApiPath.SESSION,),
|
|
84
|
+
data,
|
|
85
|
+
{ headers: makeHeader() }
|
|
86
|
+
);
|
|
87
|
+
|
|
88
|
+
return new WorkoutSessionAdapter().fromJson(response.data);
|
|
89
|
+
};
|
|
90
|
+
|
|
91
|
+
export const editSession = async (data: EditSessionParams): Promise<WorkoutSession> => {
|
|
92
|
+
const response = await axios.patch(
|
|
93
|
+
makeUrl(ApiPath.SESSION, { id: data.id }),
|
|
94
|
+
data,
|
|
95
|
+
{ headers: makeHeader() }
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
return new WorkoutSessionAdapter().fromJson(response.data);
|
|
99
|
+
};
|
|
@@ -0,0 +1,68 @@
|
|
|
1
|
+
import axios from "axios";
|
|
2
|
+
import { Slot } from "components/WorkoutRoutines/models/Slot";
|
|
3
|
+
import { addSlot, deleteSlot, editSlot } from "services/slot";
|
|
4
|
+
import { ApiPath } from "utils/consts";
|
|
5
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
6
|
+
|
|
7
|
+
jest.mock('axios');
|
|
8
|
+
|
|
9
|
+
describe("Slot service tests", () => {
|
|
10
|
+
const slotApiData = {
|
|
11
|
+
day: 1,
|
|
12
|
+
order: 1,
|
|
13
|
+
comment: 'test',
|
|
14
|
+
config: null
|
|
15
|
+
};
|
|
16
|
+
|
|
17
|
+
beforeEach(() => {
|
|
18
|
+
jest.clearAllMocks();
|
|
19
|
+
});
|
|
20
|
+
|
|
21
|
+
test('Creates a new Slot', async () => {
|
|
22
|
+
(axios.post as jest.Mock).mockImplementation(() => Promise.resolve({ data: { id: 123, ...slotApiData } }));
|
|
23
|
+
const slot = Slot.fromJson(slotApiData);
|
|
24
|
+
|
|
25
|
+
const result = await addSlot(slot);
|
|
26
|
+
|
|
27
|
+
expect(axios.post).toHaveBeenCalledWith(
|
|
28
|
+
makeUrl(ApiPath.SLOT),
|
|
29
|
+
slot.toJson(),
|
|
30
|
+
{ headers: makeHeader() }
|
|
31
|
+
);
|
|
32
|
+
expect(result).toStrictEqual(new Slot({ id: 123, dayId: 1, order: 1, comment: 'test', config: null }));
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
test('Update a Slot', async () => {
|
|
36
|
+
(axios.patch as jest.Mock).mockImplementation(() => Promise.resolve({
|
|
37
|
+
data: {
|
|
38
|
+
id: 123,
|
|
39
|
+
...slotApiData,
|
|
40
|
+
comment: 'foo'
|
|
41
|
+
}
|
|
42
|
+
}));
|
|
43
|
+
|
|
44
|
+
const slot = Slot.fromJson(slotApiData);
|
|
45
|
+
slot.comment = 'foo';
|
|
46
|
+
slot.id = 123;
|
|
47
|
+
const result = await editSlot(slot);
|
|
48
|
+
|
|
49
|
+
expect(axios.patch).toHaveBeenCalledWith(
|
|
50
|
+
makeUrl(ApiPath.SLOT, { id: 123 }),
|
|
51
|
+
slot.toJson(),
|
|
52
|
+
{ headers: makeHeader() }
|
|
53
|
+
);
|
|
54
|
+
expect(result).toStrictEqual(new Slot({ id: 123, dayId: 1, order: 1, comment: 'foo', config: null }));
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
|
|
58
|
+
test('Delete a Slot', async () => {
|
|
59
|
+
(axios.delete as jest.Mock).mockImplementation(() => Promise.resolve({}));
|
|
60
|
+
|
|
61
|
+
await deleteSlot(1);
|
|
62
|
+
|
|
63
|
+
expect(axios.delete).toHaveBeenCalledWith(
|
|
64
|
+
makeUrl(ApiPath.SLOT, { id: 1 }),
|
|
65
|
+
{ headers: makeHeader() }
|
|
66
|
+
);
|
|
67
|
+
});
|
|
68
|
+
});
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import { Slot } from "components/WorkoutRoutines/models/Slot";
|
|
3
|
+
import { ApiPath } from "utils/consts";
|
|
4
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
/*
|
|
8
|
+
* Creates a new Slot
|
|
9
|
+
*/
|
|
10
|
+
export const addSlot = async (slot: Slot): Promise<Slot> => {
|
|
11
|
+
const response = await axios.post(
|
|
12
|
+
makeUrl(ApiPath.SLOT),
|
|
13
|
+
slot.toJson(),
|
|
14
|
+
{ headers: makeHeader() }
|
|
15
|
+
);
|
|
16
|
+
|
|
17
|
+
return Slot.fromJson(response.data);
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
/*
|
|
21
|
+
* Update a Slot
|
|
22
|
+
*/
|
|
23
|
+
export const editSlot = async (slot: Slot): Promise<Slot> => {
|
|
24
|
+
const response = await axios.patch(
|
|
25
|
+
makeUrl(ApiPath.SLOT, { id: slot.id! }),
|
|
26
|
+
slot.toJson(),
|
|
27
|
+
{ headers: makeHeader() }
|
|
28
|
+
);
|
|
29
|
+
|
|
30
|
+
return Slot.fromJson(response.data);
|
|
31
|
+
};
|
|
32
|
+
|
|
33
|
+
/*
|
|
34
|
+
* Delete an existing lot
|
|
35
|
+
*/
|
|
36
|
+
export const deleteSlot = async (id: number): Promise<void> => {
|
|
37
|
+
await axios.delete(
|
|
38
|
+
makeUrl(ApiPath.SLOT, { id: id }),
|
|
39
|
+
{ headers: makeHeader() }
|
|
40
|
+
);
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import { SlotEntry } from "components/WorkoutRoutines/models/SlotEntry";
|
|
3
|
+
import { ApiPath } from "utils/consts";
|
|
4
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
/*
|
|
8
|
+
* Update a Slot entry
|
|
9
|
+
*/
|
|
10
|
+
export const editSlotEntry = async (slotEntry: SlotEntry): Promise<SlotEntry> => {
|
|
11
|
+
const response = await axios.patch(
|
|
12
|
+
makeUrl(ApiPath.SLOT_ENTRY, { id: slotEntry.id! }),
|
|
13
|
+
slotEntry.toJson(),
|
|
14
|
+
{ headers: makeHeader() }
|
|
15
|
+
);
|
|
16
|
+
|
|
17
|
+
return SlotEntry.fromJson(response.data);
|
|
18
|
+
};
|
|
19
|
+
|
|
20
|
+
/*
|
|
21
|
+
* Delete an existing slot entry
|
|
22
|
+
*/
|
|
23
|
+
export const deleteSlotEntry = async (id: number): Promise<void> => {
|
|
24
|
+
await axios.delete(
|
|
25
|
+
makeUrl(ApiPath.SLOT_ENTRY, { id: id }),
|
|
26
|
+
{ headers: makeHeader() }
|
|
27
|
+
);
|
|
28
|
+
};
|
|
29
|
+
|
|
30
|
+
/*
|
|
31
|
+
* Creates a new slot entry
|
|
32
|
+
*/
|
|
33
|
+
export const addSlotEntry = async (slotEntry: SlotEntry): Promise<SlotEntry> => {
|
|
34
|
+
const response = await axios.post(
|
|
35
|
+
makeUrl(ApiPath.SLOT_ENTRY),
|
|
36
|
+
slotEntry.toJson(),
|
|
37
|
+
{ headers: makeHeader() }
|
|
38
|
+
);
|
|
39
|
+
|
|
40
|
+
return SlotEntry.fromJson(response.data);
|
|
41
|
+
};
|
|
42
|
+
|
|
43
|
+
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
import axios from "axios";
|
|
2
|
+
import { addVariation } from "services/variation";
|
|
3
|
+
|
|
4
|
+
jest.mock("axios");
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
describe("Variation service API tests", () => {
|
|
8
|
+
|
|
9
|
+
test('POST a new variation', async () => {
|
|
10
|
+
|
|
11
|
+
// Arrange
|
|
12
|
+
const response = {
|
|
13
|
+
"id": 123
|
|
14
|
+
};
|
|
15
|
+
(axios.post as jest.Mock).mockImplementation(() => Promise.resolve({ data: response }));
|
|
16
|
+
|
|
17
|
+
// Act
|
|
18
|
+
const result = await addVariation();
|
|
19
|
+
|
|
20
|
+
// Assert
|
|
21
|
+
expect(axios.post).toHaveBeenCalledTimes(1);
|
|
22
|
+
expect(result).toEqual(123);
|
|
23
|
+
});
|
|
24
|
+
});
|
|
@@ -0,0 +1,18 @@
|
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
3
|
+
|
|
4
|
+
export const VARIATION_PATH = 'variation';
|
|
5
|
+
|
|
6
|
+
|
|
7
|
+
/*
|
|
8
|
+
* Create a new exercise base
|
|
9
|
+
*/
|
|
10
|
+
export const addVariation = async (): Promise<number> => {
|
|
11
|
+
|
|
12
|
+
const url = makeUrl(VARIATION_PATH);
|
|
13
|
+
const response = await axios.post(url, {}, {
|
|
14
|
+
headers: makeHeader(),
|
|
15
|
+
});
|
|
16
|
+
|
|
17
|
+
return response.data.id;
|
|
18
|
+
};
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
import axios from "axios";
|
|
2
|
+
import { ExerciseVideo } from "components/Exercises/models/video";
|
|
3
|
+
import { deleteExerciseVideo, postExerciseVideo } from "services";
|
|
4
|
+
|
|
5
|
+
jest.mock("axios");
|
|
6
|
+
|
|
7
|
+
describe("Exercise video service API tests", () => {
|
|
8
|
+
test("POST a new video", async () => {
|
|
9
|
+
// Arrange
|
|
10
|
+
const response = {
|
|
11
|
+
id: 1,
|
|
12
|
+
uuid: "b1c934fa-c4f8-4d84-8cb4-7802be0d284c",
|
|
13
|
+
exercise: 258,
|
|
14
|
+
exercise_uuid: "6260e3aa-e46b-4b4b-8ada-58bfd0922d3a",
|
|
15
|
+
video: "http://localhost:8000/media/exercise-video/258/b1c934fa-c4f8-4d84-8cb4-7802be0d284c.mp4",
|
|
16
|
+
is_main: false,
|
|
17
|
+
size: 0,
|
|
18
|
+
duration: "0.00",
|
|
19
|
+
width: 0,
|
|
20
|
+
height: 0,
|
|
21
|
+
codec: "",
|
|
22
|
+
codec_long: "",
|
|
23
|
+
license: 2,
|
|
24
|
+
license_author: null,
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
const video = new ExerciseVideo(
|
|
28
|
+
1,
|
|
29
|
+
"b1c934fa-c4f8-4d84-8cb4-7802be0d284c",
|
|
30
|
+
"http://localhost:8000/media/exercise-video/258/b1c934fa-c4f8-4d84-8cb4-7802be0d284c.mp4",
|
|
31
|
+
false
|
|
32
|
+
);
|
|
33
|
+
|
|
34
|
+
(axios.post as jest.Mock).mockImplementation(() => Promise.resolve({ data: response }));
|
|
35
|
+
|
|
36
|
+
// Act
|
|
37
|
+
const result = await postExerciseVideo({
|
|
38
|
+
exerciseId: 42,
|
|
39
|
+
author: "Prostetnic Vogon Jeltz",
|
|
40
|
+
video: new File([], "test.mp4"),
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
// Assert
|
|
44
|
+
expect(axios.post).toHaveBeenCalled();
|
|
45
|
+
expect(axios.post).toHaveBeenCalledWith(
|
|
46
|
+
"https://example.com/api/v2/video/",
|
|
47
|
+
expect.objectContaining({ exercise: 42 }),
|
|
48
|
+
expect.anything()
|
|
49
|
+
);
|
|
50
|
+
expect(result).toEqual(video);
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
test("DELETE an existing video", async () => {
|
|
54
|
+
// Arrange
|
|
55
|
+
(axios.delete as jest.Mock).mockImplementation(() => Promise.resolve({ status: 204 }));
|
|
56
|
+
|
|
57
|
+
// Act
|
|
58
|
+
const result = await deleteExerciseVideo(99);
|
|
59
|
+
|
|
60
|
+
// Assert
|
|
61
|
+
expect(axios.delete).toHaveBeenCalled();
|
|
62
|
+
expect(axios.delete).toHaveBeenCalledWith("https://example.com/api/v2/video/99/", expect.anything());
|
|
63
|
+
expect(result).toEqual(204);
|
|
64
|
+
});
|
|
65
|
+
});
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import { ExerciseVideo, ExerciseVideoAdapter } from "components/Exercises/models/video";
|
|
3
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
4
|
+
|
|
5
|
+
export const VIDEO_PATH = 'video';
|
|
6
|
+
|
|
7
|
+
export type PostExerciseVideoParams = {
|
|
8
|
+
exerciseId: number;
|
|
9
|
+
author: string;
|
|
10
|
+
video: File;
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Post a new exercise video
|
|
15
|
+
* @param {number} exerciseId - ID of the exercise to which the video is linked
|
|
16
|
+
* @param {string} author - Name of the video's author (for license attribution)
|
|
17
|
+
* @param {File} video - Video file to upload
|
|
18
|
+
* @returns {Promise<ExerciseVideo>} - A promise that resolves to the uploaded ExerciseVideo object
|
|
19
|
+
*/
|
|
20
|
+
export const postExerciseVideo = async ({
|
|
21
|
+
exerciseId,
|
|
22
|
+
author,
|
|
23
|
+
video,
|
|
24
|
+
}: PostExerciseVideoParams): Promise<ExerciseVideo> => {
|
|
25
|
+
const url = makeUrl(VIDEO_PATH);
|
|
26
|
+
const headers = makeHeader();
|
|
27
|
+
headers['Content-Type'] = 'multipart/form-data';
|
|
28
|
+
|
|
29
|
+
const response = await axios.post(
|
|
30
|
+
url,
|
|
31
|
+
// eslint-disable-next-line camelcase
|
|
32
|
+
{ exercise: exerciseId, license_author: author, video: video },
|
|
33
|
+
{ headers: headers }
|
|
34
|
+
);
|
|
35
|
+
return new ExerciseVideoAdapter().fromJson(response.data);
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
/*
|
|
39
|
+
* Delete an exercise video
|
|
40
|
+
*/
|
|
41
|
+
export const deleteExerciseVideo = async (videoId: number): Promise<number> => {
|
|
42
|
+
const url = makeUrl(VIDEO_PATH, { id: videoId });
|
|
43
|
+
const headers = makeHeader();
|
|
44
|
+
const response = await axios.delete(url, { headers: headers });
|
|
45
|
+
|
|
46
|
+
return response.status;
|
|
47
|
+
};
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import axios from "axios";
|
|
2
|
+
import { WeightEntry } from "components/BodyWeight/model";
|
|
3
|
+
import { createWeight, deleteWeight, getWeights, updateWeight } from "./weight";
|
|
4
|
+
|
|
5
|
+
jest.mock("axios");
|
|
6
|
+
|
|
7
|
+
describe("weight service tests", () => {
|
|
8
|
+
|
|
9
|
+
test('GET weight entries', async () => {
|
|
10
|
+
|
|
11
|
+
const weightResponse = {
|
|
12
|
+
count: 2,
|
|
13
|
+
next: null,
|
|
14
|
+
previous: null,
|
|
15
|
+
results: [
|
|
16
|
+
{ id: 1, weight: 80, date: '2021-12-10' },
|
|
17
|
+
{ id: 2, weight: 90, date: '2021-12-20' },
|
|
18
|
+
]
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
(axios.get as jest.Mock).mockImplementation(() => Promise.resolve({ data: weightResponse }));
|
|
22
|
+
|
|
23
|
+
const result = await getWeights();
|
|
24
|
+
expect(axios.get).toHaveBeenCalledTimes(1);
|
|
25
|
+
|
|
26
|
+
expect(result).toStrictEqual([
|
|
27
|
+
new WeightEntry(new Date('2021-12-10'), 80, 1),
|
|
28
|
+
new WeightEntry(new Date('2021-12-20'), 90, 2),
|
|
29
|
+
]);
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
test('DELETE weight entry', async () => {
|
|
33
|
+
|
|
34
|
+
// Arrange
|
|
35
|
+
(axios.delete as jest.Mock).mockImplementation(() => Promise.resolve({ status: 204 }));
|
|
36
|
+
|
|
37
|
+
// Act
|
|
38
|
+
const result = await deleteWeight(1);
|
|
39
|
+
|
|
40
|
+
// Assert
|
|
41
|
+
expect(axios.delete).toHaveBeenCalledTimes(1);
|
|
42
|
+
expect(result).toEqual(204);
|
|
43
|
+
});
|
|
44
|
+
|
|
45
|
+
test('PATCH weight entry', async () => {
|
|
46
|
+
|
|
47
|
+
// Arrange
|
|
48
|
+
const weightEntry = new WeightEntry(new Date('2021-12-10'), 80, 1);
|
|
49
|
+
const weightResponse = { id: 1, weight: 80, date: '2021-12-10' };
|
|
50
|
+
|
|
51
|
+
// Act
|
|
52
|
+
(axios.patch as jest.Mock).mockImplementation(() => Promise.resolve(weightResponse));
|
|
53
|
+
const result = await updateWeight(weightEntry);
|
|
54
|
+
|
|
55
|
+
// Assert
|
|
56
|
+
expect(axios.patch).toHaveBeenCalledTimes(1);
|
|
57
|
+
expect(result).toStrictEqual(new WeightEntry(new Date('2021-12-10'), 80, 1));
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
test('POST a new weight entry', async () => {
|
|
61
|
+
|
|
62
|
+
// Arrange
|
|
63
|
+
const weightEntry = new WeightEntry(new Date('2021-12-10'), 80, 1);
|
|
64
|
+
const weightResponse = { data: { id: 1, weight: 80, date: '2021-12-10' } };
|
|
65
|
+
|
|
66
|
+
// Act
|
|
67
|
+
(axios.post as jest.Mock).mockImplementation(() => Promise.resolve(weightResponse));
|
|
68
|
+
const result = await createWeight(weightEntry);
|
|
69
|
+
|
|
70
|
+
// Assert
|
|
71
|
+
expect(axios.post).toHaveBeenCalledTimes(1);
|
|
72
|
+
expect(result).toStrictEqual(new WeightEntry(new Date('2021-12-10'), 80, 1));
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
});
|
|
@@ -0,0 +1,58 @@
|
|
|
1
|
+
import axios from 'axios';
|
|
2
|
+
import { WeightEntry } from "components/BodyWeight/model";
|
|
3
|
+
import { ApiBodyWeightType } from 'types';
|
|
4
|
+
import { makeHeader, makeUrl } from "utils/url";
|
|
5
|
+
import { FilterType } from '../components/BodyWeight/widgets/FilterButtons';
|
|
6
|
+
import { calculatePastDate } from '../utils/date';
|
|
7
|
+
import { ResponseType } from "./responseType";
|
|
8
|
+
|
|
9
|
+
export const WEIGHT_PATH = 'weightentry';
|
|
10
|
+
|
|
11
|
+
/*
|
|
12
|
+
* Fetch weight entries based on filter value
|
|
13
|
+
*/
|
|
14
|
+
export const getWeights = async (filter: FilterType = ''): Promise<WeightEntry[]> => {
|
|
15
|
+
|
|
16
|
+
// eslint-disable-next-line camelcase
|
|
17
|
+
const date__gte = calculatePastDate(filter);
|
|
18
|
+
|
|
19
|
+
// eslint-disable-next-line camelcase
|
|
20
|
+
const url = makeUrl(WEIGHT_PATH, { query: { ordering: '-date', limit: 900, ...(date__gte && { date__gte }) } });
|
|
21
|
+
const { data: receivedWeights } = await axios.get<ResponseType<ApiBodyWeightType>>(url, {
|
|
22
|
+
headers: makeHeader(),
|
|
23
|
+
});
|
|
24
|
+
return receivedWeights.results.map(weight => WeightEntry.fromJson(weight));
|
|
25
|
+
};
|
|
26
|
+
|
|
27
|
+
/*
|
|
28
|
+
* Delete a weight entry
|
|
29
|
+
*/
|
|
30
|
+
export const deleteWeight = async (id: number): Promise<number> => {
|
|
31
|
+
const response = await axios.delete<number>(makeUrl(WEIGHT_PATH, { id: id }), {
|
|
32
|
+
headers: makeHeader(),
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
return response.status;
|
|
36
|
+
};
|
|
37
|
+
|
|
38
|
+
/*
|
|
39
|
+
* Update a weight entry
|
|
40
|
+
*/
|
|
41
|
+
export const updateWeight = async (entry: WeightEntry): Promise<WeightEntry> => {
|
|
42
|
+
const response = await axios.patch(makeUrl(WEIGHT_PATH, { id: entry.id }), entry.toJson(), {
|
|
43
|
+
headers: makeHeader(),
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
return WeightEntry.fromJson(response);
|
|
47
|
+
};
|
|
48
|
+
|
|
49
|
+
/*
|
|
50
|
+
* Add a new weight entry
|
|
51
|
+
*/
|
|
52
|
+
export const createWeight = async (entry: WeightEntry): Promise<WeightEntry> => {
|
|
53
|
+
const response = await axios.post(makeUrl(WEIGHT_PATH,), entry.toJson(), {
|
|
54
|
+
headers: makeHeader(),
|
|
55
|
+
});
|
|
56
|
+
|
|
57
|
+
return WeightEntry.fromJson(response.data);
|
|
58
|
+
};
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
import axios from "axios";
|
|
2
|
+
import { WorkoutLog } from "components/WorkoutRoutines/models/WorkoutLog";
|
|
3
|
+
import { getExercise } from "services";
|
|
4
|
+
import { getRoutineLogs } from "services/workoutLogs";
|
|
5
|
+
import { getRoutineRepUnits, getRoutineWeightUnits } from "services/workoutUnits";
|
|
6
|
+
import { testExerciseSquats } from "tests/exerciseTestdata";
|
|
7
|
+
import {
|
|
8
|
+
responseRoutineLogs,
|
|
9
|
+
testRepUnit1,
|
|
10
|
+
testRepUnit2,
|
|
11
|
+
testWeightUnit1,
|
|
12
|
+
testWeightUnit2
|
|
13
|
+
} from "tests/workoutRoutinesTestData";
|
|
14
|
+
|
|
15
|
+
jest.mock("axios");
|
|
16
|
+
jest.mock("services/workoutUnits");
|
|
17
|
+
jest.mock("services/exercise");
|
|
18
|
+
|
|
19
|
+
|
|
20
|
+
describe("workout logs service tests", () => {
|
|
21
|
+
|
|
22
|
+
beforeEach(() => {
|
|
23
|
+
jest.resetAllMocks();
|
|
24
|
+
});
|
|
25
|
+
|
|
26
|
+
test('GET the routine logs', async () => {
|
|
27
|
+
|
|
28
|
+
(axios.get as jest.Mock).mockImplementation(() => Promise.resolve({ data: responseRoutineLogs }));
|
|
29
|
+
(getRoutineRepUnits as jest.Mock).mockImplementation(() => Promise.resolve([testRepUnit1, testRepUnit2]));
|
|
30
|
+
(getRoutineWeightUnits as jest.Mock).mockImplementation(() => Promise.resolve([testWeightUnit1, testWeightUnit2]));
|
|
31
|
+
|
|
32
|
+
// Act
|
|
33
|
+
const result = await getRoutineLogs(1);
|
|
34
|
+
|
|
35
|
+
// Assert
|
|
36
|
+
expect(axios.get).toHaveBeenCalledTimes(1);
|
|
37
|
+
expect(result).toStrictEqual([
|
|
38
|
+
new WorkoutLog({
|
|
39
|
+
id: 2,
|
|
40
|
+
routineId: 1,
|
|
41
|
+
date: new Date("2023-05-10"),
|
|
42
|
+
iteration: 1,
|
|
43
|
+
exerciseId: 100,
|
|
44
|
+
slotEntryId: 2,
|
|
45
|
+
|
|
46
|
+
repetitionsUnit: testRepUnit1,
|
|
47
|
+
repetitionsUnitId: 1,
|
|
48
|
+
repetitions: 12,
|
|
49
|
+
repetitionsTarget: 12,
|
|
50
|
+
|
|
51
|
+
weightUnit: testWeightUnit1,
|
|
52
|
+
weightUnitId: 1,
|
|
53
|
+
weight: 10.00,
|
|
54
|
+
weightTarget: null,
|
|
55
|
+
|
|
56
|
+
rir: null,
|
|
57
|
+
rirTarget: null,
|
|
58
|
+
}),
|
|
59
|
+
|
|
60
|
+
new WorkoutLog({
|
|
61
|
+
id: 1,
|
|
62
|
+
routineId: 1,
|
|
63
|
+
date: new Date("2023-05-13"),
|
|
64
|
+
iteration: 1,
|
|
65
|
+
exerciseId: 100,
|
|
66
|
+
slotEntryId: 2,
|
|
67
|
+
|
|
68
|
+
repetitionsUnit: testRepUnit1,
|
|
69
|
+
repetitionsUnitId: 1,
|
|
70
|
+
repetitions: 10,
|
|
71
|
+
repetitionsTarget: null,
|
|
72
|
+
|
|
73
|
+
weightUnit: testWeightUnit1,
|
|
74
|
+
weightUnitId: 1,
|
|
75
|
+
weight: 20,
|
|
76
|
+
weightTarget: 20,
|
|
77
|
+
|
|
78
|
+
rir: 1.5,
|
|
79
|
+
rirTarget: 1,
|
|
80
|
+
}),
|
|
81
|
+
]);
|
|
82
|
+
});
|
|
83
|
+
|
|
84
|
+
|
|
85
|
+
test('GET the routine logs and the exercise bases', async () => {
|
|
86
|
+
|
|
87
|
+
// Arrange
|
|
88
|
+
(axios.get as jest.Mock).mockImplementation(() => Promise.resolve({ data: responseRoutineLogs }));
|
|
89
|
+
(getRoutineRepUnits as jest.Mock).mockImplementation(() => Promise.resolve([testRepUnit1, testRepUnit2]));
|
|
90
|
+
(getRoutineWeightUnits as jest.Mock).mockImplementation(() => Promise.resolve([testWeightUnit1, testWeightUnit2]));
|
|
91
|
+
(getExercise as jest.Mock).mockImplementation(() => Promise.resolve(testExerciseSquats));
|
|
92
|
+
|
|
93
|
+
// Act
|
|
94
|
+
const result = await getRoutineLogs(1, { loadExercises: true });
|
|
95
|
+
|
|
96
|
+
// Assert
|
|
97
|
+
expect(axios.get).toHaveBeenCalledTimes(1);
|
|
98
|
+
expect(result).toStrictEqual([
|
|
99
|
+
new WorkoutLog({
|
|
100
|
+
id: 2,
|
|
101
|
+
routineId: 1,
|
|
102
|
+
date: new Date("2023-05-10"),
|
|
103
|
+
iteration: 1,
|
|
104
|
+
exercise: testExerciseSquats,
|
|
105
|
+
exerciseId: 100,
|
|
106
|
+
slotEntryId: 2,
|
|
107
|
+
sessionId: null,
|
|
108
|
+
|
|
109
|
+
repetitionsUnit: testRepUnit1,
|
|
110
|
+
repetitionsUnitId: 1,
|
|
111
|
+
repetitions: 12,
|
|
112
|
+
repetitionsTarget: 12,
|
|
113
|
+
|
|
114
|
+
weightUnit: testWeightUnit1,
|
|
115
|
+
weightUnitId: 1,
|
|
116
|
+
weight: 10.00,
|
|
117
|
+
weightTarget: null,
|
|
118
|
+
|
|
119
|
+
rir: null,
|
|
120
|
+
rirTarget: null,
|
|
121
|
+
}),
|
|
122
|
+
|
|
123
|
+
new WorkoutLog({
|
|
124
|
+
id: 1,
|
|
125
|
+
routineId: 1,
|
|
126
|
+
date: new Date("2023-05-13"),
|
|
127
|
+
iteration: 1,
|
|
128
|
+
slotEntryId: 2,
|
|
129
|
+
exercise: testExerciseSquats,
|
|
130
|
+
exerciseId: 100,
|
|
131
|
+
sessionId: null,
|
|
132
|
+
|
|
133
|
+
repetitionsUnit: testRepUnit1,
|
|
134
|
+
repetitionsUnitId: 1,
|
|
135
|
+
repetitions: 10,
|
|
136
|
+
repetitionsTarget: null,
|
|
137
|
+
|
|
138
|
+
weightUnit: testWeightUnit1,
|
|
139
|
+
weightUnitId: 1,
|
|
140
|
+
weight: 20,
|
|
141
|
+
weightTarget: 20,
|
|
142
|
+
|
|
143
|
+
rir: 1.5,
|
|
144
|
+
rirTarget: 1,
|
|
145
|
+
}),
|
|
146
|
+
]);
|
|
147
|
+
});
|
|
148
|
+
});
|