From d6a40d8c262e0697cd4d720a9ff107eb92251ae5 Mon Sep 17 00:00:00 2001 From: Artur Gaspar Date: Fri, 17 Nov 2023 08:34:30 -0300 Subject: [PATCH] feat: legacy course navigation Add an option to enable the legacy course navigation where clicking a breadcrumb leads to the course index page highlighting the selected section. --- .env | 1 + .env.development | 1 + README.rst | 4 ++ src/course-home/outline-tab/OutlineTab.jsx | 14 +++++- .../outline-tab/OutlineTab.test.jsx | 20 ++++++++ src/course-home/outline-tab/Section.jsx | 10 +++- src/course-home/outline-tab/Section.scss | 14 ++++++ src/course-home/outline-tab/SequenceLink.jsx | 16 ++++++- src/course-home/outline-tab/SequenceLink.scss | 7 +++ src/courseware/course/CourseBreadcrumbs.jsx | 18 +++---- .../course/CourseBreadcrumbs.test.jsx | 47 ++++++++++++++----- src/index.jsx | 1 + 12 files changed, 129 insertions(+), 24 deletions(-) create mode 100644 src/course-home/outline-tab/Section.scss create mode 100644 src/course-home/outline-tab/SequenceLink.scss diff --git a/.env b/.env index b162247399..593c30b8c6 100644 --- a/.env +++ b/.env @@ -13,6 +13,7 @@ DISCOVERY_API_BASE_URL='' DISCUSSIONS_MFE_BASE_URL='' ECOMMERCE_BASE_URL='' ENABLE_JUMPNAV='true' +ENABLE_LEGACY_NAV='' ENABLE_NOTICES='' ENTERPRISE_LEARNER_PORTAL_HOSTNAME='' EXAMS_BASE_URL='' diff --git a/.env.development b/.env.development index 34e014efb0..ec6c3a6ced 100644 --- a/.env.development +++ b/.env.development @@ -13,6 +13,7 @@ DISCOVERY_API_BASE_URL='http://localhost:18381' DISCUSSIONS_MFE_BASE_URL='http://localhost:2002' ECOMMERCE_BASE_URL='http://localhost:18130' ENABLE_JUMPNAV='true' +ENABLE_LEGACY_NAV='' ENABLE_NOTICES='' ENTERPRISE_LEARNER_PORTAL_HOSTNAME='localhost:8734' EXAMS_BASE_URL='' diff --git a/README.rst b/README.rst index df3cf427f9..daf747ab3c 100644 --- a/README.rst +++ b/README.rst @@ -119,6 +119,10 @@ ENABLE_JUMPNAV This feature flag is slated to be removed as jumpnav becomes default. Follow the progress of this ticket here: https://openedx.atlassian.net/browse/TNL-8678 +ENABLE_LEGACY_NAV + Enables the legacy behaviour in the course breadcrumbs, where links lead to + the course index highlighting the selected course section or subsection. + SOCIAL_UTM_MILESTONE_CAMPAIGN This value is passed as the ``utm_campaign`` parameter for social-share links when celebrating learning milestones in the course. Optional. diff --git a/src/course-home/outline-tab/OutlineTab.jsx b/src/course-home/outline-tab/OutlineTab.jsx index a2ce8101ea..8c40a18c06 100644 --- a/src/course-home/outline-tab/OutlineTab.jsx +++ b/src/course-home/outline-tab/OutlineTab.jsx @@ -123,6 +123,14 @@ const OutlineTab = ({ intl }) => { } }, [location.search]); + // A section or subsection is selected by its id being the location hash part. + // location.hash will contain an initial # sign, so remove it here. + const hashValue = location.hash.substring(1); + const selectedSectionId = rootCourseId && courses[rootCourseId].sectionIds.find((sectionId) => ( + // Section is selected or contains selected subsection. + (hashValue === sectionId) || sections[sectionId].sequenceIds.includes(hashValue) + )); + return ( <>
@@ -173,7 +181,11 @@ const OutlineTab = ({ intl }) => {
diff --git a/src/course-home/outline-tab/OutlineTab.test.jsx b/src/course-home/outline-tab/OutlineTab.test.jsx index c6fc547a0c..fb06d33cbe 100644 --- a/src/course-home/outline-tab/OutlineTab.test.jsx +++ b/src/course-home/outline-tab/OutlineTab.test.jsx @@ -107,6 +107,26 @@ describe('Outline Tab', () => { expect(expandedSectionNode).toHaveAttribute('aria-expanded', 'true'); }); + it('expands selected section', async () => { + const { courseBlocks, sectionBlocks } = await buildMinimalCourseBlocks(courseId, 'Title'); + setTabData({ + course_blocks: { blocks: courseBlocks.blocks }, + }); + await fetchAndRender(`http://localhost/#${sectionBlocks[0].id}`); + const expandedSectionNode = screen.getByRole('button', { name: /Title of Section/ }); + expect(expandedSectionNode).toHaveAttribute('aria-expanded', 'true'); + }); + + it('expands section that contains selected subsection', async () => { + const { courseBlocks, sequenceBlocks } = await buildMinimalCourseBlocks(courseId, 'Title'); + setTabData({ + course_blocks: { blocks: courseBlocks.blocks }, + }); + await fetchAndRender(`http://localhost/#${sequenceBlocks[0].id}`); + const expandedSectionNode = screen.getByRole('button', { name: /Title of Section/ }); + expect(expandedSectionNode).toHaveAttribute('aria-expanded', 'true'); + }); + it('handles expand/collapse all button click', async () => { await fetchAndRender(); // Button renders as "Expand All" diff --git a/src/course-home/outline-tab/Section.jsx b/src/course-home/outline-tab/Section.jsx index 3de888a89a..d7a9397fc0 100644 --- a/src/course-home/outline-tab/Section.jsx +++ b/src/course-home/outline-tab/Section.jsx @@ -1,5 +1,7 @@ import React, { useEffect, useState } from 'react'; import PropTypes from 'prop-types'; +import classNames from 'classnames'; +import { useLocation } from 'react-router-dom'; import { injectIntl, intlShape } from '@edx/frontend-platform/i18n'; import { Collapsible, IconButton } from '@edx/paragon'; import { faCheckCircle as fasCheckCircle, faMinus, faPlus } from '@fortawesome/free-solid-svg-icons'; @@ -11,6 +13,7 @@ import { useModel } from '../../generic/model-store'; import genericMessages from '../../generic/messages'; import messages from './messages'; +import './Section.scss'; const Section = ({ courseId, @@ -29,6 +32,9 @@ const Section = ({ sequences, }, } = useModel('outline', courseId); + // Remove the initial # sign. + const hashValue = useLocation().hash.substring(1); + const selected = hashValue === section.id; const [open, setOpen] = useState(defaultOpen); @@ -74,7 +80,7 @@ const Section = ({ return (
  • ))} diff --git a/src/course-home/outline-tab/Section.scss b/src/course-home/outline-tab/Section.scss new file mode 100644 index 0000000000..9a0976b46c --- /dev/null +++ b/src/course-home/outline-tab/Section.scss @@ -0,0 +1,14 @@ +@import "~@edx/brand/paragon/variables"; +@import "~@edx/paragon/scss/core/core"; +@import "~@edx/brand/paragon/overrides"; + +.section > div > div > .collapsible-body { + /* Internal SequenceLink components will have padding so the highlighting + * reaches the top and/or bottom of the collapsible. */ + padding-top: 0; + padding-bottom: 0; +} + +.section-selected > .collapsible-trigger { + background-color: $light-300; +} diff --git a/src/course-home/outline-tab/SequenceLink.jsx b/src/course-home/outline-tab/SequenceLink.jsx index 0530d53e66..2ffe7fdf37 100644 --- a/src/course-home/outline-tab/SequenceLink.jsx +++ b/src/course-home/outline-tab/SequenceLink.jsx @@ -15,13 +15,16 @@ import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import EffortEstimate from '../../shared/effort-estimate'; import { useModel } from '../../generic/model-store'; import messages from './messages'; +import './SequenceLink.scss'; const SequenceLink = ({ id, intl, courseId, first, + last, sequence, + selected, }) => { const { complete, @@ -84,8 +87,15 @@ const SequenceLink = ({ ); return ( -
  • -
    +
  • +
    {complete ? ( @@ -129,7 +139,9 @@ SequenceLink.propTypes = { intl: intlShape.isRequired, courseId: PropTypes.string.isRequired, first: PropTypes.bool.isRequired, + last: PropTypes.bool.isRequired, sequence: PropTypes.shape().isRequired, + selected: PropTypes.bool.isRequired, }; export default injectIntl(SequenceLink); diff --git a/src/course-home/outline-tab/SequenceLink.scss b/src/course-home/outline-tab/SequenceLink.scss new file mode 100644 index 0000000000..bfcf6e950a --- /dev/null +++ b/src/course-home/outline-tab/SequenceLink.scss @@ -0,0 +1,7 @@ +@import "~@edx/brand/paragon/variables"; +@import "~@edx/paragon/scss/core/core"; +@import "~@edx/brand/paragon/overrides"; + +.sequence-link-selected { + background-color: $light-300; +} diff --git a/src/courseware/course/CourseBreadcrumbs.jsx b/src/courseware/course/CourseBreadcrumbs.jsx index a49906a894..e3295d8c9e 100644 --- a/src/courseware/course/CourseBreadcrumbs.jsx +++ b/src/courseware/course/CourseBreadcrumbs.jsx @@ -25,6 +25,15 @@ const CourseBreadcrumb = ({ const showRegularLink = getConfig().ENABLE_JUMPNAV !== 'true' || content.length < 2 || !isStaff; const [isOpen, open, close] = useToggle(false); const [target, setTarget] = useState(null); + + let regularLinkRoute; + if (getConfig().ENABLE_LEGACY_NAV === 'true') { + regularLinkRoute = `/course/${courseId}/home#${defaultContent.id}`; + } else if (defaultContent.sequences.length) { + regularLinkRoute = `/course/${courseId}/${defaultContent.sequences[0].id}`; + } else { + regularLinkRoute = `/course/${courseId}/${defaultContent.id}`; + } return ( <> {withSeparator && ( @@ -40,14 +49,7 @@ const CourseBreadcrumb = ({ data-testid="breadcrumb-item" > {showRegularLink ? ( - + {defaultContent.label} ) : ( diff --git a/src/courseware/course/CourseBreadcrumbs.test.jsx b/src/courseware/course/CourseBreadcrumbs.test.jsx index f51ead34c3..d1f557fb22 100644 --- a/src/courseware/course/CourseBreadcrumbs.test.jsx +++ b/src/courseware/course/CourseBreadcrumbs.test.jsx @@ -112,23 +112,46 @@ describe('CourseBreadcrumbs', () => { }, ], ]); - render( - - - - , - , - ); it('renders course breadcrumbs as expected', async () => { + await render( + + + + , + , + ); expect(screen.queryAllByRole('link')).toHaveLength(1); const courseHomeButtonDestination = screen.getAllByRole('link')[0].href; expect(courseHomeButtonDestination).toBe('http://localhost/course/course-v1:edX+DemoX+Demo_Course/home'); expect(screen.getByRole('navigation', { name: 'breadcrumb' })).toBeInTheDocument(); expect(screen.queryAllByTestId('breadcrumb-item')).toHaveLength(2); }); + it('renders legacy navigation links as expected', async () => { + getConfig.mockImplementation(() => ({ + ENABLE_JUMPNAV: 'false', + ENABLE_LEGACY_NAV: 'true', + })); + await render( + + + + , + , + ); + expect(screen.queryAllByRole('link')).toHaveLength(3); + const sectionButtonDestination = screen.getAllByRole('link')[1].href; + expect(sectionButtonDestination).toBe('http://localhost/course/course-v1:edX+DemoX+Demo_Course/home#block-v1:edX+DemoX+Demo_Course+type@chapter+block@interactive_demonstrations'); + const sequenceButtonDestination = screen.getAllByRole('link')[2].href; + expect(sequenceButtonDestination).toBe('http://localhost/course/course-v1:edX+DemoX+Demo_Course/home#block-v1:edX+DemoX+Demo_Course+type@sequential+block@basic_questions'); + expect(screen.queryAllByRole('button')).toHaveLength(0); + }); }); diff --git a/src/index.jsx b/src/index.jsx index d99be587a7..6de75a06ab 100755 --- a/src/index.jsx +++ b/src/index.jsx @@ -154,6 +154,7 @@ initialize({ DISCUSSIONS_MFE_BASE_URL: process.env.DISCUSSIONS_MFE_BASE_URL || null, ENTERPRISE_LEARNER_PORTAL_HOSTNAME: process.env.ENTERPRISE_LEARNER_PORTAL_HOSTNAME || null, ENABLE_JUMPNAV: process.env.ENABLE_JUMPNAV || null, + ENABLE_LEGACY_NAV: process.env.ENABLE_LEGACY_NAV || null, ENABLE_NOTICES: process.env.ENABLE_NOTICES || null, INSIGHTS_BASE_URL: process.env.INSIGHTS_BASE_URL || null, SEARCH_CATALOG_URL: process.env.SEARCH_CATALOG_URL || null,