@openedx/frontend-app-learner-dashboard 1.0.0-alpha.0
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.rst +86 -0
- package/package.json +84 -0
- package/src/Main.jsx +20 -0
- package/src/__mocks__/file.js +1 -0
- package/src/__mocks__/svg.js +1 -0
- package/src/__snapshots__/App.test.jsx.snap +83 -0
- package/src/__snapshots__/index.test.jsx.snap +43 -0
- package/src/app.scss +41 -0
- package/src/app.ts +23 -0
- package/src/assets/empty-course.svg +49 -0
- package/src/assets/more-courses-sidewidget.svg +52 -0
- package/src/assets/verified-ribbon.png +0 -0
- package/src/components/Banner.jsx +26 -0
- package/src/components/Banner.test.jsx +27 -0
- package/src/components/__snapshots__/Banner.test.jsx.snap +31 -0
- package/src/constants.ts +1 -0
- package/src/containers/CourseCard/CourseCard.scss +75 -0
- package/src/containers/CourseCard/__snapshots__/index.test.jsx.snap +111 -0
- package/src/containers/CourseCard/components/CourseCardActions/ActionButton/__snapshots__/index.test.jsx.snap +14 -0
- package/src/containers/CourseCard/components/CourseCardActions/ActionButton/hooks.js +8 -0
- package/src/containers/CourseCard/components/CourseCardActions/ActionButton/hooks.test.js +21 -0
- package/src/containers/CourseCard/components/CourseCardActions/ActionButton/index.jsx +16 -0
- package/src/containers/CourseCard/components/CourseCardActions/ActionButton/index.test.jsx +25 -0
- package/src/containers/CourseCard/components/CourseCardActions/BeginCourseButton.jsx +38 -0
- package/src/containers/CourseCard/components/CourseCardActions/BeginCourseButton.test.jsx +86 -0
- package/src/containers/CourseCard/components/CourseCardActions/ResumeButton.jsx +38 -0
- package/src/containers/CourseCard/components/CourseCardActions/ResumeButton.test.jsx +84 -0
- package/src/containers/CourseCard/components/CourseCardActions/SelectSessionButton.jsx +28 -0
- package/src/containers/CourseCard/components/CourseCardActions/SelectSessionButton.test.jsx +34 -0
- package/src/containers/CourseCard/components/CourseCardActions/ViewCourseButton.jsx +37 -0
- package/src/containers/CourseCard/components/CourseCardActions/ViewCourseButton.test.jsx +45 -0
- package/src/containers/CourseCard/components/CourseCardActions/__snapshots__/BeginCourseButton.test.jsx.snap +39 -0
- package/src/containers/CourseCard/components/CourseCardActions/__snapshots__/ResumeButton.test.jsx.snap +39 -0
- package/src/containers/CourseCard/components/CourseCardActions/__snapshots__/SelectSessionButton.test.jsx.snap +19 -0
- package/src/containers/CourseCard/components/CourseCardActions/__snapshots__/ViewCourseButton.test.jsx.snap +39 -0
- package/src/containers/CourseCard/components/CourseCardActions/index.jsx +42 -0
- package/src/containers/CourseCard/components/CourseCardActions/index.test.jsx +97 -0
- package/src/containers/CourseCard/components/CourseCardActions/messages.js +26 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CertificateBanner.jsx +95 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CertificateBanner.test.jsx +227 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CourseBanner.jsx +57 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CourseBanner.test.jsx +131 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/__snapshots__/index.test.jsx.snap +58 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/hooks.js +42 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/hooks.test.js +90 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/index.jsx +36 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/index.test.jsx +95 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/messages.js +16 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/ApprovedContent.jsx +35 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/ApprovedContent.test.jsx +64 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/EligibleContent.jsx +34 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/EligibleContent.test.jsx +82 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/MustRequestContent.jsx +36 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/MustRequestContent.test.jsx +74 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/PendingContent.jsx +30 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/PendingContent.test.jsx +63 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/RejectedContent.jsx +27 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/RejectedContent.test.jsx +54 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditContent.jsx +51 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditContent.test.jsx +60 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditRequestForm/__snapshots__/index.test.jsx.snap +32 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditRequestForm/hooks.js +13 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditRequestForm/hooks.test.js +45 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditRequestForm/index.jsx +43 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditRequestForm/index.test.jsx +65 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/CreditRequestForm/ref.test.jsx +34 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/ProviderLink.jsx +24 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/ProviderLink.test.jsx +42 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/__snapshots__/CreditContent.test.jsx.snap +60 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/components/__snapshots__/ProviderLink.test.jsx.snap +11 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/hooks.js +27 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/hooks.test.js +56 -0
- package/src/containers/CourseCard/components/CourseCardBanners/CreditBanner/views/messages.js +61 -0
- package/src/containers/CourseCard/components/CourseCardBanners/EntitlementBanner.jsx +66 -0
- package/src/containers/CourseCard/components/CourseCardBanners/EntitlementBanner.test.jsx +63 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/ProgramsList.jsx +23 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/ProgramsList.test.jsx +23 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/__snapshots__/ProgramsList.test.jsx.snap +28 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/__snapshots__/index.test.jsx.snap +29 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/index.jsx +38 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/index.test.jsx +42 -0
- package/src/containers/CourseCard/components/CourseCardBanners/RelatedProgramsBanner/messages.js +31 -0
- package/src/containers/CourseCard/components/CourseCardBanners/__snapshots__/CertificateBanner.test.jsx.snap +205 -0
- package/src/containers/CourseCard/components/CourseCardBanners/__snapshots__/CourseBanner.test.jsx.snap +38 -0
- package/src/containers/CourseCard/components/CourseCardBanners/__snapshots__/EntitlementBanner.test.jsx.snap +53 -0
- package/src/containers/CourseCard/components/CourseCardBanners/__snapshots__/index.test.jsx.snap +41 -0
- package/src/containers/CourseCard/components/CourseCardBanners/index.jsx +28 -0
- package/src/containers/CourseCard/components/CourseCardBanners/index.test.jsx +32 -0
- package/src/containers/CourseCard/components/CourseCardBanners/messages.js +106 -0
- package/src/containers/CourseCard/components/CourseCardDetails/__snapshots__/index.test.jsx.snap +56 -0
- package/src/containers/CourseCard/components/CourseCardDetails/hooks.js +67 -0
- package/src/containers/CourseCard/components/CourseCardDetails/hooks.test.js +186 -0
- package/src/containers/CourseCard/components/CourseCardDetails/index.jsx +45 -0
- package/src/containers/CourseCard/components/CourseCardDetails/index.scss +7 -0
- package/src/containers/CourseCard/components/CourseCardDetails/index.test.jsx +71 -0
- package/src/containers/CourseCard/components/CourseCardDetails/messages.js +41 -0
- package/src/containers/CourseCard/components/CourseCardImage.jsx +68 -0
- package/src/containers/CourseCard/components/CourseCardImage.test.jsx +66 -0
- package/src/containers/CourseCard/components/CourseCardMenu/SocialShareMenu.jsx +83 -0
- package/src/containers/CourseCard/components/CourseCardMenu/SocialShareMenu.test.jsx +236 -0
- package/src/containers/CourseCard/components/CourseCardMenu/__snapshots__/index.test.jsx.snap +81 -0
- package/src/containers/CourseCard/components/CourseCardMenu/hooks.js +58 -0
- package/src/containers/CourseCard/components/CourseCardMenu/hooks.test.js +153 -0
- package/src/containers/CourseCard/components/CourseCardMenu/index.jsx +85 -0
- package/src/containers/CourseCard/components/CourseCardMenu/index.test.jsx +214 -0
- package/src/containers/CourseCard/components/CourseCardMenu/messages.js +36 -0
- package/src/containers/CourseCard/components/CourseCardTitle.jsx +43 -0
- package/src/containers/CourseCard/components/CourseCardTitle.test.jsx +67 -0
- package/src/containers/CourseCard/components/RelatedProgramsBadge/__snapshots__/index.test.jsx.snap +25 -0
- package/src/containers/CourseCard/components/RelatedProgramsBadge/hooks.jsx +35 -0
- package/src/containers/CourseCard/components/RelatedProgramsBadge/hooks.test.js +76 -0
- package/src/containers/CourseCard/components/RelatedProgramsBadge/index.jsx +39 -0
- package/src/containers/CourseCard/components/RelatedProgramsBadge/index.test.jsx +30 -0
- package/src/containers/CourseCard/components/RelatedProgramsBadge/messages.js +13 -0
- package/src/containers/CourseCard/components/__snapshots__/CourseCardImage.test.jsx.snap +72 -0
- package/src/containers/CourseCard/components/__snapshots__/CourseCardTitle.test.jsx.snap +33 -0
- package/src/containers/CourseCard/components/hooks.js +33 -0
- package/src/containers/CourseCard/components/hooks.test.js +165 -0
- package/src/containers/CourseCard/hooks.js +23 -0
- package/src/containers/CourseCard/hooks.test.js +48 -0
- package/src/containers/CourseCard/index.jsx +50 -0
- package/src/containers/CourseCard/index.test.jsx +29 -0
- package/src/containers/CourseCard/messages.js +26 -0
- package/src/containers/CourseFilterControls/ActiveCourseFilters.jsx +40 -0
- package/src/containers/CourseFilterControls/ActiveCourseFilters.test.jsx +17 -0
- package/src/containers/CourseFilterControls/CourseFilterControls.jsx +115 -0
- package/src/containers/CourseFilterControls/CourseFilterControls.test.jsx +60 -0
- package/src/containers/CourseFilterControls/__snapshots__/ActiveCourseFilters.test.jsx.snap +39 -0
- package/src/containers/CourseFilterControls/__snapshots__/CourseFilterControls.test.jsx.snap +169 -0
- package/src/containers/CourseFilterControls/components/Checkbox.jsx +21 -0
- package/src/containers/CourseFilterControls/components/Checkbox.test.jsx +15 -0
- package/src/containers/CourseFilterControls/components/FilterForm.jsx +45 -0
- package/src/containers/CourseFilterControls/components/FilterForm.test.jsx +29 -0
- package/src/containers/CourseFilterControls/components/SortForm.jsx +39 -0
- package/src/containers/CourseFilterControls/components/SortForm.test.jsx +19 -0
- package/src/containers/CourseFilterControls/components/__snapshots__/Checkbox.test.jsx.snap +46 -0
- package/src/containers/CourseFilterControls/components/__snapshots__/FilterForm.test.jsx.snap +41 -0
- package/src/containers/CourseFilterControls/components/__snapshots__/SortForm.test.jsx.snap +29 -0
- package/src/containers/CourseFilterControls/hooks.js +60 -0
- package/src/containers/CourseFilterControls/hooks.test.js +113 -0
- package/src/containers/CourseFilterControls/index.jsx +6 -0
- package/src/containers/CourseFilterControls/index.scss +29 -0
- package/src/containers/CourseFilterControls/messages.js +61 -0
- package/src/containers/CoursesPanel/CourseList/__snapshots__/index.test.jsx.snap +70 -0
- package/src/containers/CoursesPanel/CourseList/hooks.js +8 -0
- package/src/containers/CoursesPanel/CourseList/index.jsx +54 -0
- package/src/containers/CoursesPanel/CourseList/index.test.jsx +64 -0
- package/src/containers/CoursesPanel/NoCoursesView/__snapshots__/index.test.jsx.snap +29 -0
- package/src/containers/CoursesPanel/NoCoursesView/index.jsx +40 -0
- package/src/containers/CoursesPanel/NoCoursesView/index.scss +15 -0
- package/src/containers/CoursesPanel/NoCoursesView/index.test.jsx +18 -0
- package/src/containers/CoursesPanel/NoCoursesView/messages.js +26 -0
- package/src/containers/CoursesPanel/__snapshots__/index.test.jsx.snap +55 -0
- package/src/containers/CoursesPanel/hooks.js +54 -0
- package/src/containers/CoursesPanel/hooks.test.js +115 -0
- package/src/containers/CoursesPanel/index.jsx +40 -0
- package/src/containers/CoursesPanel/index.scss +42 -0
- package/src/containers/CoursesPanel/index.test.jsx +58 -0
- package/src/containers/CoursesPanel/messages.js +11 -0
- package/src/containers/Dashboard/DashboardLayout.jsx +54 -0
- package/src/containers/Dashboard/DashboardLayout.test.jsx +115 -0
- package/src/containers/Dashboard/LoadingView.jsx +20 -0
- package/src/containers/Dashboard/LoadingView.test.jsx +23 -0
- package/src/containers/Dashboard/__snapshots__/DashboardLayout.test.jsx.snap +197 -0
- package/src/containers/Dashboard/__snapshots__/LoadingView.test.jsx.snap +13 -0
- package/src/containers/Dashboard/__snapshots__/index.test.jsx.snap +69 -0
- package/src/containers/Dashboard/hooks.js +34 -0
- package/src/containers/Dashboard/hooks.test.js +88 -0
- package/src/containers/Dashboard/index.jsx +43 -0
- package/src/containers/Dashboard/index.scss +29 -0
- package/src/containers/Dashboard/index.test.jsx +122 -0
- package/src/containers/EmailSettingsModal/__snapshots__/index.test.jsx.snap +133 -0
- package/src/containers/EmailSettingsModal/hooks.js +32 -0
- package/src/containers/EmailSettingsModal/hooks.test.js +71 -0
- package/src/containers/EmailSettingsModal/index.jsx +58 -0
- package/src/containers/EmailSettingsModal/index.test.jsx +57 -0
- package/src/containers/EmailSettingsModal/messages.js +37 -0
- package/src/containers/RelatedProgramsModal/__snapshots__/index.test.jsx.snap +169 -0
- package/src/containers/RelatedProgramsModal/components/ProgramCard.jsx +66 -0
- package/src/containers/RelatedProgramsModal/components/ProgramCard.test.jsx +23 -0
- package/src/containers/RelatedProgramsModal/components/__snapshots__/ProgramCard.test.jsx.snap +60 -0
- package/src/containers/RelatedProgramsModal/components/index.scss +23 -0
- package/src/containers/RelatedProgramsModal/components/messages.js +24 -0
- package/src/containers/RelatedProgramsModal/hooks.js +10 -0
- package/src/containers/RelatedProgramsModal/index.jsx +62 -0
- package/src/containers/RelatedProgramsModal/index.scss +5 -0
- package/src/containers/RelatedProgramsModal/index.test.jsx +60 -0
- package/src/containers/RelatedProgramsModal/messages.js +16 -0
- package/src/containers/SelectSessionModal/__snapshots__/index.test.jsx.snap +176 -0
- package/src/containers/SelectSessionModal/constants.js +2 -0
- package/src/containers/SelectSessionModal/hooks.js +77 -0
- package/src/containers/SelectSessionModal/hooks.test.js +194 -0
- package/src/containers/SelectSessionModal/index.jsx +75 -0
- package/src/containers/SelectSessionModal/index.test.jsx +53 -0
- package/src/containers/SelectSessionModal/messages.js +41 -0
- package/src/containers/UnenrollConfirmModal/__snapshots__/index.test.jsx.snap +101 -0
- package/src/containers/UnenrollConfirmModal/components/ConfirmPane.jsx +36 -0
- package/src/containers/UnenrollConfirmModal/components/ConfirmPane.test.jsx +14 -0
- package/src/containers/UnenrollConfirmModal/components/FinishedPane.jsx +35 -0
- package/src/containers/UnenrollConfirmModal/components/FinishedPane.test.jsx +21 -0
- package/src/containers/UnenrollConfirmModal/components/ReasonPane.jsx +65 -0
- package/src/containers/UnenrollConfirmModal/components/ReasonPane.test.jsx +26 -0
- package/src/containers/UnenrollConfirmModal/components/__snapshots__/ConfirmPane.test.jsx.snap +22 -0
- package/src/containers/UnenrollConfirmModal/components/__snapshots__/FinishedPane.test.jsx.snap +38 -0
- package/src/containers/UnenrollConfirmModal/components/__snapshots__/ReasonPane.test.jsx.snap +183 -0
- package/src/containers/UnenrollConfirmModal/components/messages.js +56 -0
- package/src/containers/UnenrollConfirmModal/constants.js +86 -0
- package/src/containers/UnenrollConfirmModal/hooks/index.js +55 -0
- package/src/containers/UnenrollConfirmModal/hooks/index.test.js +101 -0
- package/src/containers/UnenrollConfirmModal/hooks/reasons.js +79 -0
- package/src/containers/UnenrollConfirmModal/hooks/reasons.test.js +192 -0
- package/src/containers/UnenrollConfirmModal/index.jsx +59 -0
- package/src/containers/UnenrollConfirmModal/index.test.jsx +61 -0
- package/src/data/constants/app.js +20 -0
- package/src/data/constants/app.test.js +24 -0
- package/src/data/constants/course.js +17 -0
- package/src/data/constants/credit.js +9 -0
- package/src/data/constants/files.js +20 -0
- package/src/data/constants/htmlKeys.js +21 -0
- package/src/data/constants/requests.js +34 -0
- package/src/data/contexts/GlobalDataContext.jsx +15 -0
- package/src/data/contexts/GlobalDataProvider.jsx +23 -0
- package/src/data/contexts/MasqueradeUserContext.jsx +16 -0
- package/src/data/contexts/MasqueradeUserProvider.jsx +31 -0
- package/src/data/redux/app/index.js +2 -0
- package/src/data/redux/app/reducer.js +81 -0
- package/src/data/redux/app/reducer.test.js +124 -0
- package/src/data/redux/app/selectors/appSelectors.js +23 -0
- package/src/data/redux/app/selectors/appSelectors.test.js +28 -0
- package/src/data/redux/app/selectors/courseCard.js +155 -0
- package/src/data/redux/app/selectors/courseCard.test.js +398 -0
- package/src/data/redux/app/selectors/currentList.js +60 -0
- package/src/data/redux/app/selectors/currentList.test.js +185 -0
- package/src/data/redux/app/selectors/index.js +13 -0
- package/src/data/redux/app/selectors/simpleSelectors.js +38 -0
- package/src/data/redux/app/selectors/simpleSelectors.test.js +75 -0
- package/src/data/redux/hooks/app.js +106 -0
- package/src/data/redux/hooks/index.js +2 -0
- package/src/data/redux/hooks/requests.js +47 -0
- package/src/data/redux/index.js +37 -0
- package/src/data/redux/requests/index.js +2 -0
- package/src/data/redux/requests/reducer.js +53 -0
- package/src/data/redux/requests/reducer.test.js +62 -0
- package/src/data/redux/requests/selectors.js +29 -0
- package/src/data/redux/requests/selectors.test.js +110 -0
- package/src/data/services/lms/api.js +77 -0
- package/src/data/services/lms/api.test.js +156 -0
- package/src/data/services/lms/constants.js +18 -0
- package/src/data/services/lms/fakeData/courses.js +828 -0
- package/src/data/services/lms/fakeData/testUtils.js +40 -0
- package/src/data/services/lms/index.js +8 -0
- package/src/data/services/lms/urls.js +43 -0
- package/src/data/services/lms/urls.test.js +51 -0
- package/src/data/services/lms/utils.js +60 -0
- package/src/data/services/lms/utils.test.js +40 -0
- package/src/data/services/segment/utils.js +17 -0
- package/src/data/services/segment/utils.test.js +36 -0
- package/src/data/store.js +25 -0
- package/src/data/store.test.js +47 -0
- package/src/data/utils.js +19 -0
- package/src/data/utils.test.js +29 -0
- package/src/hooks/api.js +123 -0
- package/src/hooks/api.test.js +273 -0
- package/src/hooks/index.js +7 -0
- package/src/hooks/utils.js +17 -0
- package/src/hooks/utils.test.js +16 -0
- package/src/i18n/index.js +25 -0
- package/src/index.ts +3 -0
- package/src/messages.js +16 -0
- package/src/providers.ts +11 -0
- package/src/routes.jsx +14 -0
- package/src/segment.js +85 -0
- package/src/setupTest.jsx +251 -0
- package/src/slots/CourseBannerSlot/README.md +44 -0
- package/src/slots/CourseBannerSlot/images/course_banner_slot_default.png +0 -0
- package/src/slots/CourseBannerSlot/images/custom_course_banner.png +0 -0
- package/src/slots/CourseBannerSlot/index.jsx +18 -0
- package/src/slots/CourseCardActionSlot/README.md +53 -0
- package/src/slots/CourseCardActionSlot/images/post_course_card_action.png +0 -0
- package/src/slots/CourseCardActionSlot/index.jsx +15 -0
- package/src/slots/CourseListSlot/README.md +54 -0
- package/src/slots/CourseListSlot/images/course_list_slot.png +0 -0
- package/src/slots/CourseListSlot/images/readme_custom_course_list.png +0 -0
- package/src/slots/CourseListSlot/index.jsx +17 -0
- package/src/slots/DashboardModalSlot/README.md +30 -0
- package/src/slots/DashboardModalSlot/images/dashboard_modal_slot.png +0 -0
- package/src/slots/DashboardModalSlot/index.jsx +7 -0
- package/src/slots/NoCoursesViewSlot/README.md +34 -0
- package/src/slots/NoCoursesViewSlot/images/no_course_view_slot.png +0 -0
- package/src/slots/NoCoursesViewSlot/images/readme_custom_no_courses.png +0 -0
- package/src/slots/NoCoursesViewSlot/index.jsx +11 -0
- package/src/slots/README.md +8 -0
- package/src/slots/WidgetSidebarSlot/README.md +51 -0
- package/src/slots/WidgetSidebarSlot/__snapshots__/index.test.jsx.snap +14 -0
- package/src/slots/WidgetSidebarSlot/images/readme_custom_sidebar.png +0 -0
- package/src/slots/WidgetSidebarSlot/images/widget_sidebar_slot.png +0 -0
- package/src/slots/WidgetSidebarSlot/index.jsx +10 -0
- package/src/slots/WidgetSidebarSlot/index.test.jsx +18 -0
- package/src/slots.tsx +9 -0
- package/src/test/app.test.jsx +255 -0
- package/src/test/inspector.js +50 -0
- package/src/test/messages.js +29 -0
- package/src/test/utils.js +3 -0
- package/src/testUtils.js +211 -0
- package/src/tracking/constants.js +50 -0
- package/src/tracking/index.js +17 -0
- package/src/tracking/trackers/course.js +44 -0
- package/src/tracking/trackers/course.test.js +80 -0
- package/src/tracking/trackers/credit.js +20 -0
- package/src/tracking/trackers/credit.test.js +38 -0
- package/src/tracking/trackers/engagement.js +23 -0
- package/src/tracking/trackers/engagement.test.js +31 -0
- package/src/tracking/trackers/entitlements.js +34 -0
- package/src/tracking/trackers/entitlements.test.js +34 -0
- package/src/tracking/trackers/filter.js +21 -0
- package/src/tracking/trackers/filter.test.js +45 -0
- package/src/tracking/trackers/findCourses.js +16 -0
- package/src/tracking/trackers/findCourses.test.js +45 -0
- package/src/tracking/trackers/socialShare.js +11 -0
- package/src/tracking/trackers/socialShare.test.js +17 -0
- package/src/utils/StrictDict.js +19 -0
- package/src/utils/StrictDict.test.js +66 -0
- package/src/utils/dateFormatter.js +9 -0
- package/src/utils/hooks.js +12 -0
- package/src/utils/index.js +5 -0
- package/src/utils/keyStore.js +10 -0
- package/src/widgets/LearnerDashboardHeader/ConfirmEmailBanner/ConfirmEmailBanner.scss +3 -0
- package/src/widgets/LearnerDashboardHeader/ConfirmEmailBanner/assets/confirm-email.svg +76 -0
- package/src/widgets/LearnerDashboardHeader/ConfirmEmailBanner/hooks.js +45 -0
- package/src/widgets/LearnerDashboardHeader/ConfirmEmailBanner/index.jsx +66 -0
- package/src/widgets/LearnerDashboardHeader/ConfirmEmailBanner/messages.js +36 -0
- package/src/widgets/LearnerDashboardHeader/CoursesLink.jsx +14 -0
- package/src/widgets/LearnerDashboardHeader/DiscoverLinkMenuItem.jsx +30 -0
- package/src/widgets/LearnerDashboardHeader/MasqueradeBar/hooks.js +69 -0
- package/src/widgets/LearnerDashboardHeader/MasqueradeBar/index.jsx +92 -0
- package/src/widgets/LearnerDashboardHeader/MasqueradeBar/index.scss +38 -0
- package/src/widgets/LearnerDashboardHeader/MasqueradeBar/messages.js +36 -0
- package/src/widgets/LearnerDashboardHeader/OrderHistoryLinkMenuItem.jsx +29 -0
- package/src/widgets/LearnerDashboardHeader/ProgramsLinkMenuItem.jsx +26 -0
- package/src/widgets/LearnerDashboardHeader/SupportLinkMenuItem.jsx +29 -0
- package/src/widgets/LearnerDashboardHeader/app.tsx +93 -0
- package/src/widgets/LearnerDashboardHeader/hooks.js +20 -0
- package/src/widgets/LearnerDashboardHeader/index.ts +1 -0
- package/src/widgets/LearnerDashboardHeader/messages.js +91 -0
- package/src/widgets/LookingForChallengeWidget/__snapshots__/index.test.jsx.snap +45 -0
- package/src/widgets/LookingForChallengeWidget/index.jsx +47 -0
- package/src/widgets/LookingForChallengeWidget/index.scss +6 -0
- package/src/widgets/LookingForChallengeWidget/index.test.jsx +24 -0
- package/src/widgets/LookingForChallengeWidget/messages.js +16 -0
- package/src/widgets/LookingForChallengeWidget/track.js +15 -0
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { createSelector } from 'reselect';
|
|
2
|
+
import { StrictDict } from '../../../../utils';
|
|
3
|
+
|
|
4
|
+
import * as module from './simpleSelectors';
|
|
5
|
+
|
|
6
|
+
export const appSelector = (state) => state.app;
|
|
7
|
+
const mkSimpleSelector = (cb) => createSelector([module.appSelector], cb);
|
|
8
|
+
|
|
9
|
+
// top-level app data selectors
|
|
10
|
+
export const simpleSelectors = StrictDict({
|
|
11
|
+
courseData: mkSimpleSelector(app => app.courseData),
|
|
12
|
+
platformSettings: mkSimpleSelector(app => app.platformSettings),
|
|
13
|
+
suggestedCourses: mkSimpleSelector(app => app.suggestedCourses),
|
|
14
|
+
emailConfirmation: mkSimpleSelector(app => app.emailConfirmation),
|
|
15
|
+
enterpriseDashboard: mkSimpleSelector(app => app.enterpriseDashboard || {}),
|
|
16
|
+
selectSessionModal: mkSimpleSelector(app => app.selectSessionModal),
|
|
17
|
+
pageNumber: mkSimpleSelector(app => app.pageNumber),
|
|
18
|
+
filters: mkSimpleSelector(app => app.filters),
|
|
19
|
+
socialShareSettings: mkSimpleSelector(app => app.socialShareSettings),
|
|
20
|
+
});
|
|
21
|
+
|
|
22
|
+
export const cardSimpleSelectors = StrictDict({
|
|
23
|
+
certificate: ({ certificate }) => certificate,
|
|
24
|
+
course: ({ course }) => course,
|
|
25
|
+
courseProvider: ({ courseProvider }) => courseProvider,
|
|
26
|
+
courseRun: ({ courseRun }) => courseRun,
|
|
27
|
+
credit: ({ credit }) => credit,
|
|
28
|
+
enrollment: ({ enrollment }) => enrollment,
|
|
29
|
+
entitlement: ({ entitlement }) => entitlement,
|
|
30
|
+
gradeData: ({ gradeData }) => gradeData,
|
|
31
|
+
relatedPrograms: ({ programs: { relatedPrograms } }) => relatedPrograms,
|
|
32
|
+
});
|
|
33
|
+
|
|
34
|
+
export const mkCardSelector = (simpleSelector, selector) => (state, cardId) => (
|
|
35
|
+
selector(simpleSelector(module.simpleSelectors.courseData(state)[cardId]))
|
|
36
|
+
);
|
|
37
|
+
|
|
38
|
+
export default simpleSelectors;
|
|
@@ -0,0 +1,75 @@
|
|
|
1
|
+
import { keyStore } from 'utils';
|
|
2
|
+
import * as module from './simpleSelectors';
|
|
3
|
+
|
|
4
|
+
const {
|
|
5
|
+
appSelector,
|
|
6
|
+
simpleSelectors,
|
|
7
|
+
cardSimpleSelectors,
|
|
8
|
+
mkCardSelector,
|
|
9
|
+
} = module;
|
|
10
|
+
|
|
11
|
+
let keys;
|
|
12
|
+
let testData;
|
|
13
|
+
|
|
14
|
+
let testState;
|
|
15
|
+
const testString = 'test-STRING';
|
|
16
|
+
const testCardId = 'testCARD-id';
|
|
17
|
+
|
|
18
|
+
describe('app simple selectors', () => {
|
|
19
|
+
describe('base app selector', () => {
|
|
20
|
+
});
|
|
21
|
+
describe('simple selectors', () => {
|
|
22
|
+
keys = keyStore(simpleSelectors);
|
|
23
|
+
test.each([
|
|
24
|
+
keys.courseData,
|
|
25
|
+
keys.platformSettings,
|
|
26
|
+
keys.suggestedCourses,
|
|
27
|
+
keys.emailConfirmation,
|
|
28
|
+
keys.enterpriseDashboard,
|
|
29
|
+
keys.selectSessionModal,
|
|
30
|
+
keys.pageNumber,
|
|
31
|
+
keys.socialShareSettings,
|
|
32
|
+
keys.filters,
|
|
33
|
+
])('%s app simple selector forwards corresponding data from app store', (key) => {
|
|
34
|
+
testState = { app: { [key]: testString, otherField: 'fake string' } };
|
|
35
|
+
const { preSelectors, cb } = simpleSelectors[key];
|
|
36
|
+
expect(preSelectors).toEqual([appSelector]);
|
|
37
|
+
expect(cb(testState.app)).toEqual(testString);
|
|
38
|
+
});
|
|
39
|
+
describe('cardSimpleSelectors', () => {
|
|
40
|
+
keys = keyStore(cardSimpleSelectors);
|
|
41
|
+
test.each([
|
|
42
|
+
keys.certificate,
|
|
43
|
+
keys.course,
|
|
44
|
+
keys.courseProvider,
|
|
45
|
+
keys.courseRun,
|
|
46
|
+
keys.credit,
|
|
47
|
+
keys.enrollment,
|
|
48
|
+
keys.entitlement,
|
|
49
|
+
keys.gradeData,
|
|
50
|
+
])('%s card simple selector forwards corresponding data from passed object', (key) => {
|
|
51
|
+
testState = { [key]: testString };
|
|
52
|
+
expect(cardSimpleSelectors[key](testState)).toEqual(testString);
|
|
53
|
+
});
|
|
54
|
+
test('relatedPrograms simple selector forwards relatedPrograms from programs obj', () => {
|
|
55
|
+
expect(
|
|
56
|
+
cardSimpleSelectors.relatedPrograms({ programs: { relatedPrograms: testString } }),
|
|
57
|
+
).toEqual(testString);
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
describe('mkCardSelector util', () => {
|
|
61
|
+
it('takes [card simpleSelector, selector] and creates card selector from cardData', () => {
|
|
62
|
+
const selector = (data) => ({ selector: data });
|
|
63
|
+
const simpleSelector = (data) => ({ simpleSelector: data });
|
|
64
|
+
testData = { some: 'test data' };
|
|
65
|
+
const oldCourseData = simpleSelectors.courseData;
|
|
66
|
+
simpleSelectors.courseData = jest.fn().mockReturnValueOnce({ [testCardId]: testData });
|
|
67
|
+
expect(mkCardSelector(simpleSelector, selector)(testState, testCardId)).toEqual(
|
|
68
|
+
selector(simpleSelector(testData)),
|
|
69
|
+
);
|
|
70
|
+
expect(simpleSelectors.courseData).toHaveBeenCalledWith(testState);
|
|
71
|
+
simpleSelectors.courseData = oldCourseData;
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
});
|
|
@@ -0,0 +1,106 @@
|
|
|
1
|
+
/* eslint-disable react-hooks/rules-of-hooks */
|
|
2
|
+
import { useSelector, useDispatch } from 'react-redux';
|
|
3
|
+
|
|
4
|
+
import * as redux from '../../../data/redux';
|
|
5
|
+
import * as module from './app';
|
|
6
|
+
|
|
7
|
+
const selectors = redux.selectors.app;
|
|
8
|
+
const actions = redux.actions.app;
|
|
9
|
+
|
|
10
|
+
/** Simple Selectors **/
|
|
11
|
+
export const usePageNumber = () => useSelector(selectors.pageNumber);
|
|
12
|
+
export const useFilters = () => useSelector(selectors.filters);
|
|
13
|
+
export const useEmailConfirmationData = () => useSelector(selectors.emailConfirmation);
|
|
14
|
+
export const useEnterpriseDashboardData = () => useSelector(selectors.enterpriseDashboard);
|
|
15
|
+
export const usePlatformSettingsData = () => useSelector(selectors.platformSettings);
|
|
16
|
+
export const useSelectSessionModalData = () => useSelector(selectors.selectSessionModal);
|
|
17
|
+
export const useSocialShareSettings = () => useSelector(selectors.socialShareSettings);
|
|
18
|
+
|
|
19
|
+
/** global-level meta-selectors **/
|
|
20
|
+
export const useHasCourses = () => useSelector(selectors.hasCourses);
|
|
21
|
+
export const useCurrentCourseList = (opts) => useSelector(
|
|
22
|
+
state => selectors.currentList(state, opts),
|
|
23
|
+
);
|
|
24
|
+
export const useShowSelectSessionModal = () => useSelector(selectors.showSelectSessionModal);
|
|
25
|
+
|
|
26
|
+
export const useCourseCardData = (selector) => (cardId) => useSelector(
|
|
27
|
+
(state) => selector(state, cardId),
|
|
28
|
+
);
|
|
29
|
+
/** Course Card selectors **/
|
|
30
|
+
const { courseCard } = selectors;
|
|
31
|
+
export const useCardCertificateData = useCourseCardData(courseCard.certificate);
|
|
32
|
+
export const useCardCourseData = useCourseCardData(courseCard.course);
|
|
33
|
+
export const useCardCourseRunData = useCourseCardData(courseCard.courseRun);
|
|
34
|
+
export const useCardCreditData = useCourseCardData(courseCard.credit);
|
|
35
|
+
export const useCardEnrollmentData = useCourseCardData(courseCard.enrollment);
|
|
36
|
+
export const useCardEntitlementData = useCourseCardData(courseCard.entitlement);
|
|
37
|
+
export const useCardGradeData = useCourseCardData(courseCard.gradeData);
|
|
38
|
+
export const useCardProviderData = useCourseCardData(courseCard.courseProvider);
|
|
39
|
+
export const useCardRelatedProgramsData = useCourseCardData(courseCard.relatedPrograms);
|
|
40
|
+
|
|
41
|
+
export const useCardSocialSettingsData = (cardId) => {
|
|
42
|
+
const socialShareSettings = module.useSocialShareSettings();
|
|
43
|
+
const { socialShareUrl } = module.useCardCourseData(cardId);
|
|
44
|
+
const defaultSettings = { isEnabled: false, shareUrl: '' };
|
|
45
|
+
|
|
46
|
+
if (!socialShareSettings) {
|
|
47
|
+
return { facebook: defaultSettings, twitter: defaultSettings };
|
|
48
|
+
}
|
|
49
|
+
const { facebook, twitter } = socialShareSettings;
|
|
50
|
+
const loadSettings = (target) => ({
|
|
51
|
+
isEnabled: target.isEnabled,
|
|
52
|
+
shareUrl: `${socialShareUrl}?${target.utmParams}`,
|
|
53
|
+
});
|
|
54
|
+
return { facebook: loadSettings(facebook), twitter: loadSettings(twitter) };
|
|
55
|
+
};
|
|
56
|
+
|
|
57
|
+
export const useCardExecEdTrackingParam = (cardId) => {
|
|
58
|
+
const { isExecEd2UCourse } = module.useCardEnrollmentData(cardId);
|
|
59
|
+
const { authOrgId } = module.useEnterpriseDashboardData(cardId);
|
|
60
|
+
return isExecEd2UCourse ? `?org_id=${authOrgId}` : '';
|
|
61
|
+
};
|
|
62
|
+
|
|
63
|
+
/** Events **/
|
|
64
|
+
export const useUpdateSelectSessionModalCallback = (cardId) => {
|
|
65
|
+
const dispatch = useDispatch();
|
|
66
|
+
return () => dispatch(actions.updateSelectSessionModal(cardId));
|
|
67
|
+
};
|
|
68
|
+
|
|
69
|
+
export const useTrackCourseEvent = (tracker, cardId, ...args) => {
|
|
70
|
+
const { courseId } = module.useCardCourseRunData(cardId);
|
|
71
|
+
return (e) => tracker(courseId, ...args)(e);
|
|
72
|
+
};
|
|
73
|
+
|
|
74
|
+
export const useSetPageNumber = () => {
|
|
75
|
+
const dispatch = useDispatch();
|
|
76
|
+
return (value) => dispatch(actions.setPageNumber(value));
|
|
77
|
+
};
|
|
78
|
+
|
|
79
|
+
export const useSetFilters = () => {
|
|
80
|
+
const dispatch = useDispatch();
|
|
81
|
+
return (value) => dispatch(actions.setFilters(value));
|
|
82
|
+
};
|
|
83
|
+
|
|
84
|
+
export const useAddFilter = () => {
|
|
85
|
+
const dispatch = useDispatch();
|
|
86
|
+
return (value) => dispatch(actions.addFilter(value));
|
|
87
|
+
};
|
|
88
|
+
|
|
89
|
+
export const useRemoveFilter = () => {
|
|
90
|
+
const dispatch = useDispatch();
|
|
91
|
+
return (value) => dispatch(actions.removeFilter(value));
|
|
92
|
+
};
|
|
93
|
+
|
|
94
|
+
export const useClearFilters = () => {
|
|
95
|
+
const dispatch = useDispatch();
|
|
96
|
+
return (value) => dispatch(actions.clearFilters(value));
|
|
97
|
+
};
|
|
98
|
+
|
|
99
|
+
export const useLoadData = () => {
|
|
100
|
+
const dispatch = useDispatch();
|
|
101
|
+
return ({ courses, ...globalData }) => {
|
|
102
|
+
dispatch(actions.setPageNumber(1));
|
|
103
|
+
dispatch(actions.loadGlobalData(globalData));
|
|
104
|
+
dispatch(actions.loadCourses({ courses }));
|
|
105
|
+
};
|
|
106
|
+
};
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
import { useSelector, useDispatch } from 'react-redux';
|
|
2
|
+
|
|
3
|
+
import * as redux from '../../../data/redux';
|
|
4
|
+
import * as module from './requests';
|
|
5
|
+
|
|
6
|
+
const selectors = redux.selectors.requests;
|
|
7
|
+
const actions = redux.actions.requests;
|
|
8
|
+
|
|
9
|
+
export const statusSelector = selector => (requestName) => useSelector(selector(requestName));
|
|
10
|
+
export const useRequestIsPending = module.statusSelector(selectors.isPending);
|
|
11
|
+
export const useRequestIsFailed = module.statusSelector(selectors.isFailed);
|
|
12
|
+
export const useRequestIsCompleted = module.statusSelector(selectors.isCompleted);
|
|
13
|
+
export const useRequestIsInactive = module.statusSelector(selectors.isInactive);
|
|
14
|
+
export const useRequestError = module.statusSelector(selectors.error);
|
|
15
|
+
export const useRequestErrorCode = module.statusSelector(selectors.errorCode);
|
|
16
|
+
export const useRequestErrorStatus = module.statusSelector(selectors.errorStatus);
|
|
17
|
+
export const useRequestData = module.statusSelector(selectors.data);
|
|
18
|
+
|
|
19
|
+
export const useMakeNetworkRequest = () => {
|
|
20
|
+
const dispatch = useDispatch();
|
|
21
|
+
return ({
|
|
22
|
+
requestKey,
|
|
23
|
+
promise,
|
|
24
|
+
onSuccess,
|
|
25
|
+
onFailure,
|
|
26
|
+
}) => {
|
|
27
|
+
dispatch(actions.startRequest({ requestKey }));
|
|
28
|
+
return promise.then((response) => {
|
|
29
|
+
if (onSuccess) {
|
|
30
|
+
onSuccess(response);
|
|
31
|
+
}
|
|
32
|
+
dispatch(actions.completeRequest({ requestKey, response }));
|
|
33
|
+
}).catch((error) => {
|
|
34
|
+
if (onFailure) {
|
|
35
|
+
onFailure(error);
|
|
36
|
+
}
|
|
37
|
+
dispatch(actions.failRequest({ requestKey, error }));
|
|
38
|
+
});
|
|
39
|
+
};
|
|
40
|
+
};
|
|
41
|
+
|
|
42
|
+
export const useClearRequest = () => {
|
|
43
|
+
const dispatch = useDispatch();
|
|
44
|
+
return (requestKey) => {
|
|
45
|
+
dispatch(actions.clearRequest({ requestKey }));
|
|
46
|
+
};
|
|
47
|
+
};
|
|
@@ -0,0 +1,37 @@
|
|
|
1
|
+
import { combineReducers } from 'redux';
|
|
2
|
+
|
|
3
|
+
import { StrictDict } from '../../utils';
|
|
4
|
+
|
|
5
|
+
import * as app from './app';
|
|
6
|
+
import * as requests from './requests';
|
|
7
|
+
|
|
8
|
+
const modules = {
|
|
9
|
+
app,
|
|
10
|
+
requests,
|
|
11
|
+
};
|
|
12
|
+
|
|
13
|
+
/**
|
|
14
|
+
* Extracts keys from the modules object and the provided propName parameter to locate the
|
|
15
|
+
* corresponding object for that propName.
|
|
16
|
+
* Example: moduleProps('reducer') will return an aggregated object containing the reducer for each module
|
|
17
|
+
*
|
|
18
|
+
* @param {string} propName Used to locate the prop in each module
|
|
19
|
+
* @returns {object} Aggregated values for the provided propName
|
|
20
|
+
*/
|
|
21
|
+
const moduleProps = (propName) => Object.keys(modules).reduce(
|
|
22
|
+
(obj, moduleKey) => {
|
|
23
|
+
const value = modules[moduleKey][propName];
|
|
24
|
+
return value ? { ...obj, [moduleKey]: value } : obj;
|
|
25
|
+
},
|
|
26
|
+
{},
|
|
27
|
+
);
|
|
28
|
+
|
|
29
|
+
const rootReducer = combineReducers(moduleProps('reducer'));
|
|
30
|
+
|
|
31
|
+
const actions = StrictDict(moduleProps('actions'));
|
|
32
|
+
|
|
33
|
+
const selectors = StrictDict(moduleProps('selectors'));
|
|
34
|
+
|
|
35
|
+
export { actions, selectors };
|
|
36
|
+
|
|
37
|
+
export default rootReducer;
|
|
@@ -0,0 +1,53 @@
|
|
|
1
|
+
import { createSlice } from '@reduxjs/toolkit';
|
|
2
|
+
|
|
3
|
+
import { StrictDict } from '../../../utils';
|
|
4
|
+
|
|
5
|
+
import { RequestStates, RequestKeys } from '../../../data/constants/requests';
|
|
6
|
+
|
|
7
|
+
const initialState = {
|
|
8
|
+
[RequestKeys.initialize]: { status: RequestStates.inactive },
|
|
9
|
+
[RequestKeys.refreshList]: { status: RequestStates.inactive },
|
|
10
|
+
[RequestKeys.enrollEntitlementSession]: { status: RequestStates.inactive },
|
|
11
|
+
[RequestKeys.leaveEntitlementSession]: { status: RequestStates.inactive },
|
|
12
|
+
[RequestKeys.masquerade]: { status: RequestStates.inactive },
|
|
13
|
+
};
|
|
14
|
+
|
|
15
|
+
const requests = createSlice({
|
|
16
|
+
name: 'requests',
|
|
17
|
+
initialState,
|
|
18
|
+
reducers: {
|
|
19
|
+
startRequest: (state, { payload }) => ({
|
|
20
|
+
...state,
|
|
21
|
+
[payload.requestKey]: {
|
|
22
|
+
status: RequestStates.pending,
|
|
23
|
+
},
|
|
24
|
+
}),
|
|
25
|
+
completeRequest: (state, { payload }) => ({
|
|
26
|
+
...state,
|
|
27
|
+
[payload.requestKey]: {
|
|
28
|
+
status: RequestStates.completed,
|
|
29
|
+
response: payload.response,
|
|
30
|
+
},
|
|
31
|
+
}),
|
|
32
|
+
failRequest: (state, { payload }) => ({
|
|
33
|
+
...state,
|
|
34
|
+
[payload.requestKey]: {
|
|
35
|
+
status: RequestStates.failed,
|
|
36
|
+
error: payload.error,
|
|
37
|
+
},
|
|
38
|
+
}),
|
|
39
|
+
clearRequest: (state, { payload }) => ({
|
|
40
|
+
...state,
|
|
41
|
+
[payload.requestKey]: {},
|
|
42
|
+
}),
|
|
43
|
+
},
|
|
44
|
+
});
|
|
45
|
+
|
|
46
|
+
const actions = StrictDict(requests.actions);
|
|
47
|
+
const { reducer } = requests;
|
|
48
|
+
|
|
49
|
+
export {
|
|
50
|
+
actions,
|
|
51
|
+
reducer,
|
|
52
|
+
initialState,
|
|
53
|
+
};
|
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
import { RequestStates } from 'data/constants/requests';
|
|
2
|
+
import { initialState, actions, reducer } from './reducer';
|
|
3
|
+
|
|
4
|
+
const testingState = {
|
|
5
|
+
...initialState,
|
|
6
|
+
arbitraryField: 'arbitrary',
|
|
7
|
+
};
|
|
8
|
+
|
|
9
|
+
describe('requests reducer', () => {
|
|
10
|
+
it('has initial state', () => {
|
|
11
|
+
expect(reducer(undefined, {})).toEqual(initialState);
|
|
12
|
+
});
|
|
13
|
+
|
|
14
|
+
const testValue = 'roll for initiative';
|
|
15
|
+
const testKey = 'test-key';
|
|
16
|
+
describe('handling actions', () => {
|
|
17
|
+
describe('startRequest', () => {
|
|
18
|
+
it('adds a pending status for the given key', () => {
|
|
19
|
+
expect(reducer(
|
|
20
|
+
testingState,
|
|
21
|
+
actions.startRequest({ requestKey: testKey }),
|
|
22
|
+
)).toEqual({
|
|
23
|
+
...testingState,
|
|
24
|
+
[testKey]: { status: RequestStates.pending },
|
|
25
|
+
});
|
|
26
|
+
});
|
|
27
|
+
});
|
|
28
|
+
describe('completeRequest', () => {
|
|
29
|
+
it('adds a completed status with passed response', () => {
|
|
30
|
+
expect(reducer(
|
|
31
|
+
testingState,
|
|
32
|
+
actions.completeRequest({ requestKey: testKey, response: testValue }),
|
|
33
|
+
)).toEqual({
|
|
34
|
+
...testingState,
|
|
35
|
+
[testKey]: { status: RequestStates.completed, response: testValue },
|
|
36
|
+
});
|
|
37
|
+
});
|
|
38
|
+
});
|
|
39
|
+
describe('failRequest', () => {
|
|
40
|
+
it('adds a failed status with passed error', () => {
|
|
41
|
+
expect(reducer(
|
|
42
|
+
testingState,
|
|
43
|
+
actions.failRequest({ requestKey: testKey, error: testValue }),
|
|
44
|
+
)).toEqual({
|
|
45
|
+
...testingState,
|
|
46
|
+
[testKey]: { status: RequestStates.failed, error: testValue },
|
|
47
|
+
});
|
|
48
|
+
});
|
|
49
|
+
});
|
|
50
|
+
describe('clearRequest', () => {
|
|
51
|
+
it('cleanup status and error', () => {
|
|
52
|
+
expect(reducer(
|
|
53
|
+
testingState,
|
|
54
|
+
actions.clearRequest({ requestKey: testKey, error: testValue }),
|
|
55
|
+
)).toEqual({
|
|
56
|
+
...testingState,
|
|
57
|
+
[testKey]: {},
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
});
|
|
61
|
+
});
|
|
62
|
+
});
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { StrictDict } from '../../../utils';
|
|
2
|
+
import { RequestStates, RequestKeys } from '../../../data/constants/requests';
|
|
3
|
+
// import * as module from './selectors';
|
|
4
|
+
|
|
5
|
+
export const requestStatus = (state, { requestKey }) => state.requests[requestKey];
|
|
6
|
+
|
|
7
|
+
const statusSelector = (fn) => (requestKey) => (state) => fn(state.requests[requestKey]);
|
|
8
|
+
|
|
9
|
+
export const isInactive = ({ status }) => status === RequestStates.inactive;
|
|
10
|
+
export const isPending = ({ status }) => status === RequestStates.pending;
|
|
11
|
+
export const isCompleted = ({ status }) => status === RequestStates.completed;
|
|
12
|
+
export const isFailed = ({ status }) => status === RequestStates.failed;
|
|
13
|
+
export const error = (request) => request.error;
|
|
14
|
+
export const errorStatus = (request) => request.error?.response?.status;
|
|
15
|
+
export const errorCode = (request) => request.error?.response?.data;
|
|
16
|
+
|
|
17
|
+
export const data = (request) => request.data;
|
|
18
|
+
|
|
19
|
+
export default StrictDict({
|
|
20
|
+
requestStatus,
|
|
21
|
+
isInactive: statusSelector(isInactive),
|
|
22
|
+
isPending: statusSelector(isPending),
|
|
23
|
+
isCompleted: statusSelector(isCompleted),
|
|
24
|
+
isFailed: statusSelector(isFailed),
|
|
25
|
+
error: statusSelector(error),
|
|
26
|
+
errorCode: statusSelector(errorCode),
|
|
27
|
+
errorStatus: statusSelector(errorStatus),
|
|
28
|
+
data: statusSelector(data),
|
|
29
|
+
});
|
|
@@ -0,0 +1,110 @@
|
|
|
1
|
+
import { RequestStates, RequestKeys } from 'data/constants/requests';
|
|
2
|
+
|
|
3
|
+
import selectors from './selectors';
|
|
4
|
+
|
|
5
|
+
const requestKey = 'my-test-request-key';
|
|
6
|
+
const requestData = { some: 'request-data' };
|
|
7
|
+
const inactiveRequest = { status: RequestStates.inactive, some: 'request-data' };
|
|
8
|
+
const pendingRequest = { status: RequestStates.pending, some: 'request-data' };
|
|
9
|
+
const completedRequest = { status: RequestStates.completed, some: 'request-data' };
|
|
10
|
+
const failedRequest = { status: RequestStates.failed, some: 'request-data' };
|
|
11
|
+
|
|
12
|
+
const testValue = 'my-test-value';
|
|
13
|
+
|
|
14
|
+
const testErrorValue = {
|
|
15
|
+
response: {
|
|
16
|
+
status: 500,
|
|
17
|
+
data: 'my-test-error',
|
|
18
|
+
},
|
|
19
|
+
};
|
|
20
|
+
|
|
21
|
+
const testState = {
|
|
22
|
+
requests: {
|
|
23
|
+
[requestKey]: requestData,
|
|
24
|
+
},
|
|
25
|
+
};
|
|
26
|
+
const mockUseSelector = (selector, state) => selector(state);
|
|
27
|
+
const genRequests = (request) => ({
|
|
28
|
+
requests: { [requestKey]: request },
|
|
29
|
+
});
|
|
30
|
+
const select = (selector, request) => (
|
|
31
|
+
mockUseSelector(selector(requestKey), genRequests(request))
|
|
32
|
+
);
|
|
33
|
+
describe('requests selectors unit tests', () => {
|
|
34
|
+
test('requestStatus returns data associated with given key', () => {
|
|
35
|
+
expect(selectors.requestStatus(testState, { requestKey })).toEqual(requestData);
|
|
36
|
+
});
|
|
37
|
+
const testStatusSelector = (selector, matchingRequest) => {
|
|
38
|
+
expect(mockUseSelector(selector(requestKey), testState)).toEqual(false);
|
|
39
|
+
expect(mockUseSelector(
|
|
40
|
+
selector(requestKey),
|
|
41
|
+
{ requests: { [requestKey]: matchingRequest } },
|
|
42
|
+
)).toEqual(true);
|
|
43
|
+
};
|
|
44
|
+
test('isInactive returns true iff the given request is inactive', () => {
|
|
45
|
+
testStatusSelector(selectors.isInactive, inactiveRequest);
|
|
46
|
+
});
|
|
47
|
+
test('isPending returns true iff the given request is pending', () => {
|
|
48
|
+
testStatusSelector(selectors.isPending, pendingRequest);
|
|
49
|
+
});
|
|
50
|
+
test('isCompleted returns true iff the given request is completed', () => {
|
|
51
|
+
testStatusSelector(selectors.isCompleted, completedRequest);
|
|
52
|
+
});
|
|
53
|
+
test('isFailed returns true iff the given request is failed', () => {
|
|
54
|
+
testStatusSelector(selectors.isFailed, failedRequest);
|
|
55
|
+
});
|
|
56
|
+
test('error returns the error from the request', () => {
|
|
57
|
+
expect(select(selectors.error, { error: testValue })).toEqual(testValue);
|
|
58
|
+
});
|
|
59
|
+
test('errorStatus returns the error response status', () => {
|
|
60
|
+
expect(select(selectors.errorStatus, {})).toEqual(undefined);
|
|
61
|
+
expect(select(selectors.errorStatus, { error: {} })).toEqual(undefined);
|
|
62
|
+
expect(select(selectors.errorStatus, { error: { response: {} } })).toEqual(undefined);
|
|
63
|
+
expect(select(selectors.errorStatus, { error: { response: { status: testValue } } }))
|
|
64
|
+
.toEqual(testValue);
|
|
65
|
+
expect(select(selectors.errorStatus, { error: testErrorValue })).toEqual(
|
|
66
|
+
testErrorValue.response.status,
|
|
67
|
+
);
|
|
68
|
+
});
|
|
69
|
+
test('errorCode returns the error response data', () => {
|
|
70
|
+
expect(select(selectors.errorCode, {})).toEqual(undefined);
|
|
71
|
+
expect(select(selectors.errorCode, { error: {} })).toEqual(undefined);
|
|
72
|
+
expect(select(selectors.errorCode, { error: { response: {} } })).toEqual(undefined);
|
|
73
|
+
expect(select(selectors.errorCode, { error: { response: { data: testValue } } }))
|
|
74
|
+
.toEqual(testValue);
|
|
75
|
+
expect(select(selectors.errorCode, { error: testErrorValue })).toEqual(
|
|
76
|
+
testErrorValue.response.data,
|
|
77
|
+
);
|
|
78
|
+
});
|
|
79
|
+
test('data reurns the request data', () => {
|
|
80
|
+
expect(select(selectors.data, { data: testValue })).toEqual(testValue);
|
|
81
|
+
});
|
|
82
|
+
test('masquerade returns the masquerade data', () => {
|
|
83
|
+
const mockResponse = (response) => ({
|
|
84
|
+
requests: {
|
|
85
|
+
[RequestKeys.masquerade]: response,
|
|
86
|
+
},
|
|
87
|
+
});
|
|
88
|
+
expect(selectors.masquerade(mockResponse(completedRequest))).toEqual({
|
|
89
|
+
isMasquerading: true,
|
|
90
|
+
isMasqueradingFailed: false,
|
|
91
|
+
isMasqueradingPending: false,
|
|
92
|
+
masqueradeErrorStatus: undefined,
|
|
93
|
+
});
|
|
94
|
+
expect(selectors.masquerade(mockResponse(pendingRequest))).toEqual({
|
|
95
|
+
isMasquerading: false,
|
|
96
|
+
isMasqueradingFailed: false,
|
|
97
|
+
isMasqueradingPending: true,
|
|
98
|
+
masqueradeErrorStatus: undefined,
|
|
99
|
+
});
|
|
100
|
+
expect(selectors.masquerade(mockResponse({
|
|
101
|
+
...failedRequest,
|
|
102
|
+
error: testErrorValue,
|
|
103
|
+
}))).toEqual({
|
|
104
|
+
isMasquerading: false,
|
|
105
|
+
isMasqueradingFailed: true,
|
|
106
|
+
isMasqueradingPending: false,
|
|
107
|
+
masqueradeErrorStatus: testErrorValue.response.status,
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
});
|
|
@@ -0,0 +1,77 @@
|
|
|
1
|
+
import eventNames from '../../../tracking/constants';
|
|
2
|
+
import {
|
|
3
|
+
client,
|
|
4
|
+
get,
|
|
5
|
+
post,
|
|
6
|
+
stringifyUrl,
|
|
7
|
+
} from './utils';
|
|
8
|
+
import {
|
|
9
|
+
apiKeys,
|
|
10
|
+
unenrollmentAction,
|
|
11
|
+
enableEmailsAction,
|
|
12
|
+
} from './constants';
|
|
13
|
+
import urls from './urls';
|
|
14
|
+
import * as module from './api';
|
|
15
|
+
|
|
16
|
+
/*********************************************************************************
|
|
17
|
+
* GET Actions
|
|
18
|
+
*********************************************************************************/
|
|
19
|
+
export const initializeList = ({ user } = {}) => get(
|
|
20
|
+
stringifyUrl(urls.getInitApiUrl(), { [apiKeys.user]: user }),
|
|
21
|
+
);
|
|
22
|
+
|
|
23
|
+
export const updateEntitlementEnrollment = ({ uuid, courseId }) => post(
|
|
24
|
+
urls.entitlementEnrollment(uuid),
|
|
25
|
+
{ [apiKeys.courseRunId]: courseId },
|
|
26
|
+
);
|
|
27
|
+
|
|
28
|
+
export const deleteEntitlementEnrollment = ({ uuid, isRefundable }) => client()
|
|
29
|
+
.delete(
|
|
30
|
+
stringifyUrl(
|
|
31
|
+
urls.entitlementEnrollment(uuid),
|
|
32
|
+
{ [apiKeys.isRefund]: isRefundable },
|
|
33
|
+
),
|
|
34
|
+
);
|
|
35
|
+
|
|
36
|
+
export const updateEmailSettings = ({ courseId, enable }) => post(
|
|
37
|
+
urls.updateEmailSettings(),
|
|
38
|
+
{ [apiKeys.courseId]: courseId, ...(enable && enableEmailsAction) },
|
|
39
|
+
);
|
|
40
|
+
|
|
41
|
+
export const unenrollFromCourse = ({ courseId }) => post(
|
|
42
|
+
urls.courseUnenroll(),
|
|
43
|
+
{ [apiKeys.courseId]: courseId, ...unenrollmentAction },
|
|
44
|
+
);
|
|
45
|
+
|
|
46
|
+
export const logEvent = ({ eventName, data, courseId }) => post(urls.event(), {
|
|
47
|
+
courserun_key: courseId,
|
|
48
|
+
event_type: eventName,
|
|
49
|
+
page: window.location.href,
|
|
50
|
+
event: JSON.stringify(data),
|
|
51
|
+
});
|
|
52
|
+
|
|
53
|
+
export const logShare = ({ courseId, site }) => module.logEvent({
|
|
54
|
+
eventName: eventNames.shareClicked,
|
|
55
|
+
courseId,
|
|
56
|
+
data: {
|
|
57
|
+
course_id: courseId,
|
|
58
|
+
social_media_site: site,
|
|
59
|
+
location: 'dashboard',
|
|
60
|
+
},
|
|
61
|
+
});
|
|
62
|
+
|
|
63
|
+
export const createCreditRequest = ({ providerId, courseId, username }) => post(
|
|
64
|
+
urls.creditRequestUrl(providerId),
|
|
65
|
+
{ course_key: courseId, username },
|
|
66
|
+
);
|
|
67
|
+
|
|
68
|
+
export default {
|
|
69
|
+
initializeList,
|
|
70
|
+
unenrollFromCourse,
|
|
71
|
+
updateEmailSettings,
|
|
72
|
+
updateEntitlementEnrollment,
|
|
73
|
+
deleteEntitlementEnrollment,
|
|
74
|
+
logEvent,
|
|
75
|
+
logShare,
|
|
76
|
+
createCreditRequest,
|
|
77
|
+
};
|