diff --git a/locales/en.json b/locales/en.json index c0b62e1..4a9c5da 100644 --- a/locales/en.json +++ b/locales/en.json @@ -54,6 +54,7 @@ "journal.pl.course.progress": "Course progress", "journal.pl.course.completedLessons": "Completed lessons", "journal.pl.course.upcomingLessons": "Upcoming lessons", + "journal.pl.course.noCourses": "No courses available", "journal.pl.lesson.created": "Lesson created", "journal.pl.lesson.successMessage": "Lesson {{name}} successfully created", @@ -155,5 +156,40 @@ "journal.pl.statistics.noUpcoming": "Not scheduled", "journal.pl.statistics.in": "in", "journal.pl.statistics.days": "days", - "journal.pl.statistics.courseProgress": "Course Progress" + "journal.pl.statistics.courseProgress": "Course Progress", + + "journal.pl.days.sunday": "Sunday", + "journal.pl.days.monday": "Monday", + "journal.pl.days.tuesday": "Tuesday", + "journal.pl.days.wednesday": "Wednesday", + "journal.pl.days.thursday": "Thursday", + "journal.pl.days.friday": "Friday", + "journal.pl.days.saturday": "Saturday", + + "journal.pl.overview.title": "Courses Overview", + "journal.pl.overview.totalCourses": "Total Courses", + "journal.pl.overview.active": "active", + "journal.pl.overview.completed": "completed", + "journal.pl.overview.totalLessons": "Total Lessons", + "journal.pl.overview.upcoming": "upcoming", + "journal.pl.overview.totalStudents": "Total Students", + "journal.pl.overview.totalTeachers": "Total Teachers", + "journal.pl.overview.perCourse": "per course", + "journal.pl.overview.attendance": "attendance", + "journal.pl.overview.noAttendanceData": "no attendance data", + "journal.pl.overview.noActiveData": "no active courses", + "journal.pl.overview.courseTrends": "Course Trends", + "journal.pl.overview.newCourses": "New Courses", + "journal.pl.overview.olderCourses": "Older Courses", + "journal.pl.overview.last3Months": "in last 3 months", + "journal.pl.overview.beyondLast3Months": "created earlier than 3 months", + "journal.pl.overview.mostActiveDay": "Most Active Day", + "journal.pl.overview.activityStats": "Activity Statistics", + "journal.pl.overview.courseCompletion": "Course Completion", + "journal.pl.overview.studentAttendance": "Student Attendance", + "journal.pl.overview.averageRate": "Average Rate", + "journal.pl.overview.lessons": "lessons", + "journal.pl.overview.topStudents": "Top Students by Attendance", + "journal.pl.overview.topAttendanceCourses": "Courses with Best Attendance", + "journal.pl.overview.new": "new" } \ No newline at end of file diff --git a/locales/ru.json b/locales/ru.json index 88a1844..ce69ac7 100644 --- a/locales/ru.json +++ b/locales/ru.json @@ -51,6 +51,7 @@ "journal.pl.course.progress": "Прогресс курса", "journal.pl.course.completedLessons": "Завершено занятий", "journal.pl.course.upcomingLessons": "Предстоящие занятия", + "journal.pl.course.noCourses": "Нет доступных курсов", "journal.pl.lesson.created": "Лекция создана", "journal.pl.lesson.successMessage": "Лекция {{name}} успешно создана", @@ -152,5 +153,40 @@ "journal.pl.statistics.noUpcoming": "Не запланировано", "journal.pl.statistics.in": "через", "journal.pl.statistics.days": "дн.", - "journal.pl.statistics.courseProgress": "Прогресс курса" + "journal.pl.statistics.courseProgress": "Прогресс курса", + + "journal.pl.days.sunday": "Воскресенье", + "journal.pl.days.monday": "Понедельник", + "journal.pl.days.tuesday": "Вторник", + "journal.pl.days.wednesday": "Среда", + "journal.pl.days.thursday": "Четверг", + "journal.pl.days.friday": "Пятница", + "journal.pl.days.saturday": "Суббота", + + "journal.pl.overview.title": "Обзор всех курсов", + "journal.pl.overview.totalCourses": "Всего курсов", + "journal.pl.overview.active": "активных", + "journal.pl.overview.completed": "завершенных", + "journal.pl.overview.totalLessons": "Всего занятий", + "journal.pl.overview.upcoming": "предстоящих", + "journal.pl.overview.totalStudents": "Всего студентов", + "journal.pl.overview.totalTeachers": "Всего преподавателей", + "journal.pl.overview.perCourse": "на курс", + "journal.pl.overview.attendance": "посещаемость", + "journal.pl.overview.noAttendanceData": "нет данных о посещаемости", + "journal.pl.overview.noActiveData": "нет активных курсов", + "journal.pl.overview.courseTrends": "Тренды курсов", + "journal.pl.overview.newCourses": "Новые курсы", + "journal.pl.overview.olderCourses": "Старые курсы", + "journal.pl.overview.last3Months": "за последние 3 месяца", + "journal.pl.overview.beyondLast3Months": "созданы ранее 3 месяцев", + "journal.pl.overview.mostActiveDay": "Самый активный день недели", + "journal.pl.overview.activityStats": "Статистика активности", + "journal.pl.overview.courseCompletion": "Завершенность курсов", + "journal.pl.overview.studentAttendance": "Посещаемость студентов", + "journal.pl.overview.averageRate": "Средний показатель", + "journal.pl.overview.lessons": "занятий", + "journal.pl.overview.topStudents": "Лучшие студенты по посещаемости", + "journal.pl.overview.topAttendanceCourses": "Курсы с лучшей посещаемостью", + "journal.pl.overview.new": "новых" } \ No newline at end of file diff --git a/src/pages/course-list/components/CoursesOverview.tsx b/src/pages/course-list/components/CoursesOverview.tsx new file mode 100644 index 0000000..ba51261 --- /dev/null +++ b/src/pages/course-list/components/CoursesOverview.tsx @@ -0,0 +1,76 @@ +import React from 'react' +import { + Box, + Heading, + SimpleGrid, + useColorModeValue, + Card +} from '@chakra-ui/react' +import { useTranslation } from 'react-i18next' + +import { Course, Lesson } from '../../../__data__/model' +import { + useStats, + StatCards, + StudentAttendanceList, + CourseAttendanceList, + ActivityStats +} from './statistics' + +interface CoursesOverviewProps { + courses: Course[] + isLoading: boolean + // Детализированные данные с уроками (если есть) + lessonsByCourse?: Record +} + +export const CoursesOverview: React.FC = ({ + courses = [], + isLoading = false, + lessonsByCourse = {} +}) => { + const { t } = useTranslation() + const bgColor = useColorModeValue('white', 'gray.700') + + // Используем хук для расчета статистики + const stats = useStats(courses, lessonsByCourse) + + // Если загрузка или нет данных, возвращаем null + if (isLoading || !courses.length) { + return null + } + + return ( + + + {t('journal.pl.overview.title')} + + + {/* Основные показатели */} + + + {/* Дополнительная статистика */} + + {/* Статистика посещаемости и топ-студенты */} + + + + {stats.topCoursesByAttendance.length > 0 && ( + <> + + + + )} + + + {/* Статистика деятельности и активности */} + + + + + + ) +} \ No newline at end of file diff --git a/src/pages/course-list/components/index.ts b/src/pages/course-list/components/index.ts index 7461583..db1826f 100644 --- a/src/pages/course-list/components/index.ts +++ b/src/pages/course-list/components/index.ts @@ -1,2 +1,3 @@ +export * from './CreateCourseForm' export * from './YearGroup' -export * from './CreateCourseForm' \ No newline at end of file +export * from './CoursesOverview' \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/ActivityStats.tsx b/src/pages/course-list/components/statistics/ActivityStats.tsx new file mode 100644 index 0000000..662b5aa --- /dev/null +++ b/src/pages/course-list/components/statistics/ActivityStats.tsx @@ -0,0 +1,93 @@ +import React from 'react' +import { + Box, + Text, + Progress, + Flex, + Divider +} from '@chakra-ui/react' +import { useTranslation } from 'react-i18next' + +import { CourseStats } from './useStats' +import { WeekdayActivityChart } from './WeekdayActivityChart' + +interface ActivityStatsProps { + stats: CourseStats +} + +export const ActivityStats: React.FC = ({ stats }) => { + const { t } = useTranslation() + + // Определяем цвет для прогресса в зависимости от значения + const getProgressColor = (value: number) => { + if (value > 80) return 'green' + if (value > 50) return 'blue' + if (value > 30) return 'yellow' + return 'red' + } + + // Вычисляем процент завершенности курсов + const completionPercentage = + stats.totalLessons > 0 + ? (stats.completedLessons / stats.totalLessons) * 100 + : 0 + + return ( + + + {t('journal.pl.overview.activityStats')} + + + + + {t('journal.pl.overview.courseCompletion')}: + + + + + {stats.completedLessons} / {stats.totalLessons} {t('journal.pl.overview.lessons')} + + + {Math.round(completionPercentage)}% + + + + + + + {t('journal.pl.overview.studentAttendance')}: + + + + + {t('journal.pl.overview.averageRate')} + + + {Math.round(stats.averageAttendance)}% + + + + + + + + + ) +} \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/CourseAttendanceList.tsx b/src/pages/course-list/components/statistics/CourseAttendanceList.tsx new file mode 100644 index 0000000..c651a41 --- /dev/null +++ b/src/pages/course-list/components/statistics/CourseAttendanceList.tsx @@ -0,0 +1,57 @@ +import React from 'react' +import { + VStack, + HStack, + Box, + Text, + Badge +} from '@chakra-ui/react' +import { useTranslation } from 'react-i18next' + +interface CourseAttendanceProps { + courses: Array<{id: string, name: string, attendanceRate: number}> +} + +export const CourseAttendanceList: React.FC = ({ courses }) => { + const { t } = useTranslation() + + // Определяем цвет для прогресса в зависимости от значения + const getProgressColor = (value: number) => { + if (value > 80) return 'green' + if (value > 50) return 'blue' + if (value > 30) return 'yellow' + return 'red' + } + + if (!courses?.length) { + return ( + + {t('journal.pl.overview.noAttendanceData')} + + ) + } + + return ( + + + {t('journal.pl.overview.topAttendanceCourses')}: + + + + {courses.map((course, index) => ( + + + #{index + 1} + + + {course.name} + + + {Math.round(course.attendanceRate)}% + + + ))} + + + ) +} \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/StatCards.tsx b/src/pages/course-list/components/statistics/StatCards.tsx new file mode 100644 index 0000000..96968e6 --- /dev/null +++ b/src/pages/course-list/components/statistics/StatCards.tsx @@ -0,0 +1,135 @@ +import React from 'react' +import { + SimpleGrid, + Stat, + StatLabel, + StatNumber, + StatHelpText, + Flex, + HStack, + Text, + Icon, + Badge +} from '@chakra-ui/react' +import { useTranslation } from 'react-i18next' +import { + FaGraduationCap, + FaChalkboardTeacher, + FaCalendarAlt, + FaUsers +} from 'react-icons/fa' + +import { CourseStats } from './useStats' + +interface StatCardsProps { + stats: CourseStats + bgColor: string +} + +export const StatCards: React.FC = ({ stats, bgColor }) => { + const { t } = useTranslation() + + return ( + + {/* Статистика по курсам */} + + + + {t('journal.pl.overview.totalCourses')} + + {stats.totalCourses} + + + + {stats.activeCourses} {t('journal.pl.overview.active')} + + {stats.recentCoursesCount > 0 && ( + + +{stats.recentCoursesCount} {t('journal.pl.overview.new')} + + )} + + + + + {/* Статистика по урокам */} + + + + {t('journal.pl.overview.totalLessons')} + + {stats.totalLessons} + + + + {stats.completedLessons} {t('journal.pl.overview.completed')} + + + {stats.upcomingLessons} {t('journal.pl.overview.upcoming')} + + + + + + {/* Статистика по студентам */} + + + + {t('journal.pl.overview.totalStudents')} + + {stats.totalStudents.size} + + + {stats.averageAttendance > 0 ? + `~${Math.round(stats.averageAttendance)}% ${t('journal.pl.overview.attendance')}` : + t('journal.pl.overview.noAttendanceData')} + + + + + {/* Статистика по преподавателям */} + + + + {t('journal.pl.overview.totalTeachers')} + + {stats.totalTeachers.size} + + + {stats.activeCourses > 0 ? + `~${(stats.totalTeachers.size / Math.max(1, stats.activeCourses)).toFixed(1)} ${t('journal.pl.overview.perCourse')}` : + t('journal.pl.overview.noActiveData')} + + + + + ) +} \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/StudentAttendanceList.tsx b/src/pages/course-list/components/statistics/StudentAttendanceList.tsx new file mode 100644 index 0000000..1c3b932 --- /dev/null +++ b/src/pages/course-list/components/statistics/StudentAttendanceList.tsx @@ -0,0 +1,82 @@ +import React from 'react' +import { + VStack, + HStack, + Box, + Text, + Progress, + Badge, + Avatar, + Tooltip +} from '@chakra-ui/react' +import { useTranslation } from 'react-i18next' +import { StarIcon } from '@chakra-ui/icons' + +import { StudentAttendance } from './useStats' + +interface StudentAttendanceListProps { + students: StudentAttendance[] + title: string +} + +export const StudentAttendanceList: React.FC = ({ + students, + title +}) => { + const { t } = useTranslation() + + // Определяем цвет для прогресса в зависимости от значения + const getProgressColor = (value: number) => { + if (value > 80) return 'green' + if (value > 50) return 'blue' + if (value > 30) return 'yellow' + return 'red' + } + + if (!students?.length) { + return ( + + {t('journal.pl.overview.noAttendanceData')} + + ) + } + + return ( + + + + {title} + + + + {students.map((student, index) => ( + + + + + + {student.name} + + + + + + {Math.round(student.percent)}% + + + ))} + + + ) +} \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/WeekdayActivityChart.tsx b/src/pages/course-list/components/statistics/WeekdayActivityChart.tsx new file mode 100644 index 0000000..3d29faa --- /dev/null +++ b/src/pages/course-list/components/statistics/WeekdayActivityChart.tsx @@ -0,0 +1,92 @@ +import React from 'react' +import { + Box, + HStack, + Text, + Badge, + Tooltip, + VStack +} from '@chakra-ui/react' +import { useTranslation } from 'react-i18next' + +interface WeekdayActivityChartProps { + weekdayActivity: number[] + mostActiveDayIndex: number +} + +export const WeekdayActivityChart: React.FC = ({ + weekdayActivity, + mostActiveDayIndex +}) => { + const { t } = useTranslation() + + // Переводим день недели в читаемый формат + const getDayOfWeekName = (dayIndex: number) => { + const days = [ + 'journal.pl.days.sunday', + 'journal.pl.days.monday', + 'journal.pl.days.tuesday', + 'journal.pl.days.wednesday', + 'journal.pl.days.thursday', + 'journal.pl.days.friday', + 'journal.pl.days.saturday' + ] + return t(days[dayIndex]) + } + + // Получаем короткое название дня недели (первая буква) + const getShortDayName = (dayIndex: number) => { + return t(`journal.pl.days.${['sunday', 'monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday'][dayIndex]}`).charAt(0) + } + + // Формируем подсказку для дня недели + const getDayTooltip = (dayIndex: number, count: number) => { + return `${t(`journal.pl.days.${['sunday', 'monday', 'tuesday', 'wednesday', 'thursday', 'friday', 'saturday'][dayIndex]}`)}: +${count} ${t('journal.pl.overview.lessons').toLowerCase()}` + } + + // Если нет данных по активности, показываем сообщение + if (!weekdayActivity.some(count => count > 0)) { + return ( + + {t('journal.pl.overview.noAttendanceData')} + + ) + } + + return ( + + + {t('journal.pl.overview.mostActiveDay')}: + + + {getDayOfWeekName(mostActiveDayIndex)} + + + {/* Визуализация активности по дням недели */} + + + {weekdayActivity.map((count, index) => ( + + + + + {getShortDayName(index)} + + + + ))} + + + + ) +} \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/index.ts b/src/pages/course-list/components/statistics/index.ts new file mode 100644 index 0000000..6c32c6e --- /dev/null +++ b/src/pages/course-list/components/statistics/index.ts @@ -0,0 +1,6 @@ +export * from './useStats' +export * from './StatCards' +export * from './StudentAttendanceList' +export * from './CourseAttendanceList' +export * from './ActivityStats' +export * from './WeekdayActivityChart' \ No newline at end of file diff --git a/src/pages/course-list/components/statistics/useStats.ts b/src/pages/course-list/components/statistics/useStats.ts new file mode 100644 index 0000000..da24081 --- /dev/null +++ b/src/pages/course-list/components/statistics/useStats.ts @@ -0,0 +1,235 @@ +import { useMemo } from 'react' +import dayjs from 'dayjs' +import { Course, Lesson } from '../../../../__data__/model' + +export interface StudentAttendance { + id: string + name: string + attended: number + total: number + percent: number + avatarUrl?: string + email?: string +} + +export interface CourseStats { + totalCourses: number + activeCourses: number + totalLessons: number + completedLessons: number + upcomingLessons: number + averageAttendance: number + totalStudents: Set + totalTeachers: Set + recentCoursesCount: number + oldCoursesCount: number + weekdayActivity: number[] + mostActiveDayIndex: number + topStudents: StudentAttendance[] + topCoursesByAttendance: Array<{id: string, name: string, attendanceRate: number}> +} + +export const useStats = ( + courses: Course[], + lessonsByCourse: Record = {} +): CourseStats => { + + return useMemo(() => { + if (!courses?.length) { + return { + totalCourses: 0, + activeCourses: 0, + totalLessons: 0, + completedLessons: 0, + upcomingLessons: 0, + averageAttendance: 0, + totalStudents: new Set(), + totalTeachers: new Set(), + recentCoursesCount: 0, + oldCoursesCount: 0, + weekdayActivity: Array(7).fill(0), + mostActiveDayIndex: 0, + topStudents: [] as StudentAttendance[], + topCoursesByAttendance: [] as {id: string, name: string, attendanceRate: number}[] + } + } + + const now = dayjs() + const threeMonthsAgo = now.subtract(3, 'month') + const weekdayActivity = Array(7).fill(0) + + // Множества для уникальных студентов и учителей + const uniqueStudents = new Set() + const uniqueTeachers = new Set() + + // Количество курсов, созданных за последние 3 месяца + const recentCourses = courses.filter(course => + dayjs(course.created).isAfter(threeMonthsAgo) + ) + + // Количество активных курсов + const activeCourses = [] + + let totalLessonsCount = 0 + let completedLessonsCount = 0 + let upcomingLessonsCount = 0 + let totalAttendances = 0 + let totalPossibleAttendances = 0 + + // Для отслеживания посещаемости студентов по всем курсам + const globalStudentsMap = new Map() + + // Статистика посещаемости по курсам + const courseAttendanceStats: {id: string, name: string, attendanceRate: number}[] = [] + + // Для каждого курса считаем статистику на основе данных об уроках + courses.forEach(course => { + // Добавляем учителей в множество + course.teachers.forEach(teacher => { + uniqueTeachers.add(teacher.sub) + }) + + // Получаем детализированные данные об уроках курса (если доступны) + const courseLessons = lessonsByCourse[course._id] || [] + + // Если у нас есть детализированные данные по урокам + if (courseLessons.length > 0) { + // Добавляем количество уроков к общему счетчику + totalLessonsCount += courseLessons.length + + // Считаем завершенные и предстоящие уроки + const completed = courseLessons.filter(lesson => dayjs(lesson.date).isBefore(now)) + const upcoming = courseLessons.filter(lesson => dayjs(lesson.date).isAfter(now)) + + completedLessonsCount += completed.length + upcomingLessonsCount += upcoming.length + + // Если у курса есть будущие занятия, считаем его активным + if (upcoming.length > 0) { + activeCourses.push(course) + } + + // Для статистики посещаемости по курсу + let courseAttendances = 0 + let coursePossibleAttendances = 0 + + // Считаем посещаемость по прошедшим занятиям + completed.forEach(lesson => { + // Добавляем статистику по дням недели + // В dayjs 0 = воскресенье, 1 = понедельник, ... 6 = суббота + // Нужно проверить формат даты урока, что это валидная дата + if (lesson.date && dayjs(lesson.date).isValid()) { + const lessonDay = dayjs(lesson.date).day() + weekdayActivity[lessonDay]++ + } + + // Добавляем студентов в глобальное множество + const lessonStudentsCount = lesson.students?.length || 0 + + // Добавляем в статистику посещаемости + courseAttendances += lessonStudentsCount + + // Обновляем счетчики общей посещаемости + totalAttendances += lessonStudentsCount + + // Собираем статистику по каждому студенту + lesson.students?.forEach(student => { + uniqueStudents.add(student.sub) + + // Добавляем или обновляем данные студента в глобальной карте + const studentId = student.sub + const currentGlobal = globalStudentsMap.get(studentId) || { + id: studentId, + name: (student.family_name && student.given_name + ? `${student.family_name} ${student.given_name}` + : student.name || student.email || student.preferred_username || student.family_name || student.given_name), + attended: 0, + total: 0, + percent: 0, + avatarUrl: student.picture, + email: student.email + } + + currentGlobal.attended += 1 + globalStudentsMap.set(studentId, currentGlobal) + }) + }) + + // Потенциальные посещения для этого курса + // (кол-во прошедших занятий * кол-во уникальных студентов) + const courseUniqueStudents = new Set() + courseLessons.forEach(lesson => { + lesson.students?.forEach(student => { + courseUniqueStudents.add(student.sub) + }) + }) + + coursePossibleAttendances = completed.length * (courseUniqueStudents.size || 1) + totalPossibleAttendances += coursePossibleAttendances + + // Добавляем статистику курса, если есть прошедшие занятия + if (completed.length > 0 && coursePossibleAttendances > 0) { + courseAttendanceStats.push({ + id: course._id, + name: course.name, + attendanceRate: (courseAttendances / coursePossibleAttendances) * 100 + }) + } + } else { + // Если у нас нет детализированных данных, считаем на основе общих данных курса + totalLessonsCount += course.lessons.length + + // Предполагаем, что курс активен + activeCourses.push(course) + } + }) + + // Отладочная информация по активности по дням недели + console.log('Weekday activity:', weekdayActivity) + + // Обрабатываем глобальную статистику посещаемости студентов + // Устанавливаем общее число занятий для каждого студента + globalStudentsMap.forEach(student => { + // Можем установить только примерную метрику, т.к. студенты могут быть на разных курсах + student.total = completedLessonsCount + student.percent = student.attended / student.total * 100 + }) + + // Находим самый активный день недели + const maxValue = Math.max(...weekdayActivity) + // Если максимальное значение = 0, то устанавливаем понедельник как самый активный день по умолчанию + const mostActiveDayIndex = maxValue > 0 ? weekdayActivity.indexOf(maxValue) : 1 + + // Вычисляем среднюю посещаемость + const averageAttendance = totalPossibleAttendances > 0 + ? (totalAttendances / totalPossibleAttendances) * 100 + : 0 + + // Топ студенты по посещаемости (по всем курсам) + const topStudents = Array.from(globalStudentsMap.values()) + .sort((a, b) => (b.percent - a.percent) || (b.attended - a.attended)) + .slice(0, 5) + + // Сортируем курсы по посещаемости + const topCoursesByAttendance = courseAttendanceStats + .sort((a, b) => b.attendanceRate - a.attendanceRate) + .slice(0, 3) + + return { + totalCourses: courses.length, + activeCourses: activeCourses.length, + totalLessons: totalLessonsCount, + completedLessons: completedLessonsCount, + upcomingLessons: upcomingLessonsCount, + averageAttendance, + totalStudents: uniqueStudents, + totalTeachers: uniqueTeachers, + recentCoursesCount: recentCourses.length, + oldCoursesCount: courses.length - recentCourses.length, + weekdayActivity, + mostActiveDayIndex, + topStudents, + topCoursesByAttendance + } + }, [courses, lessonsByCourse]) +} \ No newline at end of file diff --git a/src/pages/course-list/course-list.tsx b/src/pages/course-list/course-list.tsx index 0db44fc..2d26d1c 100644 --- a/src/pages/course-list/course-list.tsx +++ b/src/pages/course-list/course-list.tsx @@ -1,4 +1,4 @@ -import React, { useState } from 'react' +import React, { useState, useMemo, useEffect } from 'react' import { Box, Button, @@ -15,7 +15,8 @@ import { api } from '../../__data__/api/api' import { isTeacher } from '../../utils/user' import { PageLoader } from '../../components/page-loader/page-loader' import { useGroupedCourses } from './hooks' -import { CreateCourseForm, YearGroup } from './components' +import { CreateCourseForm, YearGroup, CoursesOverview } from './components' +import { Lesson } from '../../__data__/model' /** * Основной компонент списка курсов @@ -27,11 +28,48 @@ export const CoursesList = () => { const { t } = useTranslation() const { colorMode } = useColorMode() + // Создаем API запросы для получения уроков + const [getLessons] = api.useLazyLessonListQuery() + const buttonSize = useBreakpointValue({ base: 'md', md: 'lg' }) const containerPadding = useBreakpointValue({ base: '2', md: '4' }) // Используем хук для группировки курсов по годам const groupedCourses = useGroupedCourses(data?.body) + + // Создаем объект с детализированными данными для всех курсов + const [lessonsByCourse, setLessonsByCourse] = useState>({}) + + // Используем useMemo для проверки наличия данных + const courses = useMemo(() => data?.body || [], [data]) + + // Загружаем данные для каждого курса параллельно + useEffect(() => { + if (courses.length > 0 && !showForm) { + // Создаем запросы для получения данных о занятиях каждого курса + const fetchLessonsForCourses = async () => { + const lessonsData: Record = {} + + // Получаем данные курсов параллельно (по 3 курса за раз, чтобы не перегружать сервер) + for (let i = 0; i < courses.length; i += 3) { + const batch = courses.slice(i, i + 3) + const batchPromises = batch.map(async course => { + // Используем существующий API метод с Lazy Query + const response = await getLessons(course.id) + if (response.data?.body) { + lessonsData[course._id] = response.data.body + } + }) + + await Promise.all(batchPromises) + } + + setLessonsByCourse(lessonsData) + } + + fetchLessonsForCourses() + } + }, [courses, showForm, getLessons]) if (isLoading) { return @@ -61,6 +99,14 @@ export const CoursesList = () => { )} + {!showForm && ( + + )} + {Object.keys(groupedCourses).length > 0 ? ( Object.entries(groupedCourses) .sort(([yearA], [yearB]) => Number(yearB) - Number(yearA)) // Сортируем годы по убыванию