diff --git a/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/StudentsSection/index.tsx b/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/StudentsSection/index.tsx new file mode 100644 index 0000000000..87bb737cb0 --- /dev/null +++ b/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/StudentsSection/index.tsx @@ -0,0 +1,29 @@ +import { FormattedMessage, defineMessages } from 'react-intl'; +import { DashboardCard } from 'widgets/Dashboard/components/DashboardCard'; + +const messages = defineMessages({ + loading: { + defaultMessage: 'Loading classrooms ...', + description: 'Message displayed while loading a classrooms', + id: 'components.TeacherCourseClassroomsDashboardLoader.StudentsSection.loading', + }, + teacherListTitle: { + defaultMessage: 'Educational team', + description: 'Message displayed in classrooms page as teacher section title', + id: 'components.TeacherCourseClassroomsDashboardLoader.teacherListTitle', + }, +}); + +const StudentsSection = () => { + return ( + + + + } + expandable={false} + /> + ); +}; +export default StudentsSection; diff --git a/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/TeachersSection/index.tsx b/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/TeachersSection/index.tsx new file mode 100644 index 0000000000..c2b3b827e1 --- /dev/null +++ b/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/TeachersSection/index.tsx @@ -0,0 +1,29 @@ +import { FormattedMessage, defineMessages } from 'react-intl'; +import { DashboardCard } from 'widgets/Dashboard/components/DashboardCard'; + +const messages = defineMessages({ + loading: { + defaultMessage: 'Loading classrooms ...', + description: 'Message displayed while loading a classrooms', + id: 'components.TeacherCourseClassroomsDashboardLoader.TeachersSection.loading', + }, + studentsListTitle: { + defaultMessage: 'Learners registered for training', + description: 'Message displayed in classrooms page as students section title', + id: 'components.TeacherCourseClassroomsDashboardLoader.TeachersSection.studentsListTitle', + }, +}); + +const TeachersSection = () => { + return ( + + + + } + expandable={false} + /> + ); +}; +export default TeachersSection; diff --git a/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/index.tsx b/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/index.tsx new file mode 100644 index 0000000000..9ca5ab4497 --- /dev/null +++ b/src/frontend/js/pages/TeacherCourseClassroomsDashboardLoader/index.tsx @@ -0,0 +1,99 @@ +import { FormattedMessage, defineMessages, useIntl } from 'react-intl'; +import { useParams } from 'react-router-dom'; + +import { capitalize } from 'lodash-es'; +import { useMemo } from 'react'; +import { DashboardLayout } from 'widgets/Dashboard/components/DashboardLayout'; +import { TeacherCourseDashboardSidebar } from 'widgets/Dashboard/components/TeacherCourseDashboardSidebar'; +import { useCourse } from 'hooks/useCourses'; +import { Spinner } from 'components/Spinner'; +import { DashboardCard } from 'widgets/Dashboard/components/DashboardCard'; +import { Icon, IconTypeEnum } from 'components/Icon'; +import { CourseRun } from 'types/Joanie'; +import SyllabusLink from 'widgets/Dashboard/components/SyllabusLink'; +import { getCourseUrl } from 'widgets/Dashboard/utils/course'; +import StudentsSection from './StudentsSection'; +import TeachersSection from './TeachersSection'; + +const messages = defineMessages({ + loading: { + defaultMessage: 'Loading classrooms ...', + description: 'Message displayed while loading a classrooms', + id: 'components.TeacherCourseClassroomsDashboardLoader.loading', + }, + syllabusLinkLabel: { + defaultMessage: 'Access the course', + description: 'Message displayed in classrooms page for the syllabus link label', + id: 'components.TeacherCourseClassroomsDashboardLoader.syllabusLinkLabel', + }, + classroomPeriod: { + defaultMessage: 'Session from {from} to {to}', + description: 'Message displayed in classrooms page for classroom period', + id: 'components.TeacherCourseClassroomsDashboardLoader.classroomPeriod', + }, +}); + +export const TeacherCourseClassroomsDashboardLoader = () => { + const intl = useIntl(); + const { courseCode, courseRunId } = useParams<{ courseCode: string; courseRunId: string }>(); + const { + item: course, + states: { fetching }, + } = useCourse(courseCode!); + const courseRun: CourseRun | undefined = useMemo( + () => course?.course_runs.find((courseCourseRun) => courseCourseRun.id === courseRunId), + [course, courseRunId], + ); + + return ( + }> + {fetching ? ( + + + + + + ) : ( + <> + + + + {capitalize(course.title)} + + + + + + + } + expandable={false} + > + {courseRun && ( + <> + + + + )} + + + + + + + + + + + + )} + + ); +}; diff --git a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.spec.tsx b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.spec.tsx index 860681480a..3c3a729d1e 100644 --- a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.spec.tsx +++ b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.spec.tsx @@ -2,20 +2,22 @@ import { render, screen } from '@testing-library/react'; import { IntlProvider } from 'react-intl'; import { CunninghamProvider } from '@openfun/cunningham-react'; import { capitalize } from 'lodash-es'; -import { CourseRunFactory } from 'utils/test/factories/joanie'; +import { CourseFactory, CourseRunFactory } from 'utils/test/factories/joanie'; import CourseRunList from '.'; describe('pages/TeacherCourseDashboardLoader/CourseRunList', () => { it('should render', () => { - const courseRuns = CourseRunFactory().many(2); + const course = CourseFactory({ + course_runs: CourseRunFactory().many(2), + }).one(); render( - + , ); - const [courseRunOne, courseRunTwo] = courseRuns; + const [courseRunOne, courseRunTwo] = course.course_runs; expect(screen.getByTitle(capitalize(courseRunOne.title))).toBeInTheDocument(); expect(screen.getByTitle(capitalize(courseRunTwo.title))).toBeInTheDocument(); expect(screen.getAllByRole('button', { name: 'go to classroom' }).length).toEqual(2); diff --git a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.tsx b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.tsx index 30dea8326c..5b8f31e625 100644 --- a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.tsx +++ b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/index.tsx @@ -1,20 +1,27 @@ import { useIntl } from 'react-intl'; import { DataList } from '@openfun/cunningham-react'; +import { useNavigate } from 'react-router-dom'; import { CourseRun } from 'types/Joanie'; +import { CourseMock } from 'api/mocks/joanie/courses'; import { buildCourseRunData } from './utils'; interface CourseRunListProps { + courseCode: CourseMock['code']; courseRuns: CourseRun[]; } -const CourseRunList = ({ courseRuns }: CourseRunListProps) => { +const CourseRunList = ({ courseCode, courseRuns }: CourseRunListProps) => { const intl = useIntl(); + const navigate = useNavigate(); const columns = ['title', 'period', 'status', 'action'].map((field: string) => ({ field })); return (
- +
); }; diff --git a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.spec.tsx b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.spec.tsx index 11828637ad..99a81b7c02 100644 --- a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.spec.tsx +++ b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.spec.tsx @@ -1,14 +1,18 @@ import { IntlProvider, createIntl } from 'react-intl'; import { render, screen } from '@testing-library/react'; import { capitalize } from 'lodash-es'; +import { NavigateFunction, To } from 'react-router-dom'; import { CourseRunFactory } from 'utils/test/factories/joanie'; +import { CourseMock } from 'api/mocks/joanie/courses'; import { buildCourseRunData, messages } from './utils'; describe('pages/TeacherCourseDashboardLoader/CourseRunList/buildCourseRunData', () => { + const navigate: NavigateFunction = (to: To | number) => { }; // eslint-disable-line + const courseCode: CourseMock['code'] = 'akeuj'; it('should return the right keys', () => { const courseRunList = CourseRunFactory().many(1); const intl = createIntl({ locale: 'en' }); - const listData = buildCourseRunData(intl, courseRunList); + const listData = buildCourseRunData(intl, navigate, courseCode, courseRunList); expect(listData.length).toBe(1); const listItem = listData[0]; @@ -17,7 +21,7 @@ describe('pages/TeacherCourseDashboardLoader/CourseRunList/buildCourseRunData', it('should contain a valid title', () => { const courseRun = CourseRunFactory().one(); const intl = createIntl({ locale: 'en' }); - const listItem = buildCourseRunData(intl, [courseRun])[0]; + const listItem = buildCourseRunData(intl, navigate, courseCode, [courseRun])[0]; render(listItem.title); expect(screen.getByText(capitalize(courseRun.title), { exact: false })).toBeInTheDocument(); @@ -26,7 +30,7 @@ describe('pages/TeacherCourseDashboardLoader/CourseRunList/buildCourseRunData', it('should contain a valid period', () => { const courseRun = CourseRunFactory().one(); const intl = createIntl({ locale: 'en' }); - const listItem = buildCourseRunData(intl, [courseRun])[0]; + const listItem = buildCourseRunData(intl, navigate, courseCode, [courseRun])[0]; render(listItem.period); expect( @@ -39,7 +43,7 @@ describe('pages/TeacherCourseDashboardLoader/CourseRunList/buildCourseRunData', it('should contain a valid status', () => { const courseRun = CourseRunFactory().one(); const intl = createIntl({ locale: 'en' }); - const listItem = buildCourseRunData(intl, [courseRun])[0]; + const listItem = buildCourseRunData(intl, navigate, courseCode, [courseRun])[0]; render(listItem.status); expect(screen.getByText(courseRun.state.text, { exact: false })).toBeInTheDocument(); @@ -47,7 +51,7 @@ describe('pages/TeacherCourseDashboardLoader/CourseRunList/buildCourseRunData', it('should contain a valid action', () => { const courseRun = CourseRunFactory().one(); const intl = createIntl({ locale: 'en' }); - const listItem = buildCourseRunData(intl, [courseRun])[0]; + const listItem = buildCourseRunData(intl, navigate, courseCode, [courseRun])[0]; render({listItem.action}); expect( diff --git a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.tsx b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.tsx index c8f6f74649..fd5632dd29 100644 --- a/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.tsx +++ b/src/frontend/js/pages/TeacherCourseDashboardLoader/CourseRunList/utils.tsx @@ -1,9 +1,13 @@ import { FormattedMessage, IntlShape, defineMessages } from 'react-intl'; import { capitalize } from 'lodash-es'; import { Button } from '@openfun/cunningham-react'; +import { NavigateFunction } from 'react-router-dom'; import { IconTypeEnum } from 'components/Icon'; import { CourseStateTextEnum, Priority } from 'types'; import { CourseRun } from 'types/Joanie'; +import { getDashboardRoutePath } from 'widgets/Dashboard/utils/dashboardRoutes'; +import { TeacherDashboardPaths } from 'widgets/Dashboard/utils/teacherRouteMessages'; +import { CourseMock } from 'api/mocks/joanie/courses'; import CourseRunListCell from './CourseRunListCell'; export const messages = defineMessages({ @@ -19,7 +23,13 @@ export const messages = defineMessages({ }, }); -export const buildCourseRunData = (intl: IntlShape, courseRuns: CourseRun[]) => { +export const buildCourseRunData = ( + intl: IntlShape, + navigate: NavigateFunction, + courseCode: CourseMock['code'], + courseRuns: CourseRun[], +) => { + const getRoutePath = getDashboardRoutePath(intl); const CourseStateIconMap: Record = { [CourseStateTextEnum.CLOSING_ON]: IconTypeEnum.MORE, [CourseStateTextEnum.STARTING_ON]: IconTypeEnum.CHECK_ROUNDED, @@ -67,7 +77,18 @@ export const buildCourseRunData = (intl: IntlShape, courseRuns: CourseRun[]) => ), action: ( - diff --git a/src/frontend/js/pages/TeacherCourseDashboardLoader/index.tsx b/src/frontend/js/pages/TeacherCourseDashboardLoader/index.tsx index 3c9e06346c..ffabe47253 100644 --- a/src/frontend/js/pages/TeacherCourseDashboardLoader/index.tsx +++ b/src/frontend/js/pages/TeacherCourseDashboardLoader/index.tsx @@ -68,7 +68,7 @@ export const TeacherCourseDashboardLoader = () => { expandable={false} fullWidth > - +
)} diff --git a/src/frontend/js/settings.ts b/src/frontend/js/settings.ts index ec56aeb528..19d168b3ec 100644 --- a/src/frontend/js/settings.ts +++ b/src/frontend/js/settings.ts @@ -16,7 +16,7 @@ export const REACT_QUERY_SETTINGS = { throttleTime: 500, }, // Cache is garbage collected after this delay - cacheTime: 24 * 60 * 60 * 1000, // 24h in ms + cacheTime: 0, // 24 * 60 * 60 * 1000, // 24h in ms // Data are considered as stale after this delay staleTimes: { default: 0, // Stale immediately @@ -35,4 +35,4 @@ export const PAYMENT_SETTINGS = { pollLimit: 30, }; -export const MOCK_SERVICE_WORKER_ENABLED = false; +export const MOCK_SERVICE_WORKER_ENABLED = true; diff --git a/src/frontend/js/widgets/Dashboard/components/DashboardLayout/_styles.scss b/src/frontend/js/widgets/Dashboard/components/DashboardLayout/_styles.scss index cc804a9b55..13cda2d2bd 100644 --- a/src/frontend/js/widgets/Dashboard/components/DashboardLayout/_styles.scss +++ b/src/frontend/js/widgets/Dashboard/components/DashboardLayout/_styles.scss @@ -30,6 +30,13 @@ $sidebar-spacing-top: $dashboard-breadcrumb-height + $dashboard-filters-height + } } + &__section { + margin-top: rem-calc(20px); + &:first-child { + margin-top: 0; + } + } + &__filters { min-height: $dashboard-filters-height; margin-top: $dashboard-breadcrumb-filters-spacing; diff --git a/src/frontend/js/widgets/Dashboard/components/DashboardLayout/index.tsx b/src/frontend/js/widgets/Dashboard/components/DashboardLayout/index.tsx index d39dfb626e..24c1becbcc 100644 --- a/src/frontend/js/widgets/Dashboard/components/DashboardLayout/index.tsx +++ b/src/frontend/js/widgets/Dashboard/components/DashboardLayout/index.tsx @@ -28,3 +28,7 @@ export const DashboardLayout = ({ children, sidebar, filters }: DashboardLayoutP ); }; + +DashboardLayout.Section = ({ children }: PropsWithChildren) => ( +
{children}
+); diff --git a/src/frontend/js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/_styles.scss b/src/frontend/js/widgets/Dashboard/components/SyllabusLink/_styles.scss similarity index 100% rename from src/frontend/js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/_styles.scss rename to src/frontend/js/widgets/Dashboard/components/SyllabusLink/_styles.scss diff --git a/src/frontend/js/widgets/Dashboard/components/SyllabusLink/index.tsx b/src/frontend/js/widgets/Dashboard/components/SyllabusLink/index.tsx new file mode 100644 index 0000000000..3d6d32ccfb --- /dev/null +++ b/src/frontend/js/widgets/Dashboard/components/SyllabusLink/index.tsx @@ -0,0 +1,15 @@ +import { PropsWithChildren } from 'react'; +import { Icon, IconTypeEnum } from 'components/Icon'; + +interface SyllabusLinkProps extends PropsWithChildren { + href: string; +} + +const SyllabusLink = ({ href, children }: SyllabusLinkProps) => ( + + + {children} + +); + +export default SyllabusLink; diff --git a/src/frontend/js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/index.tsx b/src/frontend/js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/index.tsx index cec56f485e..e3454365d6 100644 --- a/src/frontend/js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/index.tsx +++ b/src/frontend/js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/index.tsx @@ -9,7 +9,8 @@ import { } from 'widgets/Dashboard/utils/dashboardRoutes'; import { useCourse } from 'hooks/useCourses'; import { Spinner } from 'components/Spinner'; -import { Icon, IconTypeEnum } from 'components/Icon'; +import { getCourseUrl } from 'widgets/Dashboard/utils/course'; +import SyllabusLink from '../SyllabusLink'; export const messages = defineMessages({ header: { @@ -47,7 +48,8 @@ export const TeacherCourseDashboardSidebar = () => { ? [] : [ TeacherDashboardPaths.COURSE, - TeacherDashboardPaths.COURSE_CLASSROOMS, + // FIXME: we cannot build COURSE_CLASSROOMS url untile we get a courseRunId to load + // TeacherDashboardPaths.COURSE_CLASSROOMS, TeacherDashboardPaths.COURSE_RECORDS, TeacherDashboardPaths.COURSE_STUDENTS, TeacherDashboardPaths.COURSE_SETTINGS, @@ -75,12 +77,9 @@ export const TeacherCourseDashboardSidebar = () => { ) : ( - - - - - - + + + )} ); diff --git a/src/frontend/js/widgets/Dashboard/utils/course.ts b/src/frontend/js/widgets/Dashboard/utils/course.ts new file mode 100644 index 0000000000..492cda394b --- /dev/null +++ b/src/frontend/js/widgets/Dashboard/utils/course.ts @@ -0,0 +1,5 @@ +import { IntlShape } from 'react-intl'; + +export const getCourseUrl = (courseCode: string, intl: IntlShape) => { + return `/${intl.locale.split('-')[0]}/courses/${courseCode}`; +}; diff --git a/src/frontend/js/widgets/Dashboard/utils/teacherRouteMessages.tsx b/src/frontend/js/widgets/Dashboard/utils/teacherRouteMessages.tsx index e023e67820..c7086b3396 100644 --- a/src/frontend/js/widgets/Dashboard/utils/teacherRouteMessages.tsx +++ b/src/frontend/js/widgets/Dashboard/utils/teacherRouteMessages.tsx @@ -15,7 +15,7 @@ export enum TeacherDashboardPaths { ORGANIZATION_MEMBERS = '/teacher/organization/{organizationId}/members', COURSE = '/teacher/course/{courseCode}', COURSE_SETTINGS = '/teacher/course/{courseCode}/settings', - COURSE_CLASSROOMS = '/teacher/course/{courseCode}/classrooms', + COURSE_CLASSROOMS = '/teacher/course/{courseCode}/run/{courseRunId}', COURSE_STUDENTS = '/teacher/course/{courseCode}/students', COURSE_RECORDS = '/teacher/course/{courseCode}/records', } @@ -80,7 +80,7 @@ export const TEACHER_DASHBOARD_ROUTE_PATHS = defineMessages, + element: , }, { path: getRoutePath(TeacherDashboardPaths.COURSE_RECORDS, { diff --git a/src/frontend/scss/components/_index.scss b/src/frontend/scss/components/_index.scss index 417f672b5a..39f25a1eec 100644 --- a/src/frontend/scss/components/_index.scss +++ b/src/frontend/scss/components/_index.scss @@ -32,7 +32,7 @@ @import '../../js/widgets/Dashboard/components/DashboardLayout/styles'; @import '../../js/widgets/Dashboard/components/DashboardBreadcrumbs/styles'; @import '../../js/widgets/Dashboard/components/DashboardSidebar/styles'; -@import '../../js/widgets/Dashboard/components/TeacherCourseDashboardSidebar/styles'; +@import '../../js/widgets/Dashboard/components/SyllabusLink/styles'; @import '../../js/widgets/Dashboard/components/TeacherCourseSearchFilters/styles'; @import '../../js/widgets/Dashboard/components/TeacherProfileDashboardSidebar/components/OrganizationLinks/styles'; @import '../../js/widgets/LtiConsumer/styles';