diff --git a/locales/en.json b/locales/en.json index e086347..4f3afe2 100644 --- a/locales/en.json +++ b/locales/en.json @@ -217,5 +217,26 @@ "journal.pl.days.morning": "Morning", "journal.pl.days.day": "Day", "journal.pl.days.evening": "Evening", - "journal.pl.lesson.form.selectTime": "Select time" + "journal.pl.lesson.form.selectTime": "Select time", + "journal.pl.lesson.existingLessonHint": "There is already a lesson on this day", + "journal.pl.lesson.form.selectDate": "Select date", + "journal.pl.days.shortMonday": "Mo", + "journal.pl.days.shortTuesday": "Tu", + "journal.pl.days.shortWednesday": "We", + "journal.pl.days.shortThursday": "Th", + "journal.pl.days.shortFriday": "Fr", + "journal.pl.days.shortSaturday": "Sa", + "journal.pl.days.shortSunday": "Su", + "journal.pl.months.january": "January", + "journal.pl.months.february": "February", + "journal.pl.months.march": "March", + "journal.pl.months.april": "April", + "journal.pl.months.may": "May", + "journal.pl.months.june": "June", + "journal.pl.months.july": "July", + "journal.pl.months.august": "August", + "journal.pl.months.september": "September", + "journal.pl.months.october": "October", + "journal.pl.months.november": "November", + "journal.pl.months.december": "December" } \ No newline at end of file diff --git a/locales/ru.json b/locales/ru.json index 0e36fd4..4d751b9 100644 --- a/locales/ru.json +++ b/locales/ru.json @@ -214,5 +214,26 @@ "journal.pl.days.morning": "Утро", "journal.pl.days.day": "День", "journal.pl.days.evening": "Вечер", - "journal.pl.lesson.form.selectTime": "Выберите время" + "journal.pl.lesson.form.selectTime": "Выберите время", + "journal.pl.lesson.existingLessonHint": "В этот день уже есть лекция", + "journal.pl.lesson.form.selectDate": "Выберите дату", + "journal.pl.days.shortMonday": "Пн", + "journal.pl.days.shortTuesday": "Вт", + "journal.pl.days.shortWednesday": "Ср", + "journal.pl.days.shortThursday": "Чт", + "journal.pl.days.shortFriday": "Пт", + "journal.pl.days.shortSaturday": "Сб", + "journal.pl.days.shortSunday": "Вс", + "journal.pl.months.january": "Январь", + "journal.pl.months.february": "Февраль", + "journal.pl.months.march": "Март", + "journal.pl.months.april": "Апрель", + "journal.pl.months.may": "Май", + "journal.pl.months.june": "Июнь", + "journal.pl.months.july": "Июль", + "journal.pl.months.august": "Август", + "journal.pl.months.september": "Сентябрь", + "journal.pl.months.october": "Октябрь", + "journal.pl.months.november": "Ноябрь", + "journal.pl.months.december": "Декабрь" } \ No newline at end of file diff --git a/src/pages/lesson-list/components/lessons-form.tsx b/src/pages/lesson-list/components/lessons-form.tsx index 6363a4b..51970cd 100644 --- a/src/pages/lesson-list/components/lessons-form.tsx +++ b/src/pages/lesson-list/components/lessons-form.tsx @@ -1,4 +1,4 @@ -import React, { useEffect } from 'react' +import React, { useEffect, useState } from 'react' import { useForm, Controller } from 'react-hook-form' import { Box, @@ -27,9 +27,11 @@ import { useStyleConfig, Select, Wrap, - WrapItem + WrapItem, + IconButton, + Center } from '@chakra-ui/react' -import { AddIcon, CheckIcon, WarningIcon, RepeatIcon } from '@chakra-ui/icons' +import { AddIcon, CheckIcon, WarningIcon, RepeatIcon, ChevronLeftIcon, ChevronRightIcon } from '@chakra-ui/icons' import { useTranslation } from 'react-i18next' import { FaRobot } from 'react-icons/fa' import dayjs from 'dayjs' @@ -58,6 +60,7 @@ interface LessonFormProps { onSelectAiSuggestion?: (suggestion: any) => void // Обработчик выбора предложения selectedAiSuggestion?: any // Выбранное предложение onRetryAiGeneration?: () => void // Функция для повторного запуска генерации + existingLessons?: Array<{ date: string; name: string }> // Добавляем новый проп } export const LessonForm = ({ @@ -72,7 +75,8 @@ export const LessonForm = ({ isLoadingAiSuggestions = false, onSelectAiSuggestion = () => {}, selectedAiSuggestion, - onRetryAiGeneration = () => {} + onRetryAiGeneration = () => {}, + existingLessons }: LessonFormProps) => { const { t } = useTranslation() const isAiSuggested = lesson && !lesson._id && !lesson.id @@ -195,6 +199,197 @@ export const LessonForm = ({ return days[date.getDay()]; }; + // Добавляем вспомогательные функции для календаря + const getDaysInMonth = (year: number, month: number) => { + return new Date(year, month + 1, 0).getDate(); + }; + + const getFirstDayOfMonth = (year: number, month: number) => { + return new Date(year, month, 1).getDay(); + }; + + const isWeekend = (dayOfWeek: number) => { + return dayOfWeek === 0 || dayOfWeek === 6; // Воскресенье или суббота + }; + + const isSameDay = (date1: Date, date2: Date) => { + return date1.getFullYear() === date2.getFullYear() && + date1.getMonth() === date2.getMonth() && + date1.getDate() === date2.getDate(); + }; + // Компонент календаря + interface CalendarProps { + selectedDate: Date; + onSelectDate: (date: Date) => void; + existingLessons?: string[]; + } + + const Calendar: React.FC = ({ selectedDate, onSelectDate, existingLessons = [] }) => { + const { t } = useTranslation(); + const [viewDate, setViewDate] = useState(new Date()); + + // Используем короткие названия дней недели из локализации + const weekDays = [ + t('journal.pl.days.shortMonday'), + t('journal.pl.days.shortTuesday'), + t('journal.pl.days.shortWednesday'), + t('journal.pl.days.shortThursday'), + t('journal.pl.days.shortFriday'), + t('journal.pl.days.shortSaturday'), + t('journal.pl.days.shortSunday'), + ]; + + // Используем локализованные названия месяцев + const monthNames = [ + t('journal.pl.months.january'), + t('journal.pl.months.february'), + t('journal.pl.months.march'), + t('journal.pl.months.april'), + t('journal.pl.months.may'), + t('journal.pl.months.june'), + t('journal.pl.months.july'), + t('journal.pl.months.august'), + t('journal.pl.months.september'), + t('journal.pl.months.october'), + t('journal.pl.months.november'), + t('journal.pl.months.december'), + ]; + + const daysInMonth = getDaysInMonth(viewDate.getFullYear(), viewDate.getMonth()); + let firstDay = getFirstDayOfMonth(viewDate.getFullYear(), viewDate.getMonth()); + firstDay = firstDay === 0 ? 6 : firstDay - 1; // Корректируем для начала недели с понедельника + + const days = Array.from({ length: 42 }, (_, i) => { + const dayNumber = i - firstDay + 1; + if (dayNumber > 0 && dayNumber <= daysInMonth) { + const date = new Date(viewDate.getFullYear(), viewDate.getMonth(), dayNumber); + return { + date, + dayOfMonth: dayNumber, + isCurrentMonth: true, + isWeekend: isWeekend(date.getDay()), + isToday: isSameDay(date, new Date()), + isSelected: isSameDay(date, selectedDate) + }; + } + return null; + }); + + // Добавим функцию проверки наличия лекции в определенный день + const hasLessonOnDate = (date: Date) => { + return existingLessons.some(lessonDate => + isSameDay(new Date(lessonDate), date) + ); + }; + + return ( + + {t('journal.pl.lesson.form.selectDate')} + + } + size="sm" + onClick={() => { + const newDate = new Date(viewDate); + newDate.setMonth(newDate.getMonth() - 1); + setViewDate(newDate); + }} + /> + + + + + } + size="sm" + onClick={() => { + const newDate = new Date(viewDate); + newDate.setMonth(newDate.getMonth() + 1); + setViewDate(newDate); + }} + /> + + + + {weekDays.map(day => ( +
+ + {day} + +
+ ))} + {days.map((day, i) => { + const hasLesson = day?.isCurrentMonth && hasLessonOnDate(day.date); + + return ( + + ); + })} +
+
+ ); + }; + return ( @@ -225,57 +420,33 @@ export const LessonForm = ({ name="date" rules={{ required: t('journal.pl.common.required') }} render={({ field }) => { - // Разделяем текущее значение на дату и время const [currentDate = '', currentTime = '00:00:00'] = field.value.split('T'); - // Получаем часы и минуты без секунд для сравнения const currentTimeShort = currentTime.split(':').slice(0, 2).join(':'); + const selectedDate = new Date(currentDate); + + // Получаем существующие лекции из пропсов компонента + const existingLessons2 = existingLessons?.map(lesson => lesson.date) || []; return ( {t('journal.pl.lesson.form.date')} - - - {[0, 1, 2].map(daysToAdd => { - const date = new Date(); - date.setDate(date.getDate() + daysToAdd); - const formattedDate = dateToCalendarFormat(date.toISOString()).split('T')[0]; - const dayOfWeek = getDayOfWeek(date); - - return ( - - ); - })} - - - { - // При ручном изменении даты сохраняем текущее время - field.onChange(`${e.target.value}T${currentTime}:00`); - }} - type="date" - size="sm" - /> + + {/* Календарь */} + + { + const formattedDate = dateToCalendarFormat(date.toISOString()).split('T')[0]; + field.onChange(`${formattedDate}T${currentTimeShort}:00`); + }} + /> + + {/* Временные слоты */} {t('journal.pl.lesson.form.selectTime')}: - + {Object.entries(timeGroups).map(([groupName, slots]) => ( @@ -284,7 +455,6 @@ export const LessonForm = ({ {slots.map(slot => { const isSelected = currentTimeShort === slot; - return (