diff --git a/src/containers/CoursesPanel/CourseList/index.jsx b/src/containers/CoursesPanel/CourseList/index.jsx
index 2f6111ae..2b178b23 100644
--- a/src/containers/CoursesPanel/CourseList/index.jsx
+++ b/src/containers/CoursesPanel/CourseList/index.jsx
@@ -9,9 +9,10 @@ import CourseCard from 'containers/CourseCard';
import { useIsCollapsed } from './hooks';
-export const CourseList = ({
- filterOptions, setPageNumber, numPages, showFilters, visibleList,
-}) => {
+export const CourseList = ({ courseListData }) => {
+ const {
+ filterOptions, setPageNumber, numPages, showFilters, visibleList,
+ } = courseListData;
const isCollapsed = useIsCollapsed();
return (
<>
@@ -38,14 +39,16 @@ export const CourseList = ({
);
};
-CourseList.propTypes = {
+export const courseListDataShape = PropTypes.shape({
showFilters: PropTypes.bool.isRequired,
- // eslint-disable-next-line react/forbid-prop-types
- visibleList: PropTypes.arrayOf(PropTypes.object).isRequired,
- // eslint-disable-next-line react/forbid-prop-types
- filterOptions: PropTypes.object.isRequired,
+ visibleList: PropTypes.arrayOf(PropTypes.shape()).isRequired,
+ filterOptions: PropTypes.shape().isRequired,
numPages: PropTypes.number.isRequired,
setPageNumber: PropTypes.func.isRequired,
+});
+
+CourseList.propTypes = {
+ courseListData: courseListDataShape,
};
export default CourseList;
diff --git a/src/containers/CoursesPanel/CourseList/index.test.jsx b/src/containers/CoursesPanel/CourseList/index.test.jsx
index a2e09b3f..0584eda5 100644
--- a/src/containers/CoursesPanel/CourseList/index.test.jsx
+++ b/src/containers/CoursesPanel/CourseList/index.test.jsx
@@ -23,7 +23,7 @@ describe('CourseList', () => {
useIsCollapsed.mockReturnValue(false);
const createWrapper = (courseListData = defaultCourseListData) => (
- shallow()
+ shallow()
);
describe('no courses or filters', () => {
diff --git a/src/containers/CoursesPanel/__snapshots__/index.test.jsx.snap b/src/containers/CoursesPanel/__snapshots__/index.test.jsx.snap
index 9ec9d18a..16874328 100644
--- a/src/containers/CoursesPanel/__snapshots__/index.test.jsx.snap
+++ b/src/containers/CoursesPanel/__snapshots__/index.test.jsx.snap
@@ -18,11 +18,7 @@ exports[`CoursesPanel no courses snapshot 1`] = `
-
-
-
+
`;
@@ -44,16 +40,16 @@ exports[`CoursesPanel with courses snapshot 1`] = `
-
-
-
+
`;
diff --git a/src/containers/CoursesPanel/index.jsx b/src/containers/CoursesPanel/index.jsx
index dacb8079..6bf525ab 100644
--- a/src/containers/CoursesPanel/index.jsx
+++ b/src/containers/CoursesPanel/index.jsx
@@ -1,15 +1,13 @@
import React from 'react';
-import { PluginSlot } from '@openedx/frontend-plugin-framework';
import { useIntl } from '@edx/frontend-platform/i18n';
import { reduxHooks } from 'hooks';
import {
CourseFilterControls,
} from 'containers/CourseFilterControls';
-import NoCoursesView from './NoCoursesView';
-
-import CourseList from './CourseList';
+import CourseListSlot from 'plugin-slots/CourseListSlot';
+import NoCoursesViewSlot from 'plugin-slots/NoCoursesViewSlot';
import { useCourseListData } from './hooks';
@@ -34,19 +32,7 @@ export const CoursesPanel = () => {
- {hasCourses ? (
-
-
-
- ) : (
-
-
-
- )}
+ {hasCourses ? : }
);
};
diff --git a/src/containers/Dashboard/DashboardLayout.jsx b/src/containers/Dashboard/DashboardLayout.jsx
index 8a5db140..d970998c 100644
--- a/src/containers/Dashboard/DashboardLayout.jsx
+++ b/src/containers/Dashboard/DashboardLayout.jsx
@@ -3,7 +3,7 @@ import PropTypes from 'prop-types';
import { Container, Col, Row } from '@openedx/paragon';
-import WidgetSidebar from '../WidgetContainers/WidgetSidebar';
+import WidgetSidebarSlot from 'plugin-slots/WidgetSidebarSlot';
import hooks from './hooks';
@@ -42,7 +42,7 @@ export const DashboardLayout = ({ children }) => {
{!isCollapsed && (
)}
-
+
diff --git a/src/containers/Dashboard/DashboardLayout.test.jsx b/src/containers/Dashboard/DashboardLayout.test.jsx
index 247ea79c..4e926375 100644
--- a/src/containers/Dashboard/DashboardLayout.test.jsx
+++ b/src/containers/Dashboard/DashboardLayout.test.jsx
@@ -36,9 +36,9 @@ describe('DashboardLayout', () => {
const columns = el.instance.findByType(Row)[0].findByType(Col);
expect(columns[0].children).not.toHaveLength(0);
});
- it('displays WidgetSidebar in second column', () => {
+ it('displays WidgetSidebarSlot in second column', () => {
const columns = el.instance.findByType(Row)[0].findByType(Col);
- expect(columns[1].findByType('WidgetSidebar')).toHaveLength(1);
+ expect(columns[1].findByType('WidgetSidebarSlot')).toHaveLength(1);
});
};
const testSidebarLayout = () => {
diff --git a/src/containers/Dashboard/__snapshots__/DashboardLayout.test.jsx.snap b/src/containers/Dashboard/__snapshots__/DashboardLayout.test.jsx.snap
index 161196b2..dc7e7a39 100644
--- a/src/containers/Dashboard/__snapshots__/DashboardLayout.test.jsx.snap
+++ b/src/containers/Dashboard/__snapshots__/DashboardLayout.test.jsx.snap
@@ -38,7 +38,7 @@ exports[`DashboardLayout collapsed sidebar not showing snapshot 1`] = `
}
}
>
-
+
@@ -82,7 +82,7 @@ exports[`DashboardLayout collapsed sidebar showing snapshot 1`] = `
}
}
>
-
+
@@ -131,7 +131,7 @@ exports[`DashboardLayout not collapsed sidebar not showing snapshot 1`] = `
>
-
+
@@ -180,7 +180,7 @@ exports[`DashboardLayout not collapsed sidebar showing snapshot 1`] = `
>
-
+
diff --git a/src/containers/WidgetContainers/WidgetSidebar/__snapshots__/index.test.jsx.snap b/src/containers/WidgetContainers/WidgetSidebar/__snapshots__/index.test.jsx.snap
deleted file mode 100644
index c2d1774d..00000000
--- a/src/containers/WidgetContainers/WidgetSidebar/__snapshots__/index.test.jsx.snap
+++ /dev/null
@@ -1,15 +0,0 @@
-// Jest Snapshot v1, https://goo.gl/fbAQLP
-
-exports[`WidgetSidebar snapshots 1`] = `
-
-`;
diff --git a/src/containers/WidgetContainers/WidgetSidebar/index.jsx b/src/containers/WidgetContainers/WidgetSidebar/index.jsx
deleted file mode 100644
index 744387ce..00000000
--- a/src/containers/WidgetContainers/WidgetSidebar/index.jsx
+++ /dev/null
@@ -1,23 +0,0 @@
-import React from 'react';
-import classNames from 'classnames';
-
-import { reduxHooks } from 'hooks';
-import { PluginSlot } from '@openedx/frontend-plugin-framework';
-
-// eslint-disable-next-line arrow-body-style
-export const WidgetSidebar = () => {
- const hasCourses = reduxHooks.useHasCourses();
-
- const widgetSidebarClassNames = classNames('widget-sidebar', { 'px-2': !hasCourses });
- const innerWrapperClassNames = classNames('d-flex', { 'flex-column': hasCourses });
-
- return (
-
- );
-};
-
-export default WidgetSidebar;
diff --git a/src/plugin-slots/CourseListSlot/README.md b/src/plugin-slots/CourseListSlot/README.md
new file mode 100644
index 00000000..6ced154d
--- /dev/null
+++ b/src/plugin-slots/CourseListSlot/README.md
@@ -0,0 +1,60 @@
+# Course List Slot
+
+### Slot ID: `course_list_slot`
+
+## Plugin Props
+
+* courseListData
+
+## Description
+
+This slot is used for replacing or adding content around the `CourseList` component. The `CourseListSlot` is only rendered if the learner has enrolled in at least one course.
+
+## Example
+
+The space will show the `CourseList` component by default. This can be disabled in the configuration with the `keepDefault` boolean.
+
+![Screenshot of the CourseListSlot](./images/course_list_slot.png)
+
+Setting the MFE's `env.config.jsx` to the following will replace the default experience with a list of course titles.
+
+![Screenshot of a custom course list](./images/readme_custom_course_list.png)
+
+```js
+import { DIRECT_PLUGIN, PLUGIN_OPERATIONS } from '@openedx/frontend-plugin-framework';
+
+const config = {
+ pluginSlots: {
+ course_list_slot: {
+ // Hide the default CourseList component
+ keepDefault: false,
+ plugins: [
+ {
+ op: PLUGIN_OPERATIONS.Insert,
+ widget: {
+ id: 'custom_course_list',
+ type: DIRECT_PLUGIN,
+ priority: 60,
+ RenderWidget: ({ courseListData }) => {
+ // Extract the "visibleList"
+ const courses = courseListData.visibleList;
+ // Render a list of course names
+ return (
+
+ {courses.map(courseData => (
+
+ {courseData.course.courseName}
+
+ ))}
+
+ )
+ },
+ },
+ },
+ ],
+ },
+ },
+}
+
+export default config;
+```
\ No newline at end of file
diff --git a/src/plugin-slots/CourseListSlot/images/course_list_slot.png b/src/plugin-slots/CourseListSlot/images/course_list_slot.png
new file mode 100644
index 00000000..76136d53
Binary files /dev/null and b/src/plugin-slots/CourseListSlot/images/course_list_slot.png differ
diff --git a/src/plugin-slots/CourseListSlot/images/readme_custom_course_list.png b/src/plugin-slots/CourseListSlot/images/readme_custom_course_list.png
new file mode 100644
index 00000000..f47accc8
Binary files /dev/null and b/src/plugin-slots/CourseListSlot/images/readme_custom_course_list.png differ
diff --git a/src/plugin-slots/CourseListSlot/index.jsx b/src/plugin-slots/CourseListSlot/index.jsx
new file mode 100644
index 00000000..d9533978
--- /dev/null
+++ b/src/plugin-slots/CourseListSlot/index.jsx
@@ -0,0 +1,16 @@
+import React from 'react';
+
+import { PluginSlot } from '@openedx/frontend-plugin-framework';
+import { CourseList, courseListDataShape } from 'containers/CoursesPanel/CourseList';
+
+export const CourseListSlot = ({ courseListData }) => (
+
+
+
+);
+
+CourseListSlot.propTypes = {
+ courseListData: courseListDataShape,
+};
+
+export default CourseListSlot;
diff --git a/src/plugin-slots/NoCoursesViewSlot/README.md b/src/plugin-slots/NoCoursesViewSlot/README.md
new file mode 100644
index 00000000..92389fae
--- /dev/null
+++ b/src/plugin-slots/NoCoursesViewSlot/README.md
@@ -0,0 +1,47 @@
+# No Courses View Slot
+
+### Slot ID: `no_courses_view_slot`
+
+## Description
+
+This slot is used for replacing or adding content around the `NoCoursesView` component. The `NoCoursesViewSlot` only renders if the learner has not yet enrolled in any courses.
+
+## Example
+
+The space will show the `NoCoursesView` by default. This can be disabled in the configuration with the `keepDefault` boolean.
+
+![Screenshot of the no courses view](./images/no_courses_view_slot.png)
+
+Setting the MFE's `env.config.jsx` to the following will replace the default experience with a custom call-to-action component.
+
+![Screenshot of a custom no courses view](./images/readme_custom_no_courses_view.png)
+
+```js
+import { DIRECT_PLUGIN, PLUGIN_OPERATIONS } from '@openedx/frontend-plugin-framework';
+
+const config = {
+ pluginSlots: {
+ no_courses_view_slot: {
+ // Hide the default NoCoursesView component
+ keepDefault: false,
+ plugins: [
+ {
+ op: PLUGIN_OPERATIONS.Insert,
+ widget: {
+ id: 'custom_no_courses_CTA',
+ type: DIRECT_PLUGIN,
+ priority: 60,
+ RenderWidget: () => (
+
+ Check out our catalog of courses and start learning today!
+
+ ),
+ },
+ },
+ ],
+ },
+ },
+}
+
+export default config;
+```
\ No newline at end of file
diff --git a/src/plugin-slots/NoCoursesViewSlot/images/no_course_view_slot.png b/src/plugin-slots/NoCoursesViewSlot/images/no_course_view_slot.png
new file mode 100644
index 00000000..ecbdbacc
Binary files /dev/null and b/src/plugin-slots/NoCoursesViewSlot/images/no_course_view_slot.png differ
diff --git a/src/plugin-slots/NoCoursesViewSlot/images/readme_custom_no_courses.png b/src/plugin-slots/NoCoursesViewSlot/images/readme_custom_no_courses.png
new file mode 100644
index 00000000..6d33e0c8
Binary files /dev/null and b/src/plugin-slots/NoCoursesViewSlot/images/readme_custom_no_courses.png differ
diff --git a/src/plugin-slots/NoCoursesViewSlot/index.jsx b/src/plugin-slots/NoCoursesViewSlot/index.jsx
new file mode 100644
index 00000000..cbb70e41
--- /dev/null
+++ b/src/plugin-slots/NoCoursesViewSlot/index.jsx
@@ -0,0 +1,12 @@
+import React from 'react';
+
+import { PluginSlot } from '@openedx/frontend-plugin-framework';
+import NoCoursesView from 'containers/CoursesPanel/NoCoursesView';
+
+export const NoCoursesViewSlot = () => (
+
+
+
+);
+
+export default NoCoursesViewSlot;
diff --git a/src/plugin-slots/README.md b/src/plugin-slots/README.md
index 30a6a047..5a134b99 100644
--- a/src/plugin-slots/README.md
+++ b/src/plugin-slots/README.md
@@ -1,3 +1,6 @@
# `frontend-app-learner-dashboard` Plugin Slots
* [`footer_slot`](./FooterSlot/)
+* [`widget_sidebar_slot`](./WidgetSidebarSlot/)
+* [`course_list_slot`](./CourseListSlot/)
+* [`no_courses_view_slot`](./NoCoursesViewSlot/)
\ No newline at end of file
diff --git a/src/plugin-slots/WidgetSidebarSlot/README.md b/src/plugin-slots/WidgetSidebarSlot/README.md
new file mode 100644
index 00000000..39fcbf1a
--- /dev/null
+++ b/src/plugin-slots/WidgetSidebarSlot/README.md
@@ -0,0 +1,58 @@
+# Widget Sidebar Slot
+
+### Slot ID: `widget_sidebar_slot`
+
+## Description
+
+This slot is used for adding content to the right-hand sidebar.
+
+## Example
+
+The space will show the `LookingForChallengeWidget` by default. This can be disabled in the configuration with the `keepDefault` boolean.
+
+![Screenshot of the widget sidebar](./images/widget_sidebar_slot.png)
+
+Setting the MFE's `env.config.jsx` to the following will replace the default experience with a custom sidebar component.
+
+![Screenshot of a custom call-to-action in the sidebar](./images/readme_custom_sidebar.png)
+
+```js
+import { DIRECT_PLUGIN, PLUGIN_OPERATIONS } from '@openedx/frontend-plugin-framework';
+
+const config = {
+ pluginSlots: {
+ widget_sidebar_slot: {
+ // Hide the default LookingForChallenge component
+ keepDefault: false,
+ plugins: [
+ {
+ op: PLUGIN_OPERATIONS.Insert,
+ widget: {
+ id: 'custom_sidebar_panel',
+ type: DIRECT_PLUGIN,
+ priority: 60,
+ RenderWidget: () => (
+
+
+ Sidebar Menu
+
+
+ sidebar item #1
+
+
+ sidebar item #2
+
+
+ sidebar item #3
+
+
+ ),
+ },
+ },
+ ],
+ },
+ },
+}
+
+export default config;
+```
\ No newline at end of file
diff --git a/src/plugin-slots/WidgetSidebarSlot/__snapshots__/index.test.jsx.snap b/src/plugin-slots/WidgetSidebarSlot/__snapshots__/index.test.jsx.snap
new file mode 100644
index 00000000..8a7ac4d7
--- /dev/null
+++ b/src/plugin-slots/WidgetSidebarSlot/__snapshots__/index.test.jsx.snap
@@ -0,0 +1,9 @@
+// Jest Snapshot v1, https://goo.gl/fbAQLP
+
+exports[`WidgetSidebar snapshots 1`] = `
+
+`;
diff --git a/src/plugin-slots/WidgetSidebarSlot/images/readme_custom_sidebar.png b/src/plugin-slots/WidgetSidebarSlot/images/readme_custom_sidebar.png
new file mode 100644
index 00000000..370d69d2
Binary files /dev/null and b/src/plugin-slots/WidgetSidebarSlot/images/readme_custom_sidebar.png differ
diff --git a/src/plugin-slots/WidgetSidebarSlot/images/widget_sidebar_slot.png b/src/plugin-slots/WidgetSidebarSlot/images/widget_sidebar_slot.png
new file mode 100644
index 00000000..b2e7f081
Binary files /dev/null and b/src/plugin-slots/WidgetSidebarSlot/images/widget_sidebar_slot.png differ
diff --git a/src/plugin-slots/WidgetSidebarSlot/index.jsx b/src/plugin-slots/WidgetSidebarSlot/index.jsx
new file mode 100644
index 00000000..d4bebe42
--- /dev/null
+++ b/src/plugin-slots/WidgetSidebarSlot/index.jsx
@@ -0,0 +1,13 @@
+import React from 'react';
+
+import { PluginSlot } from '@openedx/frontend-plugin-framework';
+import LookingForChallengeWidget from 'widgets/LookingForChallengeWidget';
+
+// eslint-disable-next-line arrow-body-style
+export const WidgetSidebarSlot = () => (
+
+);
+
+export default WidgetSidebarSlot;
diff --git a/src/containers/WidgetContainers/WidgetSidebar/index.test.jsx b/src/plugin-slots/WidgetSidebarSlot/index.test.jsx
similarity index 81%
rename from src/containers/WidgetContainers/WidgetSidebar/index.test.jsx
rename to src/plugin-slots/WidgetSidebarSlot/index.test.jsx
index a8a6e239..591ddccb 100644
--- a/src/containers/WidgetContainers/WidgetSidebar/index.test.jsx
+++ b/src/plugin-slots/WidgetSidebarSlot/index.test.jsx
@@ -1,6 +1,6 @@
import { shallow } from '@edx/react-unit-test-utils';
-import WidgetSidebar from '.';
+import WidgetSidebarSlot from '.';
jest.mock('widgets/LookingForChallengeWidget', () => 'LookingForChallengeWidget');
@@ -12,7 +12,7 @@ describe('WidgetSidebar', () => {
beforeEach(() => jest.resetAllMocks());
test('snapshots', () => {
- const wrapper = shallow();
+ const wrapper = shallow();
expect(wrapper.snapshot).toMatchSnapshot();
});
});
diff --git a/src/test/app.test.jsx b/src/test/app.test.jsx
index 89b2d323..1d5ddacc 100644
--- a/src/test/app.test.jsx
+++ b/src/test/app.test.jsx
@@ -42,7 +42,7 @@ jest.unmock('react-redux');
jest.unmock('reselect');
jest.unmock('hooks');
-jest.mock('containers/WidgetContainers/WidgetSidebar', () => jest.fn(() => 'widget-sidebar'));
+jest.mock('plugin-slots/WidgetSidebarSlot', () => jest.fn(() => 'widget-sidebar'));
jest.mock('components/NoticesWrapper', () => 'notices-wrapper');
jest.mock('@edx/frontend-platform', () => ({
diff --git a/src/widgets/LookingForChallengeWidget/index.jsx b/src/widgets/LookingForChallengeWidget/index.jsx
index abef370a..9410c99a 100644
--- a/src/widgets/LookingForChallengeWidget/index.jsx
+++ b/src/widgets/LookingForChallengeWidget/index.jsx
@@ -8,7 +8,7 @@ import { reduxHooks } from 'hooks';
import moreCoursesSVG from 'assets/more-courses-sidewidget.svg';
import { baseAppUrl } from 'data/services/lms/urls';
-import track from './track';
+import { findCoursesWidgetClicked } from './track';
import messages from './messages';
import './index.scss';
@@ -17,6 +17,8 @@ export const arrowIcon = ();
export const LookingForChallengeWidget = () => {
const { formatMessage } = useIntl();
const { courseSearchUrl } = reduxHooks.usePlatformSettingsData();
+ const hyperlinkDestination = baseAppUrl(courseSearchUrl) || '';
+
return (
{
{formatMessage(messages.findCoursesButton, { arrow: arrowIcon })}
diff --git a/src/widgets/LookingForChallengeWidget/track.js b/src/widgets/LookingForChallengeWidget/track.js
index 70549d80..d18f1e17 100644
--- a/src/widgets/LookingForChallengeWidget/track.js
+++ b/src/widgets/LookingForChallengeWidget/track.js
@@ -8,3 +8,8 @@ export const linkNames = StrictDict({
export const findCoursesWidgetClicked = (href) => track.findCourses.findCoursesClicked(href, {
linkName: linkNames.findCoursesWidget,
});
+
+export default {
+ linkNames,
+ findCoursesWidgetClicked,
+};