diff --git a/src/pages/attendance/hooks/useAttendanceStats.ts b/src/pages/attendance/hooks/useAttendanceStats.ts index a44e012..3f0de76 100644 --- a/src/pages/attendance/hooks/useAttendanceStats.ts +++ b/src/pages/attendance/hooks/useAttendanceStats.ts @@ -1,4 +1,5 @@ import { useMemo } from 'react' +import dayjs from 'dayjs' import { AttendanceData } from './useAttendanceData' export interface AttendanceStats { @@ -27,13 +28,17 @@ export const useAttendanceStats = (data: AttendanceData): AttendanceStats => { } } - const totalLessons = data.attendance.length + const now = dayjs() + // Фильтруем лекции, оставляя только те, которые уже прошли (исключаем будущие) + const pastLessons = data.attendance.filter(lesson => dayjs(lesson.date).isBefore(now)) + + const totalLessons = pastLessons.length // Рассчитываем посещаемость для каждого студента const studentAttendance = data.students.map(student => { let attended = 0 - data.attendance.forEach(lesson => { + pastLessons.forEach(lesson => { if (lesson.students.some(s => s.sub === student.sub)) { attended++ } @@ -48,7 +53,7 @@ export const useAttendanceStats = (data: AttendanceData): AttendanceStats => { }) // Рассчитываем статистику посещаемости для каждого урока - const lessonsAttendance = data.attendance.map(lesson => { + const lessonsAttendance = pastLessons.map(lesson => { const attendedStudents = lesson.students.length const attendancePercent = data.students.length > 0 ? (attendedStudents / data.students.length) * 100 diff --git a/src/pages/course-list/course-card.tsx b/src/pages/course-list/course-card.tsx index 1d7da75..dc5f952 100644 --- a/src/pages/course-list/course-card.tsx +++ b/src/pages/course-list/course-card.tsx @@ -126,9 +126,21 @@ export const CourseCard = ({ course }: { course: Course }) => { } const studentsMap = new Map() + const now = dayjs() - // Собираем данные о всех студентах - lessonList.forEach(lesson => { + // Фильтруем только прошедшие лекции + const pastLessons = lessonList.filter(lesson => dayjs(lesson.date).isBefore(now)) + + // Если прошедших лекций нет, возвращаем пустую статистику + if (pastLessons.length === 0) { + return { + topStudents: [], + lowAttendanceStudents: [] + } + } + + // Собираем данные о всех студентах (только для прошедших лекций) + pastLessons.forEach(lesson => { lesson.students?.forEach(student => { const studentId = student.sub const current = studentsMap.get(studentId) || { @@ -147,9 +159,9 @@ export const CourseCard = ({ course }: { course: Course }) => { }) }) - // Для каждого студента установить общее количество лекций + // Для каждого студента установить общее количество лекций (только прошедших) studentsMap.forEach(student => { - student.total = lessonList.length + student.total = pastLessons.length student.percent = (student.attended / student.total) * 100 }) diff --git a/src/pages/lesson-list/components/item.tsx b/src/pages/lesson-list/components/item.tsx index cb94266..58fa5ee 100644 --- a/src/pages/lesson-list/components/item.tsx +++ b/src/pages/lesson-list/components/item.tsx @@ -11,6 +11,11 @@ import { MenuItem, MenuList, useToast, + Flex, + Text, + useColorMode, + Box, + Image, } from '@chakra-ui/react' import { EditIcon } from '@chakra-ui/icons' import { useTranslation } from 'react-i18next' @@ -32,6 +37,7 @@ type ItemProps = { setlessonToDelete(): void setEditLesson?: () => void students: unknown[] + isMobile?: boolean } export const Item: React.FC = ({ @@ -43,6 +49,7 @@ export const Item: React.FC = ({ setlessonToDelete, setEditLesson, students, + isMobile = false, }) => { const [edit, setEdit] = useState(false) const toastRef = useRef(null) @@ -50,6 +57,23 @@ export const Item: React.FC = ({ const [updateLesson, updateLessonRqst] = api.useUpdateLessonMutation() const createdLessonRef = useRef(null) const { t } = useTranslation() + const { colorMode } = useColorMode() + + // QR-код с применением фильтра инверсии для тёмной темы + const QRCodeImage = () => ( + + QR код + + ) const onSubmit = (lessonData) => { toastRef.current = toast({ @@ -104,6 +128,58 @@ export const Item: React.FC = ({ ) } + if (isMobile) { + return ( + <> + + {name} + {dayjs(date).format(groupByDate ? 'HH:mm' : 'HH:mm DD.MM.YY')} + + + + {isTeacher && ( + + + + )} + + + + {t('journal.pl.common.marked')}: {students.length} + + + {isTeacher && !edit && ( + + + + + + { + if (setEditLesson) { + setEditLesson(); + } else { + setEdit(true); + } + }} + > + {t('journal.pl.edit')} + + {t('journal.pl.delete')} + + + )} + {edit && } + + + + ) + } + + // Стандартное отображение return ( {isTeacher && ( @@ -112,7 +188,7 @@ export const Item: React.FC = ({ to={`${getNavigationValue('journal.main')}/lesson/${courseId}/${id}`} style={{ display: 'flex' }} > - + )} diff --git a/src/pages/lesson-list/components/lesson-items.tsx b/src/pages/lesson-list/components/lesson-items.tsx index e0075bc..109918f 100644 --- a/src/pages/lesson-list/components/lesson-items.tsx +++ b/src/pages/lesson-list/components/lesson-items.tsx @@ -3,6 +3,10 @@ import dayjs from 'dayjs' import { Tr, Td, + Box, + Flex, + Text, + useBreakpointValue, } from '@chakra-ui/react' import { Lesson } from '../../../__data__/model' @@ -25,24 +29,70 @@ export const LessonItems: React.FC = ({ courseId, setlessonToDelete, setEditLesson, -}) => ( - <> - {date && ( - - - {dayjs(date).format('DD MMMM YYYY')} - - - )} - {lessons.map((lesson) => ( - setlessonToDelete(lesson)} - setEditLesson={setEditLesson ? () => setEditLesson(lesson) : undefined} - courseId={courseId} - isTeacher={isTeacher} - /> - ))} - -) +}) => { + // Использование useBreakpointValue для определения мобильного отображения + const isMobile = useBreakpointValue({ base: true, md: false }) + + // Мобильное отображение + if (isMobile) { + return ( + <> + {date && ( + + {dayjs(date).format('DD MMMM YYYY')} + + )} + {lessons.map((lesson) => ( + + setlessonToDelete(lesson)} + setEditLesson={setEditLesson ? () => setEditLesson(lesson) : undefined} + courseId={courseId} + isTeacher={isTeacher} + isMobile={true} + /> + + ))} + + ) + } + + // Стандартное отображение для планшетов и больших экранов + return ( + <> + {date && ( + + + {dayjs(date).format('DD MMMM YYYY')} + + + )} + {lessons.map((lesson) => ( + setlessonToDelete(lesson)} + setEditLesson={setEditLesson ? () => setEditLesson(lesson) : undefined} + courseId={courseId} + isTeacher={isTeacher} + isMobile={false} + /> + ))} + + ) +} diff --git a/src/pages/lesson-list/lesson-list.tsx b/src/pages/lesson-list/lesson-list.tsx index 950d978..fd333c6 100644 --- a/src/pages/lesson-list/lesson-list.tsx +++ b/src/pages/lesson-list/lesson-list.tsx @@ -25,6 +25,7 @@ import { AlertDialogFooter, AlertDialogHeader, AlertDialogOverlay, + useBreakpointValue, } from '@chakra-ui/react' import { AddIcon } from '@chakra-ui/icons' import { useTranslation } from 'react-i18next' @@ -257,6 +258,9 @@ const LessonList = () => { } } + // Добавляем определение размера экрана + const isMobile = useBreakpointValue({ base: true, md: false }) + if (isLoading) { return } @@ -352,38 +356,54 @@ const LessonList = () => { /> )} - - - - - {isTeacher(user) && ( - + {isTeacher(user) && } + + + + + {lessonCalc?.map(({ data: lessons, date }) => ( + + ))} + +
- {t('journal.pl.lesson.link')} + {isMobile ? ( + + {lessonCalc?.map(({ data: lessons, date }) => ( + + ))} + + ) : ( + + + + + {isTeacher(user) && ( + + )} + - )} - - - {isTeacher(user) && } - - - - - {lessonCalc?.map(({ data: lessons, date }) => ( - - ))} - -
+ {t('journal.pl.lesson.link')} + + {groupByDate ? t('journal.pl.lesson.time') : t('journal.pl.common.date')} - {groupByDate ? t('journal.pl.lesson.time') : t('journal.pl.common.date')} - {t('journal.pl.common.name')}{t('journal.pl.lesson.action')}{t('journal.pl.common.marked')}
-
+
{t('journal.pl.common.name')}{t('journal.pl.lesson.action')}{t('journal.pl.common.marked')}
+
+ )} )