diff --git a/package-lock.json b/package-lock.json index fd5e6a8..90ba61c 100644 --- a/package-lock.json +++ b/package-lock.json @@ -28,6 +28,7 @@ "react-dom": "^18.3.1", "react-helmet": "^6.1.0", "react-hook-form": "^7.51.2", + "react-icons": "^5.5.0", "react-redux": "^9.1.0", "react-router-dom": "^6.22.1", "redux": "^5.0.1", @@ -8894,6 +8895,15 @@ } } }, + "node_modules/react-icons": { + "version": "5.5.0", + "resolved": "https://registry.npmjs.org/react-icons/-/react-icons-5.5.0.tgz", + "integrity": "sha512-MEFcXdkP3dLo8uumGI5xN3lDFNsRtrjbOEKDLD7yv76v4wpnEq2Lt2qeHaQOr34I/wPN3s3+N08WkQ+CW37Xiw==", + "license": "MIT", + "peerDependencies": { + "react": "*" + } + }, "node_modules/react-is": { "version": "16.13.1", "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", diff --git a/package.json b/package.json index ca6a10c..ae335fd 100644 --- a/package.json +++ b/package.json @@ -44,6 +44,7 @@ "react-dom": "^18.3.1", "react-helmet": "^6.1.0", "react-hook-form": "^7.51.2", + "react-icons": "^5.5.0", "react-redux": "^9.1.0", "react-router-dom": "^6.22.1", "redux": "^5.0.1", diff --git a/src/pages/attendance/attendance.tsx b/src/pages/attendance/attendance.tsx index 521d54b..54304a7 100644 --- a/src/pages/attendance/attendance.tsx +++ b/src/pages/attendance/attendance.tsx @@ -5,12 +5,10 @@ import { Heading, Container, useColorMode, - IconButton, Flex, Spacer, Badge } from '@chakra-ui/react' -import { MoonIcon, SunIcon } from '@chakra-ui/icons' import { PageLoader } from '../../components/page-loader/page-loader' import { useAttendanceData, useAttendanceStats } from './hooks' @@ -18,7 +16,7 @@ import { AttendanceTable, StatsCard } from './components' export const Attendance = () => { const { courseId } = useParams() - const { colorMode, toggleColorMode } = useColorMode() + const { colorMode } = useColorMode() const data = useAttendanceData(courseId) const stats = useAttendanceStats(data) diff --git a/src/pages/attendance/components/AttendanceTable.tsx b/src/pages/attendance/components/AttendanceTable.tsx index 26524fe..a91c273 100644 --- a/src/pages/attendance/components/AttendanceTable.tsx +++ b/src/pages/attendance/components/AttendanceTable.tsx @@ -1,4 +1,4 @@ -import React from 'react' +import React, { useState } from 'react' import { Table, Thead, @@ -8,24 +8,41 @@ import { Td, Box, useColorMode, - useTheme, Button, useToast, - Flex + Flex, + Collapse, + HStack, + Text, + Icon, + Tooltip, + Avatar, + AvatarBadge } from '@chakra-ui/react' -import { CopyIcon } from '@chakra-ui/icons' +import { CopyIcon, ChevronDownIcon, ChevronUpIcon } from '@chakra-ui/icons' +import { FaSmile, FaMeh, FaFrown, FaSadTear } from 'react-icons/fa' import dayjs from 'dayjs' +import { sha256 } from 'js-sha256' import { ShortText } from './ShortText' import { AttendanceData } from '../hooks' +// Функция для получения URL аватарки через Gravatar +function getGravatarURL(email) { + if (!email) return undefined + const address = String(email).trim().toLowerCase() + const hash = sha256(address) + + return `https://www.gravatar.com/avatar/${hash}?d=robohash` +} + interface AttendanceTableProps { data: AttendanceData } export const AttendanceTable: React.FC = ({ data }) => { const { colorMode } = useColorMode() - const theme = useTheme() const toast = useToast() + const [showTable, setShowTable] = useState(false) const getPresentColor = () => { return colorMode === 'dark' ? 'green.600' : 'green.100' @@ -35,6 +52,37 @@ export const AttendanceTable: React.FC = ({ data }) => { return colorMode === 'dark' ? 'red.800' : 'red.100' } + // Получаем эмоджи на основе посещаемости + const getAttendanceEmoji = (attendedCount: number, totalLessons: number) => { + const attendanceRate = totalLessons > 0 ? attendedCount / totalLessons : 0 + + if (attendanceRate >= 0.9) { + return { + icon: FaSmile, + color: 'green.500', + label: 'Отличная посещаемость' + } + } else if (attendanceRate >= 0.75) { + return { + icon: FaMeh, + color: 'blue.400', + label: 'Хорошая посещаемость' + } + } else if (attendanceRate >= 0.5) { + return { + icon: FaFrown, + color: 'orange.400', + label: 'Низкая посещаемость' + } + } else { + return { + icon: FaSadTear, + color: 'red.500', + label: 'Критически низкая посещаемость' + } + } + } + // Функция для копирования данных таблицы без сокращений const copyTableData = () => { if (!data.attendance?.length) return @@ -110,79 +158,179 @@ export const AttendanceTable: React.FC = ({ data }) => { }) } + // Расчет статистики посещаемости для каждого студента + const getStudentAttendance = () => { + const totalLessons = data.attendance.length + return data.students.map(student => { + let attendedCount = 0 + data.attendance.forEach(lesson => { + if (lesson.students.findIndex(s => s.sub === student.sub) !== -1) { + attendedCount++ + } + }) + + return { + student, + name: student.name || student.value || 'Имя не определено', + email: student.email, + picture: student.picture || getGravatarURL(student.email), + attendedCount, + totalLessons, + attendance: totalLessons > 0 ? (attendedCount / totalLessons) * 100 : 0 + } + }) + } + if (!data.attendance?.length || !data.students?.length) { return Нет данных для отображения } return ( - + + + - - - - {data.teachers?.map(teacher => ( - - ))} - - - {data.students.map((student) => ( - - ))} - - - - {data.attendance.map((lesson) => ( - - {data.teachers?.map((teacher) => { - const wasThere = Boolean(lesson.teachers) && - lesson.teachers.findIndex((u) => u.sub === teacher.sub) !== -1 - return ( - - ) - })} - - + + {/* Краткая статистика по каждому студенту с эмоджи */} + + + {getStudentAttendance().map(({ student, name, attendedCount, totalLessons, attendance, picture }) => { + const emoji = getAttendanceEmoji(attendedCount, totalLessons) + + return ( + + + + + + + + + + {name} + + {attendedCount} из {totalLessons} ({attendance.toFixed(0)}%) + + + + + + ) + })} + + + + {/* Полная таблица с возможностью скрытия/показа */} + + +
{teacher.value}ДатаНазвание занятия - {student.name || student.value || 'Имя не определено'} -
- {wasThere ? '+' : '-'} - {dayjs(lesson.date).format('DD.MM.YYYY')}
+ + + {data.teachers?.map(teacher => ( + + ))} + + + {data.students.map((student) => ( + + ))} + + + + {data.attendance.map((lesson) => ( + + {data.teachers?.map((teacher) => { + const wasThere = Boolean(lesson.teachers) && + lesson.teachers.findIndex((u) => u.sub === teacher.sub) !== -1 + return ( + + ) + })} + + - {data.students.map((st) => { - const wasThere = - lesson.students.findIndex((u) => u.sub === st.sub) !== -1 - return ( - - ) - })} - - ))} - -
{teacher.value}ДатаНазвание занятия + + + {student.name || student.value || 'Имя не определено'} + +
+ {wasThere ? ( + + ) : ( + + )} + {dayjs(lesson.date).format('DD.MM.YYYY')} - {wasThere ? '+' : '-'} -
+ {data.students.map((st) => { + const wasThere = + lesson.students.findIndex((u) => u.sub === st.sub) !== -1 + return ( + + {wasThere ? ( + + ) : ( + + )} + + ) + })} + + ))} + + +
+ ) } \ No newline at end of file