Добавлено новое зависимость "react-icons" версии 5.5.0. Обновлен компонент AttendanceTable: добавлены эмоджи для отображения посещаемости студентов, возможность скрытия/показа таблицы, а также улучшена логика расчета статистики посещаемости.

This commit is contained in:
Primakov Alexandr Alexandrovich 2025-03-23 09:24:37 +03:00
parent 49a26edabf
commit d5b5838e51
4 changed files with 218 additions and 61 deletions

10
package-lock.json generated
View File

@ -28,6 +28,7 @@
"react-dom": "^18.3.1", "react-dom": "^18.3.1",
"react-helmet": "^6.1.0", "react-helmet": "^6.1.0",
"react-hook-form": "^7.51.2", "react-hook-form": "^7.51.2",
"react-icons": "^5.5.0",
"react-redux": "^9.1.0", "react-redux": "^9.1.0",
"react-router-dom": "^6.22.1", "react-router-dom": "^6.22.1",
"redux": "^5.0.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": { "node_modules/react-is": {
"version": "16.13.1", "version": "16.13.1",
"resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz", "resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz",

View File

@ -44,6 +44,7 @@
"react-dom": "^18.3.1", "react-dom": "^18.3.1",
"react-helmet": "^6.1.0", "react-helmet": "^6.1.0",
"react-hook-form": "^7.51.2", "react-hook-form": "^7.51.2",
"react-icons": "^5.5.0",
"react-redux": "^9.1.0", "react-redux": "^9.1.0",
"react-router-dom": "^6.22.1", "react-router-dom": "^6.22.1",
"redux": "^5.0.1", "redux": "^5.0.1",

View File

@ -5,12 +5,10 @@ import {
Heading, Heading,
Container, Container,
useColorMode, useColorMode,
IconButton,
Flex, Flex,
Spacer, Spacer,
Badge Badge
} from '@chakra-ui/react' } from '@chakra-ui/react'
import { MoonIcon, SunIcon } from '@chakra-ui/icons'
import { PageLoader } from '../../components/page-loader/page-loader' import { PageLoader } from '../../components/page-loader/page-loader'
import { useAttendanceData, useAttendanceStats } from './hooks' import { useAttendanceData, useAttendanceStats } from './hooks'
@ -18,7 +16,7 @@ import { AttendanceTable, StatsCard } from './components'
export const Attendance = () => { export const Attendance = () => {
const { courseId } = useParams() const { courseId } = useParams()
const { colorMode, toggleColorMode } = useColorMode() const { colorMode } = useColorMode()
const data = useAttendanceData(courseId) const data = useAttendanceData(courseId)
const stats = useAttendanceStats(data) const stats = useAttendanceStats(data)

View File

@ -1,4 +1,4 @@
import React from 'react' import React, { useState } from 'react'
import { import {
Table, Table,
Thead, Thead,
@ -8,24 +8,41 @@ import {
Td, Td,
Box, Box,
useColorMode, useColorMode,
useTheme,
Button, Button,
useToast, useToast,
Flex Flex,
Collapse,
HStack,
Text,
Icon,
Tooltip,
Avatar,
AvatarBadge
} from '@chakra-ui/react' } 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 dayjs from 'dayjs'
import { sha256 } from 'js-sha256'
import { ShortText } from './ShortText' import { ShortText } from './ShortText'
import { AttendanceData } from '../hooks' 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 { interface AttendanceTableProps {
data: AttendanceData data: AttendanceData
} }
export const AttendanceTable: React.FC<AttendanceTableProps> = ({ data }) => { export const AttendanceTable: React.FC<AttendanceTableProps> = ({ data }) => {
const { colorMode } = useColorMode() const { colorMode } = useColorMode()
const theme = useTheme()
const toast = useToast() const toast = useToast()
const [showTable, setShowTable] = useState(false)
const getPresentColor = () => { const getPresentColor = () => {
return colorMode === 'dark' ? 'green.600' : 'green.100' return colorMode === 'dark' ? 'green.600' : 'green.100'
@ -35,6 +52,37 @@ export const AttendanceTable: React.FC<AttendanceTableProps> = ({ data }) => {
return colorMode === 'dark' ? 'red.800' : 'red.100' 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 = () => { const copyTableData = () => {
if (!data.attendance?.length) return if (!data.attendance?.length) return
@ -110,79 +158,179 @@ export const AttendanceTable: React.FC<AttendanceTableProps> = ({ 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) { if (!data.attendance?.length || !data.students?.length) {
return <Box>Нет данных для отображения</Box> return <Box>Нет данных для отображения</Box>
} }
return ( return (
<Box <Box
overflowX="auto"
boxShadow="md" boxShadow="md"
borderRadius="lg" borderRadius="lg"
bg={colorMode === 'dark' ? 'gray.700' : 'white'} bg={colorMode === 'dark' ? 'gray.700' : 'white'}
> >
<Flex justifyContent="flex-end" p={2}> <Flex justifyContent="space-between" p={3} alignItems="center">
<Button <Button
leftIcon={<CopyIcon />} leftIcon={<CopyIcon />}
size="sm" size="sm"
colorScheme="blue" colorScheme="blue"
onClick={copyTableData} onClick={copyTableData}
mb={2} mr={2}
> >
Копировать таблицу Копировать таблицу
</Button> </Button>
<Button
rightIcon={showTable ? <ChevronUpIcon /> : <ChevronDownIcon />}
size="sm"
variant="outline"
onClick={() => setShowTable(!showTable)}
>
{showTable ? 'Скрыть таблицу' : 'Показать таблицу'}
</Button>
</Flex> </Flex>
<Table variant="simple" size="sm">
<Thead> {/* Краткая статистика по каждому студенту с эмоджи */}
<Tr> <Box p={4} borderTop="1px" borderColor={colorMode === 'dark' ? 'gray.600' : 'gray.200'}>
{data.teachers?.map(teacher => ( <Flex flexWrap="wrap" gap={3}>
<Th key={teacher.id}>{teacher.value}</Th> {getStudentAttendance().map(({ student, name, attendedCount, totalLessons, attendance, picture }) => {
))} const emoji = getAttendanceEmoji(attendedCount, totalLessons)
<Th>Дата</Th>
<Th>Название занятия</Th> return (
{data.students.map((student) => ( <Tooltip
<Th key={student.sub}> key={student.sub}
{student.name || student.value || 'Имя не определено'} label={`${emoji.label}: ${attendedCount} из ${totalLessons} занятий (${attendance.toFixed(0)}%)`}
</Th> hasArrow
))} >
</Tr> <Box
</Thead> p={3}
<Tbody> borderRadius="md"
{data.attendance.map((lesson) => ( bg={colorMode === 'dark' ? 'gray.800' : 'gray.50'}
<Tr key={lesson.name}> boxShadow="sm"
{data.teachers?.map((teacher) => { minWidth="180px"
const wasThere = Boolean(lesson.teachers) && >
lesson.teachers.findIndex((u) => u.sub === teacher.sub) !== -1 <HStack spacing={3}>
return ( <Avatar
<Td size="md"
key={teacher.sub} src={picture}
textAlign="center" name={name}
bg={wasThere ? getPresentColor() : getAbsentColor()} >
> <AvatarBadge boxSize='2em' bg={emoji.color}>
{wasThere ? '+' : '-'} <Icon as={emoji.icon} color="white" boxSize={6} />
</Td> </AvatarBadge>
) </Avatar>
})} <Box>
<Td>{dayjs(lesson.date).format('DD.MM.YYYY')}</Td> <Text fontSize="sm" fontWeight="medium" isTruncated maxW="110px">{name}</Text>
<Td><ShortText text={lesson.name} /></Td> <Text fontSize="xs" mt={1} color={colorMode === 'dark' ? 'gray.400' : 'gray.600'}>
{attendedCount} из {totalLessons} ({attendance.toFixed(0)}%)
</Text>
</Box>
</HStack>
</Box>
</Tooltip>
)
})}
</Flex>
</Box>
{/* Полная таблица с возможностью скрытия/показа */}
<Collapse in={showTable} animateOpacity>
<Box
overflowX="auto"
p={3}
borderTop="1px"
borderColor={colorMode === 'dark' ? 'gray.600' : 'gray.200'}
>
<Table variant="simple" size="sm">
<Thead>
<Tr>
{data.teachers?.map(teacher => (
<Th key={teacher.id}>{teacher.value}</Th>
))}
<Th>Дата</Th>
<Th>Название занятия</Th>
{data.students.map((student) => (
<Th key={student.sub}>
<HStack>
<Avatar
size="xs"
src={student.picture || getGravatarURL(student.email)}
name={student.name || student.value || 'Имя не определено'}
/>
<Text>{student.name || student.value || 'Имя не определено'}</Text>
</HStack>
</Th>
))}
</Tr>
</Thead>
<Tbody>
{data.attendance.map((lesson) => (
<Tr key={lesson.name}>
{data.teachers?.map((teacher) => {
const wasThere = Boolean(lesson.teachers) &&
lesson.teachers.findIndex((u) => u.sub === teacher.sub) !== -1
return (
<Td
key={teacher.sub}
textAlign="center"
bg={wasThere ? getPresentColor() : getAbsentColor()}
>
{wasThere ? (
<Icon as={FaSmile} color="green.500" />
) : (
<Icon as={FaFrown} color="red.500" />
)}
</Td>
)
})}
<Td>{dayjs(lesson.date).format('DD.MM.YYYY')}</Td>
<Td><ShortText text={lesson.name} /></Td>
{data.students.map((st) => { {data.students.map((st) => {
const wasThere = const wasThere =
lesson.students.findIndex((u) => u.sub === st.sub) !== -1 lesson.students.findIndex((u) => u.sub === st.sub) !== -1
return ( return (
<Td <Td
key={st.sub} key={st.sub}
textAlign="center" textAlign="center"
bg={wasThere ? getPresentColor() : getAbsentColor()} bg={wasThere ? getPresentColor() : getAbsentColor()}
> >
{wasThere ? '+' : '-'} {wasThere ? (
</Td> <Icon as={FaSmile} color="green.500" />
) ) : (
})} <Icon as={FaFrown} color="red.500" />
</Tr> )}
))} </Td>
</Tbody> )
</Table> })}
</Tr>
))}
</Tbody>
</Table>
</Box>
</Collapse>
</Box> </Box>
) )
} }