Added parser function

This commit is contained in:
VityaSchel
2023-10-01 20:18:37 +04:00
parent 0bb5f022e9
commit b6ea0217c3
19 changed files with 1664 additions and 153 deletions

361
src/app/agregator/mock.js Normal file
View File

@@ -0,0 +1,361 @@
export const content = `<html><head><title>Расписание занятий</title>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<link rel="StyleSheet" type="text/css" href="lib/css/wv_20230927.css">
</head>
<body bgcolor="#FFFFFF" text="#000000" link="#000000" vlink="#000000" alink="#ABB8DA" leftmargin="0" topmargin="0" marginwidth="0" marginheight="0">
<table border="0" cellpadding="0" cellspacing="0" width="100%" bgcolor="3481A6"><tbody><tr><td height="30" align="right" bgcolor="ffffff" colspan="2" background="lib/img/itf_ht/mn_top_bg.gif"><a href="?mnt=2"></a><a href="?mnt=3"></a><table border="0" cellpadding="0" cellspacing="0"><tbody><tr><td background="lib/img/itf_ht/mn_top_bg.gif" align="right" bgcolor="ffffff" colspan="1"><a href="?mn=2" class="t_on">&nbsp;Расписание занятий&nbsp;</a></td><td background="lib/img/itf_ht/mn_top_bg_btn.gif" align="right" bgcolor="ffffff"><img src="lib/img/itf_ht/mn_top_btn_r.gif"></td><td background="lib/img/itf_ht/mn_top_bg_btn.gif" align="right" bgcolor="ffffff"><a href="?mn=3" class="t_on">Расписание преподавателей</a></td><td background="lib/img/itf_ht/mn_top_bg_btn.gif" align="right" bgcolor="ffffff"><img src="lib/img/itf_ht/mn_top_btn_l.gif"><a href="?mnt=4"></a></td><td background="lib/img/itf_ht/mn_top_bg_btn.gif" align="right" bgcolor="ffffff"><img src="lib/img/itf_ht/mn_top_btn_r.gif"></td><td background="lib/img/itf_ht/mn_top_bg_btn.gif" align="right" bgcolor="ffffff"><a href="?mn=4" class="t_on"></a><a href="https://lk.ks.psuti.ru/std">Личный кабинет студента</a></td><td background="lib/img/itf_ht/mn_top_bg_btn.gif" align="right" bgcolor="ffffff"><img src="lib/img/itf_ht/mn_top_btn_l.gif"></td></tr></tbody></table></td></tr></tbody></table><center><a href="https://lk.ks.psuti.ru/lib/doc/lk_ks_psuti_manual.pdf" target="blank" class="t_green_14"><b><u><i>Инструкция: "Доступ в личный кабинет студента"</i></u></b></a><br>
<a href="https://disk.yandex.ru/d/ioxTvdQXkPpU5w" target="_blank"><u><i>Расписание занятий на 1 семестр 2023-2024 уч. год (версия для печати)</i></u></a><br>
<a href="https://disk.yandex.ru/i/WyzJ4DVeq5e_Vg" target="_blank"><u><i>График учебного процесса на 2023/2024 уч. год (версия для печати)</i></u></a><br>
</center><br>
<table border="0" cellpadding="0" cellspacing="0" width="100%" bgcolor="3481A6">
<tbody><tr><td height="20" bgcolor="dddddd" align="center" colspan="7"><h7>Расписание занятий</h7></td></tr>
<tr><td bgcolor="3481A6" align="center" colspan="7" background="lib/img/itf_ht/hr_b_01.gif"><img src="lib/img/itf_ht/hr_b_01.gif"></td></tr><tr><td height="10" bgcolor="ffffff" colspan="7" align="center">с 25.09.2023 по 01.10.2023
</td></tr>
<tr><td bgcolor="3481A6" align="center" colspan="7" background="lib/img/itf_ht/bg_h2_03.gif"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=194"><img src="lib/img/itf_ht/frw_l.gif"></a></td><td height="20" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=194"><h3>предыдущая неделя</h3></a></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif">&nbsp;&nbsp;&nbsp;&nbsp;</td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><img src="lib/img/itf_ht/rbl04.gif"></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif">&nbsp;&nbsp;&nbsp;&nbsp;</td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><img src="lib/img/itf_ht/rbl04.gif"></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif">&nbsp;&nbsp;&nbsp;&nbsp;</td><td height="20" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=196"><h3>следующая неделя</h3></a></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=196"><img src="lib/img/itf_ht/frw_r.gif"></a></td></tr></tbody></table></td></tr>
</tbody></table><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td height="20" bgcolor="ffffff" colspan="7">&nbsp;&nbsp;&nbsp;<b>ПС-7</b></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
<tr><td height="20" bgcolor="C0D8E3" colspan="7"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td bgcolor="3481A6"><h3>Понедельник 25.09.2023 / 4 неделя</h3></td><td bgcolor="C0D8E3"><img src="lib/img/itf_ht/days_bg.gif"></td></tr></tbody></table></td></tr>
<tr align="center"><td bgcolor="ffffff"><b>№ пары</b></td>
<td bgcolor="ffffff"><b>Время занятий</b></td>
<td bgcolor="ffffff"><b>Способ</b></td>
<td bgcolor="ffffff"><b>Дисциплина, преподаватель</b></td>
<td bgcolor="ffffff"><b>Тема занятия</b></td>
<td bgcolor="ffffff"><b>Ресурс</b></td>
<td bgcolor="ffffff"><b>Задание для выполнения</b></td>
</tr>
<tr align="center"><td bgcolor="ffffff">1</td>
<td bgcolor="ffffff">08:00 09:30
</td>
<td bgcolor="ffffff">Самостоятельная работа</td>
<td bgcolor="ffffff">Физика<br>Кусаева Зарина Владимировна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff">Силы в природе.</td>
<td bgcolor="ffffff"><b><a href="https://cloud.mail.ru/public/eDk2/eV51SaUoU" target="blank">Лекция Задачи</a></b><br></td>
<td bgcolor="ffffff">Сделать конспект по теме: выписать определения, формулы. Самостоятельно решить задачи.<br></td></tr>
<tr align="center"><td bgcolor="ffffff">2</td>
<td bgcolor="ffffff">09:40 11:10
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Математика<br>Амукова Светлана Николаевна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">3</td>
<td bgcolor="ffffff">11:40 13:10
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Математика<br>Амукова Светлана Николаевна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">4</td>
<td bgcolor="ffffff">13:20 14:50
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Химия<br>Тарасова Таисия Евгеньевна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
<tr><td height="20" bgcolor="C0D8E3" colspan="7"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td bgcolor="3481A6"><h3>Вторник 26.09.2023 / 4 неделя</h3></td><td bgcolor="C0D8E3"><img src="lib/img/itf_ht/days_bg.gif"></td></tr></tbody></table></td></tr>
<tr align="center"><td bgcolor="ffffff"><b>№ пары</b></td>
<td bgcolor="ffffff"><b>Время занятий</b></td>
<td bgcolor="ffffff"><b>Способ</b></td>
<td bgcolor="ffffff"><b>Дисциплина, преподаватель</b></td>
<td bgcolor="ffffff"><b>Тема занятия</b></td>
<td bgcolor="ffffff"><b>Ресурс</b></td>
<td bgcolor="ffffff"><b>Задание для выполнения</b></td>
</tr>
<tr align="center"><td bgcolor="ffffff">3</td>
<td bgcolor="ffffff">11:40 13:10
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">История<br>Арефьев Андрей Андреевич<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 401</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr align="center"><td bgcolor="ffffbb">4</td>
<td bgcolor="ffffbb">13:20 14:50
<br><a class="t_zm">дистанционно</a></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb">Физическая культура<br> </td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">5</td>
<td bgcolor="ffffff">15:10 16:40
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Информатика<br>Ларионова Софья Николаевна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 208</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
<tr><td height="20" bgcolor="C0D8E3" colspan="7"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td bgcolor="3481A6"><h3>Среда 27.09.2023 / 4 неделя</h3></td><td bgcolor="C0D8E3"><img src="lib/img/itf_ht/days_bg.gif"></td></tr></tbody></table></td></tr>
<tr align="center"><td bgcolor="ffffff"><b>№ пары</b></td>
<td bgcolor="ffffff"><b>Время занятий</b></td>
<td bgcolor="ffffff"><b>Способ</b></td>
<td bgcolor="ffffff"><b>Дисциплина, преподаватель</b></td>
<td bgcolor="ffffff"><b>Тема занятия</b></td>
<td bgcolor="ffffff"><b>Ресурс</b></td>
<td bgcolor="ffffff"><b>Задание для выполнения</b></td>
</tr>
<tr align="center"><td bgcolor="ffffff">3</td>
<td bgcolor="ffffff">11:40 13:10
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Физика<br>Кусаева Зарина Владимировна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 310</font></td>
<td bgcolor="ffffff">Импульс. Закон сохранения импульса.</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Проверка конспекта и задач<br></td></tr>
<tr align="center"><td bgcolor="ffffbb">4</td>
<td bgcolor="ffffbb">13:20 14:50
<br><a class="t_zm">перенос с 02.10.23</a></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb">Физика<br>Кусаева Зарина Владимировна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 310</font></td>
<td bgcolor="ffffbb">Закон сохранения энергии</td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">5</td>
<td bgcolor="ffffff">15:10 16:40
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Иностранный язык<br>Карпеева Александра Сергеевна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 234</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr align="center"><td bgcolor="ffffbb">6</td>
<td bgcolor="ffffbb">16:50 18:20
<br><a class="t_zm">дистанционно</a></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb">География<br> </td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"><br></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
<tr><td height="20" bgcolor="C0D8E3" colspan="7"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td bgcolor="3481A6"><h3>Четверг 28.09.2023 / 4 неделя</h3></td><td bgcolor="C0D8E3"><img src="lib/img/itf_ht/days_bg.gif"></td></tr></tbody></table></td></tr>
<tr align="center"><td bgcolor="ffffff"><b>№ пары</b></td>
<td bgcolor="ffffff"><b>Время занятий</b></td>
<td bgcolor="ffffff"><b>Способ</b></td>
<td bgcolor="ffffff"><b>Дисциплина, преподаватель</b></td>
<td bgcolor="ffffff"><b>Тема занятия</b></td>
<td bgcolor="ffffff"><b>Ресурс</b></td>
<td bgcolor="ffffff"><b>Задание для выполнения</b></td>
</tr>
<tr align="center"><td bgcolor="ffffbb">2</td>
<td bgcolor="ffffbb">09:40 11:10
<br><a class="t_zm">дистанционно</a></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb">География<br> </td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"></td>
<td bgcolor="ffffbb"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">3</td>
<td bgcolor="ffffff">11:40 13:10
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Математика<br>Амукова Светлана Николаевна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">4</td>
<td bgcolor="ffffff">13:20 14:50
</td>
<td bgcolor="ffffff">Очно</td>
<td bgcolor="ffffff">Русский язык<br>Назарова Елена Федоровна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff">Признаки заимствованного слова. Практическая работа 2</td>
<td bgcolor="ffffff"><b><a href="https://cloud.mail.ru/public/7mwL/Ui6a7ftsP" target="blank">Практическая работа 2</a></b><br></td>
<td bgcolor="ffffff"><br></td></tr>
<tr align="center"><td bgcolor="ffffff">5</td>
<td bgcolor="ffffff">15:10 16:40
</td>
<td bgcolor="ffffff">Очно</td>
<td bgcolor="ffffff">Литература<br>Назарова Елена Федоровна<font class="t_ur2"><br>Л. Толстого, 23<br>Кабинет: 410-2</font></td>
<td bgcolor="ffffff">Роман И. А. Гончарова "Обломов"</td>
<td bgcolor="ffffff"><b><a href="https://cloud.mail.ru/public/5f3Y/vYGed416U" target="blank">лекция</a></b><br></td>
<td bgcolor="ffffff">пересказ лекции, ответить на вопросы викторины<br></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
<tr><td height="20" bgcolor="C0D8E3" colspan="7"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td bgcolor="3481A6"><h3>Пятница 29.09.2023 / 5 неделя</h3></td><td bgcolor="C0D8E3"><img src="lib/img/itf_ht/days_bg.gif"></td></tr></tbody></table></td></tr>
<tr align="center"><td bgcolor="ffffff"><b>№ пары</b></td>
<td bgcolor="ffffff"><b>Время занятий</b></td>
<td bgcolor="ffffff"><b>Способ</b></td>
<td bgcolor="ffffff"><b>Дисциплина, преподаватель</b></td>
<td bgcolor="ffffff"><b>Тема занятия</b></td>
<td bgcolor="ffffff"><b>Ресурс</b></td>
<td bgcolor="ffffff"><b>Задание для выполнения</b></td>
</tr>
<tr align="center"><td bgcolor="ffffff">2</td>
<td bgcolor="ffffff">09:40 11:10
</td>
<td bgcolor="ffffff">Очно</td>
<td bgcolor="ffffff">Русский язык<br>Назарова Елена Федоровна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 324</font></td>
<td bgcolor="ffffff">Язык как система</td>
<td bgcolor="ffffff"><b><a href="https://cloud.mail.ru/public/EBc5/MYaGgXqKe" target="blank">презентация</a></b><br></td>
<td bgcolor="ffffff">изучить презентацию, выучить определения<br></td></tr>
<tr align="center"><td bgcolor="ffffff">3</td>
<td bgcolor="ffffff">11:40 13:10
</td>
<td bgcolor="ffffff">Очно</td>
<td bgcolor="ffffff">Литература<br>Назарова Елена Федоровна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 321</font></td>
<td bgcolor="ffffff">"Илья Ильич Обломов как временной тип и одна из граней национального характера"</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">написать сочинение на тему "Что во мне есть от Обломова"<br></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
<tr><td height="20" bgcolor="C0D8E3" colspan="7"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td bgcolor="3481A6"><h3>Суббота 30.09.2023 / 5 неделя</h3></td><td bgcolor="C0D8E3"><img src="lib/img/itf_ht/days_bg.gif"></td></tr></tbody></table></td></tr>
<tr align="center"><td bgcolor="ffffff"><b>№ пары</b></td>
<td bgcolor="ffffff"><b>Время занятий</b></td>
<td bgcolor="ffffff"><b>Способ</b></td>
<td bgcolor="ffffff"><b>Дисциплина, преподаватель</b></td>
<td bgcolor="ffffff"><b>Тема занятия</b></td>
<td bgcolor="ffffff"><b>Ресурс</b></td>
<td bgcolor="ffffff"><b>Задание для выполнения</b></td>
</tr>
<tr align="center"><td bgcolor="ffffff">1</td>
<td bgcolor="ffffff">08:00 09:30
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">Основы безопасности жизнедеятельности<br>Корнилова Светлана Александровна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 110</font></td>
<td bgcolor="ffffff">Пр№1 Обеспечение личной безопасности на дорогах</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">оформить отчет<br></td></tr>
<tr align="center"><td bgcolor="ffffff">2</td>
<td bgcolor="ffffff">09:40 11:10
</td>
<td bgcolor="ffffff">Очно</td>
<td bgcolor="ffffff">Литература<br>Назарова Елена Федоровна<font class="t_green_10"><br>Московское шоссе, 120<br>Кабинет: 321</font></td>
<td bgcolor="ffffff">Новый герой, «отрицающий все», в романе И. С. Тургенева (1818- 1883) «Отцы и дети»</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff">читать роман "Отцы и дети"<br></td></tr>
<tr align="center"><td bgcolor="ffffff">3</td>
<td bgcolor="ffffff">11:30 13:00
</td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><i>Свободное время</i><br> </td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"></td>
<td bgcolor="ffffff"><br></td></tr>
<tr><td height="5" bgcolor="ffffff" colspan="7"></td></tr>
</tbody></table><table border="0" cellpadding="0" cellspacing="0" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="3481A6" align="center" colspan="7" background="lib/img/itf_ht/bg_h2_03.gif"><table border="0" cellpadding="0" cellspacing="0" bgcolor="ffffff">
<tbody><tr><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=194"><img src="lib/img/itf_ht/frw_l.gif"></a></td><td height="20" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=194"><h3>предыдущая неделя</h3></a></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif">&nbsp;&nbsp;&nbsp;&nbsp;</td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><img src="lib/img/itf_ht/rbl04.gif"></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif">&nbsp;&nbsp;&nbsp;&nbsp;</td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><img src="lib/img/itf_ht/rbl04.gif"></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif">&nbsp;&nbsp;&nbsp;&nbsp;</td><td height="20" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=196"><h3>следующая неделя</h3></a></td><td width="1" valign="bottom" background="lib/img/itf_ht/bg_h2_03.gif"><a href="?mn=2&amp;obj=146&amp;wk=196"><img src="lib/img/itf_ht/frw_r.gif"></a></td></tr></tbody></table></td></tr>
</tbody></table><br><table border="0" cellpadding="0" cellspacing="0" width="100%" bgcolor="ffffff">
<tbody><tr><td height="20" bgcolor="dddddd" align="center" colspan="8"><h7>Дневное отделение:</h7></td></tr>
<tr><td bgcolor="3481A6" align="center" colspan="8" background="lib/img/itf_ht/hr_b_01.gif"><img src="lib/img/itf_ht/hr_b_01.gif"></td></tr><tr><td height="20" bgcolor="3481A6" align="center" colspan="8" background="lib/img/itf_ht/bg_h2_03.gif"><h3>Выберите группу:</h3></td></tr>
<tr><td bgcolor="3481A6" align="center" colspan="8" background="lib/img/itf_ht/hr_b_03.gif"><img src="lib/img/itf_ht/hr_b_03.gif"></td></tr><tr><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td></tr><tr><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td><td height="20" width="25%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Первый курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td><td height="20" width="25%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Второй курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td><td height="20" width="25%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Третий курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td><td height="20" width="25%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Четвертый курс:</h3></td></tr>
<tr><td width="1" valign="bottom"></td><td width="25%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=144"><b>ИБ-5</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=145"><b>ИБ-6</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=152"><b>ИКС-6</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=153"><b>ИКС-7к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=150"><b>ИСПВ-6</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=151"><b>ИСПВ-7</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=161"><b>ИСПВ-8к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=154"><b>ИСПИ-5</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=155"><b>ИСПИ-6к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=156"><b>ИСПП-22</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=157"><b>ИСПП-23</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=158"><b>ИСПП-24к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=159"><b>ИСПП-25к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=160"><b>ИСПП-26к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=163"><b>ИСПП-27к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=146"><b>ПС-7</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=162"><b>СБ-1к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=147"><b>ССА-10</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=148"><b>ССА-11</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="25%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=123"><b>ИБ-3</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=138"><b>ИБ-4к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=124"><b>ИКС-4</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=126"><b>ИКС-5к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=127"><b>ИСПВ-4</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=128"><b>ИСПВ-5к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=129"><b>ИСПИ-3</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=130"><b>ИСПИ-4к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=131"><b>ИСПП-16</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=132"><b>ИСПП-17</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=133"><b>ИСПП-18к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=134"><b>ИСПП-19к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=140"><b>ИСПП-20к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=141"><b>ИСПП-21к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=135"><b>ПС-6</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=136"><b>ССА-7</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=137"><b>ССА-8к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=149"><b>ССА-9к</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="25%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=114"><b>ИБ-1к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=115"><b>ИБ-2к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=117"><b>ИКС-2</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=121"><b>ИКС-3к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=111"><b>ИСПВ-1</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=112"><b>ИСПВ-2к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=113"><b>ИСПВ-3к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=109"><b>ИСПИ-1</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=110"><b>ИСПИ-2к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=105"><b>ИСПП-11</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=106"><b>ИСПП-12</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=107"><b>ИСПП-13к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=108"><b>ИСПП-14к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=122"><b>ИСПП-15к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=116"><b>ПС-5</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=102"><b>ССА-4</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=103"><b>ССА-5</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=104"><b>ССА-6к</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="25%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=101"><b>ИКС-1</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=67"><b>ИС-21</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=68"><b>ИС-22</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=70"><b>ИСПП-5</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=71"><b>ИСПП-6</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=72"><b>ИСПП-7к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=73"><b>ИСПП-8к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=74"><b>ИСПП-9к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=75"><b>МТС-78</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=76"><b>ПКС-33</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=77"><b>ПКС-34</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=78"><b>ПКС-35к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=80"><b>СК-69</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=81"><b>ССА-1к</b></a> </td></tr>
<tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=100"><b>ССА-3к</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
</tr>
</tbody></table><br><table border="0" cellpadding="0" cellspacing="0" width="100%" bgcolor="ffffff">
<tbody><tr><td height="20" bgcolor="dddddd" align="center" colspan="12"><h7>Заочное отделение:</h7></td></tr>
<tr><td bgcolor="3481A6" align="center" colspan="12" background="lib/img/itf_ht/hr_b_01.gif"><img src="lib/img/itf_ht/hr_b_01.gif"></td></tr><tr><td height="20" bgcolor="3481A6" align="center" colspan="12" background="lib/img/itf_ht/bg_h2_03.gif"><h3>Выберите группу:</h3></td></tr>
<tr><td bgcolor="3481A6" align="center" colspan="12" background="lib/img/itf_ht/hr_b_03.gif"><img src="lib/img/itf_ht/hr_b_03.gif"></td></tr><tr><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td><td bgcolor="3481A6" valign="bottom" colspan="2"><img src="lib/img/itf_ht/rtl02.gif"></td></tr><tr><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td><td height="20" width="20%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Первый курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td>
<td height="20" width="20%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Второй курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td>
<td height="20" width="20%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Третий курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td>
<td height="20" width="20%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Четвертый курс:</h3></td><td width="1" valign="bottom"><img src="lib/img/itf_ht/rbl04.gif"></td>
<td height="20" width="20%" bgcolor="3481A6" align="center" background="lib/img/itf_ht/bg_h2_02.gif"><h3>Пятый курс:</h3></td></tr>
<tr><td width="1" valign="bottom"></td><td width="20%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=166"><b>(з/о)иксс-23</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=165"><b>(з/о)ПС-23к</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="20%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=142"><b>(з/о)ИКСС-22к</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="20%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=118"><b>(з/о)ИКСС-21к</b></a> </td></tr>
<tr><td bgcolor="dddddd" align="center"><a href="?mn=2&amp;obj=143"><b>(з/о)ПС-22к</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="20%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=87"><b>(з/о)ПКС-20</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
<td width="1" valign="bottom"></td><td width="20%" bgcolor="ffffff" align="center" valign="top"><table border="0" cellpadding="0" cellspacing="0" width="100%"><tbody><tr><td><table border="0" cellpadding="1" cellspacing="1" width="100%" bgcolor="3481A6">
<tbody><tr><td bgcolor="eeeeee" align="center"><a href="?mn=2&amp;obj=46"><b>(з/о)ПКС-19</b></a> </td></tr>
</tbody></table></td></tr><tr><td align="right" colspan="8" background="lib/img/itf_ht/hr_b_04.gif"><img src="lib/img/itf_ht/hr_b_05.gif"></td></tr></tbody></table></td>
</tr>
</tbody></table><br>
</body></html>`

View File

@@ -0,0 +1,21 @@
import { Day } from '@/shared/model/day'
import { parsePage } from '@/app/parser/schedule'
import contentTypeParser from 'content-type'
import { parse } from 'node-html-parser'
import { content as mockContent } from './mock'
// ПС-7: 146
export async function getSchedule(groupID: number): Promise<Day[]> {
// const page = await fetch(`https://lk.ks.psuti.ru/?mn=2&obj=${groupID}`)
const page = { text: async () => mockContent, status: 200, headers: { get: (s: string) => s && 'text/html' } }
const content = await page.text()
const contentType = page.headers.get('content-type')
if (page.status === 200 && contentType && contentTypeParser.parse(contentType).type === 'text/html') {
const root = parse(content)
return parsePage(root)
} else {
console.error(page.status, contentType)
console.error(content.length > 500 ? content.slice(0, 500 - 3) + '...' : content)
throw new Error('Error while fetching lk.ks.psuti.ru')
}
}

View File

@@ -0,0 +1,97 @@
import { Day } from '@/shared/model/day'
import { Lesson } from '@/shared/model/lesson'
import { HTMLElement } from 'node-html-parser'
const dayTitleParser = (text: string) => {
const [dateString, week] = text.trim().split(' / ')
const weekNumber = Number(week.trim().match(/^(\d+) неделя$/)![1])
const [, day, month, year] = dateString.trim().match(/^[а-яА-Я]+ (\d{1,2})\.(\d{1,2})\.(\d{4})$/)!
const date = new Date(Number(year), Number(month) - 1, Number(day), 12)
return { date, weekNumber }
}
const parseLesson = (row: HTMLElement): Lesson => {
const cells = row.querySelectorAll(':scope > td')
const isChange = cells.every(td => td.getAttribute('bgcolor') === 'ffffbb')
const timeCell = cells[1].childNodes
const [startTime, endTime] = timeCell[0].textContent.trim().split(' ')
const time: Lesson['time'] = {
start: startTime,
end: endTime
}
if (timeCell[2]) {
time.hint = timeCell[2].textContent.trim()
}
const subject = ''
const teacher = ''
const place: Lesson['place'] = {
address: '1',
classroom: 1
}
const topic = cells[4].textContent.trim()
const resources: Lesson['resources'] = []
// {
// type: 'link'
// title: string
// url: string
// } []
return {
isChange,
time,
type: cells[2].textContent.trim(),
subject,
teacher,
place,
...(topic && { topic }),
resources,
homework: cells[6].textContent.trim()
}
}
export function parsePage(document: HTMLElement): Day[] {
const tables = Array.from(document.querySelectorAll('body > table'))
const table = tables.find(table => table.querySelector(':scope > tbody > tr:first-child')?.textContent?.trim() === 'ПС-7')
const rows = Array.from(table!.querySelectorAll(':scope > tbody > tr')).slice(1)
const days = []
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
let dayInfo: Day = {}
let dayLessons: Lesson[] = []
let previousRowIsDayTitle = false
for (let i = 0; i < rows.length; i++) {
const row = rows[i]
const isDivider = row.textContent?.trim() === ''
const isDayTitle = dayLessons.length === 0 && !('date' in dayInfo)
const isTableHeader = previousRowIsDayTitle
if (isDivider) {
days.push({ ...dayInfo, lessons: dayLessons })
dayLessons = []
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore
dayInfo = {}
previousRowIsDayTitle = false
} else if (isTableHeader) {
previousRowIsDayTitle = false
continue
} else if (isDayTitle) {
const { date, weekNumber } = dayTitleParser(row.querySelector('h3')!.textContent!)
dayInfo.date = date
dayInfo.weekNumber = weekNumber
previousRowIsDayTitle = true
} else {
dayLessons.push(parseLesson(row))
}
}
return days
}

View File

@@ -0,0 +1,44 @@
const isDate = (value: any): boolean => Object.prototype.toString.call(value) === '[object Date]'
export const nextSerialized = (obj: any): any => {
if (Array.isArray(obj)) {
return obj.map(nextSerialized)
}
if (typeof obj === 'object' && obj !== null) {
const newObj: any = {}
for (const [key, value] of Object.entries(obj)) {
newObj[key] = isDate(value as Date) ? (value as Date).toISOString() : nextSerialized(value)
}
return newObj
}
return obj
}
const looksLikeISODate = (value: string): boolean => /^(\d{4})-(\d{2})-(\d{2})T(\d{2}):(\d{2}):(\d{2}(?:\.\d*)?)Z?$/.test(value)
export const nextDeserializer = (obj: any): any => {
if (Array.isArray(obj)) {
return obj.map(nextDeserializer)
}
if (typeof obj === 'object' && obj !== null) {
const newObj: any = {}
for (const [key, value] of Object.entries(obj)) {
newObj[key] = typeof value === 'string' && looksLikeISODate(value) ? new Date(value) : nextDeserializer(value)
}
return newObj
}
return obj
}
export type NextSerialized<T> = {
[K in keyof T]:
T[K] extends Date ? string :
T[K] extends Array<infer U> ? NextSerialized<U>[] :
T[K] extends object ? NextSerialized<T[K]> :
T[K]
};

View File

@@ -1,4 +1,4 @@
import '@/styles/globals.css'
import '@/shared/styles/globals.css'
import type { AppProps } from 'next/app'
export default function App({ Component, pageProps }: AppProps) {

View File

@@ -1,118 +1,27 @@
import Image from 'next/image'
import { Inter } from 'next/font/google'
import { Schedule } from '@/widgets/schedule'
import { Day } from '@/shared/model/day'
import { GetServerSidePropsResult } from 'next'
import { getSchedule } from '@/app/agregator/schedule'
import { NextSerialized, nextDeserializer, nextSerialized } from '@/app/utils/date-serializer'
const inter = Inter({ subsets: ['latin'] })
type PageProps = NextSerialized<{
schedule: Day[]
}>
export default function HomePage(props: PageProps) {
const { schedule } = nextDeserializer(props)
export default function Home() {
return (
<main
className={`flex min-h-screen flex-col items-center justify-between p-24 ${inter.className}`}
>
<div className="z-10 max-w-5xl w-full items-center justify-between font-mono text-sm lg:flex">
<p className="fixed left-0 top-0 flex w-full justify-center border-b border-gray-300 bg-gradient-to-b from-zinc-200 pb-6 pt-8 backdrop-blur-2xl dark:border-neutral-800 dark:bg-zinc-800/30 dark:from-inherit lg:static lg:w-auto lg:rounded-xl lg:border lg:bg-gray-200 lg:p-4 lg:dark:bg-zinc-800/30">
Get started by editing&nbsp;
<code className="font-mono font-bold">src/pages/index.tsx</code>
</p>
<div className="fixed bottom-0 left-0 flex h-48 w-full items-end justify-center bg-gradient-to-t from-white via-white dark:from-black dark:via-black lg:static lg:h-auto lg:w-auto lg:bg-none">
<a
className="pointer-events-none flex place-items-center gap-2 p-8 lg:pointer-events-auto lg:p-0"
href="https://vercel.com?utm_source=create-next-app&utm_medium=default-template-tw&utm_campaign=create-next-app"
target="_blank"
rel="noopener noreferrer"
>
By{' '}
<Image
src="/vercel.svg"
alt="Vercel Logo"
className="dark:invert"
width={100}
height={24}
priority
/>
</a>
</div>
</div>
<div className="relative flex place-items-center before:absolute before:h-[300px] before:w-[480px] before:-translate-x-1/2 before:rounded-full before:bg-gradient-radial before:from-white before:to-transparent before:blur-2xl before:content-[''] after:absolute after:-z-20 after:h-[180px] after:w-[240px] after:translate-x-1/3 after:bg-gradient-conic after:from-sky-200 after:via-blue-200 after:blur-2xl after:content-[''] before:dark:bg-gradient-to-br before:dark:from-transparent before:dark:to-blue-700/10 after:dark:from-sky-900 after:dark:via-[#0141ff]/40 before:lg:h-[360px]">
<Image
className="relative dark:drop-shadow-[0_0_0.3rem_#ffffff70] dark:invert"
src="/next.svg"
alt="Next.js Logo"
width={180}
height={37}
priority
/>
</div>
<div className="mb-32 grid text-center lg:max-w-5xl lg:w-full lg:mb-0 lg:grid-cols-4 lg:text-left">
<a
href="https://nextjs.org/docs?utm_source=create-next-app&utm_medium=default-template-tw&utm_campaign=create-next-app"
className="group rounded-lg border border-transparent px-5 py-4 transition-colors hover:border-gray-300 hover:bg-gray-100 hover:dark:border-neutral-700 hover:dark:bg-neutral-800/30"
target="_blank"
rel="noopener noreferrer"
>
<h2 className={`mb-3 text-2xl font-semibold`}>
Docs{' '}
<span className="inline-block transition-transform group-hover:translate-x-1 motion-reduce:transform-none">
-&gt;
</span>
</h2>
<p className={`m-0 max-w-[30ch] text-sm opacity-50`}>
Find in-depth information about Next.js features and API.
</p>
</a>
<a
href="https://nextjs.org/learn?utm_source=create-next-app&utm_medium=default-template-tw&utm_campaign=create-next-app"
className="group rounded-lg border border-transparent px-5 py-4 transition-colors hover:border-gray-300 hover:bg-gray-100 hover:dark:border-neutral-700 hover:dark:bg-neutral-800/30"
target="_blank"
rel="noopener noreferrer"
>
<h2 className={`mb-3 text-2xl font-semibold`}>
Learn{' '}
<span className="inline-block transition-transform group-hover:translate-x-1 motion-reduce:transform-none">
-&gt;
</span>
</h2>
<p className={`m-0 max-w-[30ch] text-sm opacity-50`}>
Learn about Next.js in an interactive course with&nbsp;quizzes!
</p>
</a>
<a
href="https://vercel.com/templates?framework=next.js&utm_source=create-next-app&utm_medium=default-template-tw&utm_campaign=create-next-app"
className="group rounded-lg border border-transparent px-5 py-4 transition-colors hover:border-gray-300 hover:bg-gray-100 hover:dark:border-neutral-700 hover:dark:bg-neutral-800/30"
target="_blank"
rel="noopener noreferrer"
>
<h2 className={`mb-3 text-2xl font-semibold`}>
Templates{' '}
<span className="inline-block transition-transform group-hover:translate-x-1 motion-reduce:transform-none">
-&gt;
</span>
</h2>
<p className={`m-0 max-w-[30ch] text-sm opacity-50`}>
Discover and deploy boilerplate example Next.js&nbsp;projects.
</p>
</a>
<a
href="https://vercel.com/new?utm_source=create-next-app&utm_medium=default-template-tw&utm_campaign=create-next-app"
className="group rounded-lg border border-transparent px-5 py-4 transition-colors hover:border-gray-300 hover:bg-gray-100 hover:dark:border-neutral-700 hover:dark:bg-neutral-800/30"
target="_blank"
rel="noopener noreferrer"
>
<h2 className={`mb-3 text-2xl font-semibold`}>
Deploy{' '}
<span className="inline-block transition-transform group-hover:translate-x-1 motion-reduce:transform-none">
-&gt;
</span>
</h2>
<p className={`m-0 max-w-[30ch] text-sm opacity-50`}>
Instantly deploy your Next.js site to a shareable URL with Vercel.
</p>
</a>
</div>
</main>
<Schedule days={schedule} />
)
}
export async function getServerSideProps(): Promise<GetServerSidePropsResult<PageProps>> {
const schedule = await getSchedule(146)
return {
props: {
schedule: nextSerialized(schedule)
}
}
}

56
src/shadcn/ui/button.tsx Normal file
View File

@@ -0,0 +1,56 @@
import * as React from "react"
import { Slot } from "@radix-ui/react-slot"
import { cva, type VariantProps } from "class-variance-authority"
import { cn } from "@/shared/utils"
const buttonVariants = cva(
"inline-flex items-center justify-center rounded-md text-sm font-medium ring-offset-background transition-colors focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50",
{
variants: {
variant: {
default: "bg-primary text-primary-foreground hover:bg-primary/90",
destructive:
"bg-destructive text-destructive-foreground hover:bg-destructive/90",
outline:
"border border-input bg-background hover:bg-accent hover:text-accent-foreground",
secondary:
"bg-secondary text-secondary-foreground hover:bg-secondary/80",
ghost: "hover:bg-accent hover:text-accent-foreground",
link: "text-primary underline-offset-4 hover:underline",
},
size: {
default: "h-10 px-4 py-2",
sm: "h-9 rounded-md px-3",
lg: "h-11 rounded-md px-8",
icon: "h-10 w-10",
},
},
defaultVariants: {
variant: "default",
size: "default",
},
}
)
export interface ButtonProps
extends React.ButtonHTMLAttributes<HTMLButtonElement>,
VariantProps<typeof buttonVariants> {
asChild?: boolean
}
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
({ className, variant, size, asChild = false, ...props }, ref) => {
const Comp = asChild ? Slot : "button"
return (
<Comp
className={cn(buttonVariants({ variant, size, className }))}
ref={ref}
{...props}
/>
)
}
)
Button.displayName = "Button"
export { Button, buttonVariants }

25
src/shadcn/ui/input.tsx Normal file
View File

@@ -0,0 +1,25 @@
import * as React from "react"
import { cn } from "@/shared/utils"
export interface InputProps
extends React.InputHTMLAttributes<HTMLInputElement> {}
const Input = React.forwardRef<HTMLInputElement, InputProps>(
({ className, type, ...props }, ref) => {
return (
<input
type={type}
className={cn(
"flex h-10 w-full rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background file:border-0 file:bg-transparent file:text-sm file:font-medium placeholder:text-muted-foreground focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50",
className
)}
ref={ref}
{...props}
/>
)
}
)
Input.displayName = "Input"
export { Input }

24
src/shadcn/ui/label.tsx Normal file
View File

@@ -0,0 +1,24 @@
import * as React from "react"
import * as LabelPrimitive from "@radix-ui/react-label"
import { cva, type VariantProps } from "class-variance-authority"
import { cn } from "@/shared/utils"
const labelVariants = cva(
"text-sm font-medium leading-none peer-disabled:cursor-not-allowed peer-disabled:opacity-70"
)
const Label = React.forwardRef<
React.ElementRef<typeof LabelPrimitive.Root>,
React.ComponentPropsWithoutRef<typeof LabelPrimitive.Root> &
VariantProps<typeof labelVariants>
>(({ className, ...props }, ref) => (
<LabelPrimitive.Root
ref={ref}
className={cn(labelVariants(), className)}
{...props}
/>
))
Label.displayName = LabelPrimitive.Root.displayName
export { Label }

119
src/shadcn/ui/select.tsx Normal file
View File

@@ -0,0 +1,119 @@
import * as React from "react"
import * as SelectPrimitive from "@radix-ui/react-select"
import { Check, ChevronDown } from "lucide-react"
import { cn } from "@/shared/utils"
const Select = SelectPrimitive.Root
const SelectGroup = SelectPrimitive.Group
const SelectValue = SelectPrimitive.Value
const SelectTrigger = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Trigger>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Trigger>
>(({ className, children, ...props }, ref) => (
<SelectPrimitive.Trigger
ref={ref}
className={cn(
"flex h-10 w-full items-center justify-between rounded-md border border-input bg-background px-3 py-2 text-sm ring-offset-background placeholder:text-muted-foreground focus:outline-none focus:ring-2 focus:ring-ring focus:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50",
className
)}
{...props}
>
{children}
<SelectPrimitive.Icon asChild>
<ChevronDown className="h-4 w-4 opacity-50" />
</SelectPrimitive.Icon>
</SelectPrimitive.Trigger>
))
SelectTrigger.displayName = SelectPrimitive.Trigger.displayName
const SelectContent = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Content>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Content>
>(({ className, children, position = "popper", ...props }, ref) => (
<SelectPrimitive.Portal>
<SelectPrimitive.Content
ref={ref}
className={cn(
"relative z-50 min-w-[8rem] overflow-hidden rounded-md border bg-popover text-popover-foreground shadow-md data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2",
position === "popper" &&
"data-[side=bottom]:translate-y-1 data-[side=left]:-translate-x-1 data-[side=right]:translate-x-1 data-[side=top]:-translate-y-1",
className
)}
position={position}
{...props}
>
<SelectPrimitive.Viewport
className={cn(
"p-1",
position === "popper" &&
"h-[var(--radix-select-trigger-height)] w-full min-w-[var(--radix-select-trigger-width)]"
)}
>
{children}
</SelectPrimitive.Viewport>
</SelectPrimitive.Content>
</SelectPrimitive.Portal>
))
SelectContent.displayName = SelectPrimitive.Content.displayName
const SelectLabel = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Label>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Label>
>(({ className, ...props }, ref) => (
<SelectPrimitive.Label
ref={ref}
className={cn("py-1.5 pl-8 pr-2 text-sm font-semibold", className)}
{...props}
/>
))
SelectLabel.displayName = SelectPrimitive.Label.displayName
const SelectItem = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Item>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Item>
>(({ className, children, ...props }, ref) => (
<SelectPrimitive.Item
ref={ref}
className={cn(
"relative flex w-full cursor-default select-none items-center rounded-sm py-1.5 pl-8 pr-2 text-sm outline-none focus:bg-accent focus:text-accent-foreground data-[disabled]:pointer-events-none data-[disabled]:opacity-50",
className
)}
{...props}
>
<span className="absolute left-2 flex h-3.5 w-3.5 items-center justify-center">
<SelectPrimitive.ItemIndicator>
<Check className="h-4 w-4" />
</SelectPrimitive.ItemIndicator>
</span>
<SelectPrimitive.ItemText>{children}</SelectPrimitive.ItemText>
</SelectPrimitive.Item>
))
SelectItem.displayName = SelectPrimitive.Item.displayName
const SelectSeparator = React.forwardRef<
React.ElementRef<typeof SelectPrimitive.Separator>,
React.ComponentPropsWithoutRef<typeof SelectPrimitive.Separator>
>(({ className, ...props }, ref) => (
<SelectPrimitive.Separator
ref={ref}
className={cn("-mx-1 my-1 h-px bg-muted", className)}
{...props}
/>
))
SelectSeparator.displayName = SelectPrimitive.Separator.displayName
export {
Select,
SelectGroup,
SelectValue,
SelectTrigger,
SelectContent,
SelectLabel,
SelectItem,
SelectSeparator,
}

7
src/shared/model/day.ts Normal file
View File

@@ -0,0 +1,7 @@
import { Lesson } from '@/shared/model/lesson'
export type Day = {
date: Date
weekNumber: number
lessons: Lesson[]
}

View File

@@ -0,0 +1,22 @@
export type Lesson = {
isChange?: boolean
time: {
start: string
end: string
hint?: string
}
type: string
subject: string
teacher: string
place: {
address: string
classroom: number
}
topic?: string
resources: {
type: 'link'
title: string
url: string
}[]
homework: string
}

View File

@@ -0,0 +1,9 @@
import type { Day as DayType } from '@/shared/model/day'
export function Day({ day }: {
day: DayType
}) {
return (
<p></p>
)
}

View File

@@ -0,0 +1,15 @@
import type { Day as DayType } from '@/shared/model/day'
import { Day } from '@/widgets/schedule/day'
export function Schedule({ days }: {
days: DayType[]
}) {
return (
<div>
{days.map((day, i) => (
<Day day={day} key={i} />
))}
</div>
)
}

View File

@@ -0,0 +1,57 @@
import { Button } from '@/shadcn/ui/button'
import {
Card,
CardContent,
CardDescription,
CardFooter,
CardHeader,
CardTitle,
} from '@/shadcn/ui/card'
import { Input } from '@/shadcn/ui/input'
import { Label } from '@/shadcn/ui/label'
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from '@/shadcn/ui/select'
export function Lesson() {
return (
<Card className="w-[350px]">
<CardHeader>
<CardTitle>Create project</CardTitle>
<CardDescription>Deploy your new project in one-click.</CardDescription>
</CardHeader>
<CardContent>
<form>
<div className="grid w-full items-center gap-4">
<div className="flex flex-col space-y-1.5">
<Label htmlFor="name">Name</Label>
<Input id="name" placeholder="Name of your project" />
</div>
<div className="flex flex-col space-y-1.5">
<Label htmlFor="framework">Framework</Label>
<Select>
<SelectTrigger id="framework">
<SelectValue placeholder="Select" />
</SelectTrigger>
<SelectContent position="popper">
<SelectItem value="next">Next.js</SelectItem>
<SelectItem value="sveltekit">SvelteKit</SelectItem>
<SelectItem value="astro">Astro</SelectItem>
<SelectItem value="nuxt">Nuxt.js</SelectItem>
</SelectContent>
</Select>
</div>
</div>
</form>
</CardContent>
<CardFooter className="flex justify-between">
<Button variant="outline">Cancel</Button>
<Button>Deploy</Button>
</CardFooter>
</Card>
)
}