Commit Graph

21 Commits

Author SHA1 Message Date
Maxim Dolgolyov d0484f9e55 fix(textbooks): убрать вызовы несуществующих initSquares и initRationality
Главная причина почему «Существует или нет?» (§1) не работал:
В buildP1 setTimeout цепочка была:
  initRing() → initCalc() → initSquares() → initExists() → initDual()

initSquares() — функция-не-существует (игра запускается по кнопке через squaresStart). ReferenceError рушил цепочку, поэтому initExists() и initDual() НЕ ВЫЗЫВАЛИСЬ → у dropzones не было event-listeners для drag/click → drag-and-drop не работал.

Та же проблема была в §2 с initRationality() — функция отсутствует, riStart() запускает игру по клику.

Исправил обе цепочки.
2026-05-27 12:48:16 +03:00
Maxim Dolgolyov 62d50e00ae fix(textbooks): infinite loop в §4 dragRender зависал страницу
В dragRender() (Drag 'упрости √') был while-цикл, который требовал 5 уникальных значений из набора [4,9,16,25,36,49,64,81]. Логика:
- если делит t.n нацело → всегда добавляем
- иначе → добавляем только если size<3

Для t.n=50: единственный делитель из набора это 25. После добавления sq+2 произвольных (size=3), цикл требует только делители — других нет → бесконечный цикл → зависание.

Аналогично ломалось на: 200, 48 и др.

Фикс:
1) сначала добавляем ВСЕ делители-квадраты из расширенного набора (100, 121 включены)
2) затем добивает случайными до 5 штук с лимитом 30 итераций (страховка)
3) берётся slice(0,5) на случай если ВСЕ 10 кандидатов делят t.n
2026-05-27 12:44:22 +03:00
Maxim Dolgolyov ed93b696bd fix(textbooks): Алгебра 8 — два бага
1. Hover-preview карточек §§ перекрывал соседнюю карточку
   - Было: position absolute, left:105% — превью §4 ложилось на §5
   - Стало: top:calc(100% + 8px), left:50% center, max-width:90vw
   - Добавлена 'стрелка-носик' указывающая на карточку
   - z-index 100 → 200 (поверх соседей)

2. 'Существует или нет?' в §1 — добавлен click-fallback
   - HTML5 drag&drop сохранён для desktop
   - Альтернатива: клик на корне → клик на зоне (yes/no/обратно в pool)
   - Жёлтый outline для выбранного элемента
   - Hover-эффект на .drag-item (translateY -1px + shadow)
   - Подсказка в widget-описании обновлена
2026-05-27 12:40:54 +03:00
Maxim Dolgolyov 0248e3db61 fix(textbooks): legacy initSearch() больше не бросает TypeError на старте
После Wave 3 поле #search-inp в шапке было заменено на модальный поиск Ctrl+K с #search-modal-input. Но старая функция initSearch() в init() продолжала вызывать getElementById('search-inp').addEventListener(...) — что бросало TypeError на null и крашило init() до построения первого параграфа (отсюда подвисание страницы при загрузке).

Фикс: добавлен guard 'if(!inp) return;' — функция остаётся для обратной совместимости (на случай восстановления старого input).
2026-05-27 12:35:49 +03:00
Maxim Dolgolyov 42408ee301 feat(textbooks): Wave 4 — геймификация Алгебры 8 (+1064 строк, итог 5595)
1. XP/уровни: XP_LEVELS[11], addXp(source) во всех тренажёрах и квизах, синий level-up popup, XP-карточка в сайдбаре. Persists в LocalStorage algebra8_ch1_xp
2. Streak-серии: текущая+рекорд, milestones 3/5/7/10 → оранжевый popup + ачивки streak3/5/7/10. Сброс на ошибке
3. Daily Challenge: 7 задач в DAILY_TASKS, дата-гарда, кнопка в шапке с пульсирующим индикатором, модалка с вопросом, +30 XP за прохождение
4. Achievements Gallery: кнопка 'Трофеи' в шапке, модалка с сеткой 20 ачивок (ACH_DEFS), SVG-иконки, статус earned/locked
5. Circular Progress: SVG-кольцо вместо линейной полосы на карточках §§ в para-selector
6. Финальный фейерверк: при общем прогрессе ≥95% автомодалка с confetti×5, статистикой XP/streak/achievements, освоенными темами
7. Sound effects: playTone() через Web Audio, sounds.correct/wrong/levelUp/achievement, кнопка mute в шапке с LocalStorage флагом

Все существующие функции (BUILDERS, STATE.progress, achievement, goTo, buildPN) — без изменений, новое добавлено через IIFE-обёртки.
2026-05-27 12:28:44 +03:00
Maxim Dolgolyov 898629a5b6 feat(textbooks): Wave 3 — UX-фичи Алгебры 8 (+636 строк)
1. Ctrl+K поиск: модалка со списком, индексирует параграфы, виджеты, карточки, термины глоссария. Стрелками выбор, Enter переход
2. Клавишные шорткаты: 1-7 → §§, ←/→ навигация, Esc закрыть модалки, ? показать справку. Игнорируется при фокусе в input
3. Закладки: SVG-кнопка в углу каждой .card (filled/outlined), хранятся в LocalStorage algebra8_bookmarks. В сайдбаре раздел 'Мои закладки' с переходом и удалением
4. Глоссарий-tooltips: 13 терминов (арифметический корень, радикал, иррациональное, модуль, промежуток, интервал, отрезок, система, совокупность, двойное неравенство и др.). DOM-walker оборачивает термины в .gloss с подчёркиванием, hover показывает определение в floating-tooltip
5. Mini-map: фиксированная панель справа с точкой на каждый .card/.wg в активной секции, активная подсвечивается по скроллу, скрывается на ≤980px
6. 3-уровневая подсказка: 'Подсказка' рядом с 'Проверить' в simp4 и compare. Уровень 1: намёк, 2: шаг, 3: полный ответ (−5 очков)
7. Шпаргалка drawer на мобильном: hamburger-кнопка в шапке, sidebar выезжает справа на ≤980px (transform translateX)
2026-05-27 12:18:11 +03:00
Maxim Dolgolyov 1ee16a3a38 feat(textbooks): Wave 2 — прокачка интерактивов Алгебры 8 (+422 строки)
1. Боксёрский ринг (§1): SVG-канаты вокруг квадрата + 4 цветные угловые подушки + ковёр-pattern + bell-звук через Web Audio API при S=36 + анимация боксёра-победителя на 2с
2. Доказательство √(ab)=√a·√b (§3): кнопка 'Воспроизвести' запускает 5-шаговую анимацию (подсветка прямоугольника → разрез на единичные клетки → склейка в квадрат → бейдж 'Доказано!')
3. Drag&drop с инерцией (§4): pointer-based DnD с ghost-карточкой следующей за курсором, drop-zone подсветка, неверный → тряска и возврат с инерцией, кнопочный fallback для тача
4. Match-игра (§3): SVG-overlay рисует линии соединения между парами выражений (синяя dashed pending → зелёная при совпадении / красная мигающая при ошибке)
5. Real-time валидация (liveCheck): ✓/✗ индикатор появляется при вводе во всех числовых input'ах без нажатия 'Проверить'
6. Game-over modal (squares): красивая модалка с рекордом, SVG-кубком, confetti
7. Hover-preview карточек §§: tooltip с темами параграфа и прогресс-баром
8. Fade-переходы между секциями: 180ms fadeOut + 220ms fadeIn с translateY
2026-05-27 12:08:30 +03:00
Maxim Dolgolyov 0417f51427 feat(textbooks): Wave 1 — визуальная полировка Алгебры 8 (+477 строк)
1. Цветовое разделение по §§: --sec-acc для p1 розовый, p2 фиолетовый, p3 голубой, p4 оранжевый, p5 зелёный, p6 индиго, final янтарный. Применено к .sec-num, .sec-h, .wg, .btn.primary
2. Шрифт Unbounded для всех заголовков (header, секций, hero, card-title, achievement)
3. Watermark-символы: √ ℝ × ↓ [;] { ★ — фоном в каждой секции
4. 3D-тени и translateY(-2px) на .card и .wg при hover
5. Анимированный градиент в hero (heroShift 12s loop)
6. Confetti canvas (70-100 частиц) — при правильном ответе в 14 интерактивах + при достижениях
7. Sparkle-эффект — 5 SVG-точек разлетаются из feedback-элемента
8. Achievement popup — bounce-анимация + pulse-иконка
9. card-icon с outline в цвет секции
10. Mobile polish: sidebar в drawer на ≤768, psel-grid horizontal scroll, padding 12px, шрифты −5-10% на ≤480

Не тронуто: BUILDERS, STATE, achievement logic, goTo, buildPN, классы (.psel-card, .card, .wg, .sidecard), Cache-Control.
2026-05-27 11:56:54 +03:00
Maxim Dolgolyov 8c0506ba23 fix(textbooks): Алгебра 8 — KaTeX в самооценке + щедрая шапка
1. Финальная самооценка (10 вопросов 'Я проверяю свои знания'):
   - Все 10 вопросов и 40 опций переписаны через KaTeX ($...$)
   - Корни, дроби, системы, ℕℤℚℝ, ∞, ≥, √(n-√...) — теперь рендерятся настоящей математической типографикой
   - Пример: было '√2 принадлежит множеству: ℕ ℤ ℚ I' (Unicode) → стало '\sqrt{2} ... \mathbb{N} \mathbb{Z} \mathbb{Q} \mathbb{I}'

2. Шапка с большим воздухом:
   - padding 34/24 → 46/30 (top/bottom)
   - min-height 130px — гарантия не сжаться
   - h1: line-height 1.25 → 1.3, padding-top 2 → 4px
   - sub: line-height 1.35 → 1.4
   - Watermark теперь центрируется через top:50%/translateY(-50%) — больше не лезет на текст
2026-05-27 11:40:51 +03:00
Maxim Dolgolyov 055599bb01 fix(textbooks): KaTeX в финале + анти-кэш в Алгебре 8
1. Финал главы:
   - После buildAssessment() повторный renderMath(body) — захватывает формулы из квиза
   - Дополнительный renderMath через 300ms — на случай если KaTeX не успел загрузиться

2. Мета-теги Cache-Control no-cache, no-store, must-revalidate / Pragma no-cache / Expires 0 — чтобы прежняя версия страницы не зависала в кэше браузера (поэтому и шапка не обновлялась)
2026-05-27 11:36:39 +03:00
Maxim Dolgolyov 5e7098a610 perf(textbooks): lazy-build параграфов Алгебры 8 — стартовая загрузка стала мгновенной
Было: init() синхронно вызывал buildP1...buildFinal — 7 секций × ~500 строк HTML, плюс KaTeX renderMathInElement сканировал весь body. На медленном CPU могло подвисать на 2-5 секунд.

Стало: init() строит только §1 (через goTo('p1')). Остальные секции строятся лениво при первом goTo(id) — кэшируются в BUILT Set.

Профит: первая отрисовка в 7 раз быстрее. KaTeX-рендер тоже только для активной секции.
2026-05-27 11:32:39 +03:00
Maxim Dolgolyov c335f33e25 fix(textbooks): retroactive-фикс существующих достижений Алгебры 8
Прошлый коммит хранил название в Map, но старые записи в LocalStorage (Set из id-ов) подгружались с id в качестве текста — пользователь по-прежнему видел 'ring36', 'start'.

Фикс: словарь ACH_LABELS (id → название) применяется при загрузке как fallback:
- старый формат массив id-ов: id → ACH_LABELS[id]
- новый формат объект {id:text}: если text === id, используем ACH_LABELS[id]

Теперь при следующем открытии учебника старые достижения автоматически получат красивые названия.
2026-05-27 11:30:52 +03:00
Maxim Dolgolyov 0927605bd0 fix(textbooks): не обрезать заголовок Алгебры 8
Заголовок 'Алгебра 8 · Глава 1' визуально обрезался сверху из-за тяжёлого шрифта (font-weight:900) без явного line-height на тесном padding-top 24px.

Фикс:
- padding header: 24px → 34px сверху, 22px → 24px снизу
- h1: добавлены line-height:1.25 и padding-top:2px
- hdr-sub: line-height 1.35, margin-top 4 → 6px
- watermark 'АЛГЕБРА': top -18% → -10%, max font 13rem → 12rem (меньше залезает на текст)
2026-05-27 11:23:41 +03:00
Maxim Dolgolyov 8838f963a3 fix(textbooks): шпаргалка показывает человеч. названия достижений вместо id
Было: 'ring36', 'start' — внутренние id-ы достижений
Стало: 'Начало пути по корням!', 'Нашёл сторону ринга'

STATE.achievements теперь Map(id → text). Старый формат массива id-ов читается с обратной совместимостью (id используется как текст). При сохранении пишется как объект.
2026-05-27 11:21:33 +03:00
Maxim Dolgolyov e43f14b5e1 chore(textbooks): убрать introblock и info-grid c hub-страницы Физики 8
Убраны:
- <section class="intro"> с заголовком 'Изучаем 3 раздела физики' (там всё ещё видна была старая надпись 'Автор: Исаченкова Л. А.')
- <section class="info-grid"> с 4 info-карточками (Интерактив в каждом §, Прогресс сохраняется, Формулы — KaTeX, Светлая и тёмная тема)

Hub теперь чище: шапка → общий прогресс → 3 карточки разделов → подвал.
2026-05-27 11:05:07 +03:00
Maxim Dolgolyov ea753cf5b0 chore(textbooks): убрать упоминания авторов — учебники теперь как внутренние работы
- Миграция 011: UPDATE textbooks SET author='' (все 4 записи)
- algebra_8.html: убрано из <title> и футера
- physics_8.html (hub): убрано из title/header/intro/footer, заменено на LearnSpace
- physics8_*.html (3 файла): убраны все вхождения '· Исаченкова' в подписях §
- physics_9.html: убраны все вхождения '· Исаченкова' в подписях §
- chemistry_9.html: убраны 3 упоминания '· Шиманович' в подписях

В каталоге /textbooks автор больше не отображается под названием (так как поле пустое).
2026-05-27 11:03:25 +03:00
Maxim Dolgolyov 87e78714b7 feat(textbooks): объединить 3 части Физики 8 в один hub-учебник
Подход: hub-страница, а не слияние файлов.

Проблема: 3 готовых файла-главы (thermal/electro/optics) занимали 3 карточки в каталоге. Физическое слияние в один файл = 800КБ+, конфликты CSS/JS namespaces, риск сломать KaTeX.

Решение:
- Создан frontend/textbooks/physics_8.html — hub-страница с 3 крупными карточками-разделами (амбер/синий/фиолетовый)
- Карточки ссылаются напрямую на /textbooks/physics8_thermal.html и т.д. (express.static уже отдаёт эти файлы)
- Из каталога /textbooks теперь видна ОДНА карточка «Физика 8», sort_order 4
- Hub-страница показывает прогресс по каждой главе через LocalStorage (best-effort парсинг)
- Header «К каталогу», переключатель темы синхронизирован с главами

Миграция 010: удалила 3 прежние записи (physics-8-thermal/electro/optics), добавила физическо-8 → physics_8.html, para_count=40.

Эмодзи в hub не используются (только inline SVG). Эмодзи в файлах глав остались — это контент.
2026-05-27 09:55:24 +03:00
Maxim Dolgolyov 6a65e223b8 feat(textbooks): добавить учебник «Физика 8» (3 части)
Интегрирован готовый интерактивный учебник по физике 8 класса (40 параграфов, разбитых на 3 файла):

- physics8_thermal.html (§1–11)   — Тепловые явления
- physics8_electro.html (§12–31)  — Электрические явления
- physics8_optics.html  (§32–40)  — Световые явления

Все три самодостаточные (KaTeX через CDN, шрифт Outfit, dark mode, анимации, эмодзи).
Автор: Исаченкова Л. А.

Миграция 009 регистрирует три новых textbook-записи:
- physics-8-thermal (amber, sort 4)
- physics-8-electro (blue, sort 5)
- physics-8-optics (violet, sort 6)

После миграции доступны через /textbook/physics-8-thermal и т.д. и видны в каталоге /textbooks.

Pre-commit hook на эмодзи обойден --no-verify по разрешению пользователя: эмодзи здесь являются частью авторского контента учебника (визуальные маркеры разделов: тепловые/электрические/оптические явления), а не нашим кодом.
2026-05-27 09:51:00 +03:00
Maxim Dolgolyov fff3ddc45e fix(textbooks): KaTeX-рендер в шпаргалке Алгебры 8
Боковая шпаргалка строилась обычным HTML (Unicode-символы √ ≤ ⊂), формулы не оформлялись как настоящие математические.

Фикс:
- Все формулы в SIDEBARS обёрнуты $-делимитерами KaTeX (\sqrt, \mathbb, \cap, \subset, \Leftrightarrow и т.д.)
- После buildSidebar() вызывается renderMathInElement(box) для встроенного рендера
- Учебник теперь показывает корни и множества в правильной типографике
2026-05-27 09:41:20 +03:00
Maxim Dolgolyov 92a0a364ea feat(textbooks): интерактивный учебник «Алгебра 8 · Глава 1» по Арефьевой/Пирютко
algebra_8.html (3226 строк, 192KB) — полная Глава 1 «Квадратные корни и их свойства. Действительные числа»:

§ 1. Квадратный корень. Арифметический корень:
- Hero «Боксёрский ринг 36 м²» с draggable углом
- Калькулятор √ с проверкой r²
- Игра «Таблица квадратов 10-99» (speedrun, рейтинг в LocalStorage)
- Drag «существует/не существует» для √(-25), √121 и т.д.
- Связка x² ↔ √x с слайдером
- Алгоритм извлечения «в столбик»

§ 2. Иррациональные числа / Действительные числа:
- Анимированная иерархия ℕ⊂ℤ⊂ℚ⊂ℝ
- Drag-сортировка чисел в 4 коробки
- Числовая прямая с √2, √3, √5, π
- Конвертер дробь ⇄ периодическая десятичная
- Пошаговое доказательство √2∈I (5 раскрывающихся шагов)
- Игра «Кто рациональнее?»

§ 3. Свойства корней:
- Геометрическое доказательство √(ab)=√a·√b (SVG)
- Слайдер-проверка свойств (a, b → live)
- Match-игра «выражение ↔ ответ»
- Калькулятор |a|=√(a²)
- Тренажёр «Упрости» (12 задач)

§ 4. Применение свойств:
- Drag «упрости √» (9 заданий, ищи точный квадрат)
- Конвертер a√b ⇄ √c (двусторонний)
- Помощник освобождения от иррациональности (пошагово)
- «Кто больше?» с подсказкой через квадрат
- Тренажёр «a√b» (11 заданий)

§ 5. Числовые промежутки:
- 9 типов промежутков в таблице
- Конструктор промежутка (drag границ, переключение скобок)
- Объединение/пересечение визуально (4 слайдера, 4 промежутка)
- «По картинке — неравенство» (4 задачи)
- «По записи — нарисуй» (4 задачи)

§ 6. Системы неравенств:
- Решатель системы с автоматическим пересечением
- Переключатель «Система ∩ / Совокупность ∪»
- Двойное неравенство как система (пошаговое разложение)
- Игра «Найди целые решения»
- Задача про тарифы 1.382 из учебника

Финал главы:
- Итоговая самооценка (10 заданий с авто-проверкой)
- 3 практические задачи (дорожка с розами, цемент, часовые пояса)
- Историческая справка (Рудольф, Бхаскара, Герон, пифагорейцы)
- Олимпиадная расшифровка кода 25-324-441-64-4-1 → ДРУЖБА
- Метод Герона интерактивно (с итерациями)
- Олимпиадная задача про a+√15 и 1/a−√15

Сквозные фичи:
- KaTeX через CDN, шрифт Inter+Manrope
- Розово-голубая палитра учебника Арефьевой
- Dark mode toggle
- LocalStorage прогресс по §§ + достижения
- Sticky шпаргалка справа (мобильно — снизу)
- Поиск по карточкам параграфов
- Адаптив ≤ 980px
2026-05-27 09:32:09 +03:00
Maxim Dolgolyov e8018d85c1 feat: textbooks — модуль учебников + чтение как ДЗ (3 фазы)
Фаза 1 — структура и каталог:
  - frontend/textbooks/chemistry_9.html (Шиманович, 60 §) + physics_9.html (Исаченкова, 38 §)
  - frontend/textbooks.html — каталог в стиле LearnSpace (карточки с обложками)
  - Маршруты: /textbooks (каталог), /textbook/<slug> (полноэкранный учебник)
  - Сайдбар: пункт «Учебники» (book-open-text)
  - Feature flag feature_textbooks_enabled, hideDisabledFeatures map

Фаза 2 — прогресс в localStorage + UI чтения:
  - frontend/js/textbook-tracker.js — инжектится в каждый учебник:
    - «← Учебники» overlay-кнопка (top-left, semi-transparent)
    - «Прочитано» чекбокс рядом с каждым §-заголовком
    - Зелёный dot на pill уже прочитанных параграфов
    - Авто-открытие последнего параграфа при возврате
  - Каталог показывает прогресс-бар «X из Y прочитано» + кнопку «Продолжить»

Фаза 3 — серверный прогресс + назначение чтения как ДЗ:
  - Таблица textbooks (slug, subject, grade, title, author, color, ...)
  - Таблица textbook_progress (user_id, textbook_id, JSON read[], last_para)
  - Колонки assignments.textbook_id + textbook_paragraphs
  - API: GET /api/textbooks (с прогрессом), GET /:slug, POST /:slug/progress,
    GET /:slug/class-progress (учитель)
  - tracker.js синхронизирует прогресс через POST /progress (если залогинен)
  - На каталоге у учителей кнопка «Назначить чтение» — модалка с выбором
    классов + параграфы («1-5» или «1,3,5») + deadline
  - bulkCreateAssignment расширен: принимает textbook_slug, резолвит в id

Миграция 004 идемпотентная; сиды двух учебников включены.
2026-05-16 14:05:19 +03:00