feat(trainer): контент по программе учебников + геометрия + фильтр предмета

- классы тем выровнены по нашим учебникам (степени/формулы/упрощение/неравенства=7, пропорции/проценты=6, квадратные=8, прогрессии=9)
- Прогрессии (9 кл): n-й член арифм./геом. прогрессии (compute)
- ГЕОМЕТРИЯ (subject geometry): Углы (сумма углов треугольника, смежные, внешний — 7 кл), Пифагор (гипотенуза/катет через тройки — 8 кл), Площади (прямоугольник/треугольник/квадрат — 8 кл)
- 36 генераторов, 12 тем; всё kind compute (числовой ответ, проверка подстановкой, sqrt в SimExpr)
- страница: фильтр предмета Алгебра/Геометрия (segmented), синхрон с adaptive/ручным выбором; иерархия Предмет → Тема → Навык
- смоук движка 572/572, страница 33/33; эмодзи/eval 0

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
Maxim Dolgolyov
2026-06-25 15:59:11 +03:00
parent d5587b4eb1
commit 6eaf68a158
3 changed files with 210 additions and 12 deletions
+156 -8
View File
@@ -20,15 +20,20 @@
════════════════════════════════════════════════════════════════════════ */
(function (global) {
// Классы по программе наших учебников (Алгебра/Геометрия 7–9).
var TOPICS = [
{ key: 'linear-eq', label: 'Уравнения', subject: 'algebra', grade: 7, order: 1 },
{ key: 'proportions', label: 'Пропорции', subject: 'algebra', grade: 7, order: 2 },
{ key: 'percents', label: 'Проценты', subject: 'algebra', grade: 7, order: 3 },
{ key: 'simplify', label: 'Упрощение', subject: 'algebra', grade: 7, order: 4 },
{ key: 'quadratic', label: 'Квадратные', subject: 'algebra', grade: 8, order: 5 },
{ key: 'powers', label: 'Степени', subject: 'algebra', grade: 8, order: 6 },
{ key: 'formulas', label: 'Формулы', subject: 'algebra', grade: 8, order: 7 },
{ key: 'inequalities', label: 'Неравенства', subject: 'algebra', grade: 8, order: 8 }
{ key: 'linear-eq', label: 'Уравнения', subject: 'algebra', grade: 7, order: 1 },
{ key: 'proportions', label: 'Пропорции', subject: 'algebra', grade: 6, order: 2 },
{ key: 'percents', label: 'Проценты', subject: 'algebra', grade: 6, order: 3 },
{ key: 'simplify', label: 'Упрощение', subject: 'algebra', grade: 7, order: 4 },
{ key: 'powers', label: 'Степени', subject: 'algebra', grade: 7, order: 5 },
{ key: 'formulas', label: 'Формулы', subject: 'algebra', grade: 7, order: 6 },
{ key: 'inequalities', label: 'Неравенства', subject: 'algebra', grade: 7, order: 7 },
{ key: 'quadratic', label: 'Квадратные', subject: 'algebra', grade: 8, order: 8 },
{ key: 'progressions', label: 'Прогрессии', subject: 'algebra', grade: 9, order: 9 },
{ key: 'g-angles', label: 'Углы', subject: 'geometry', grade: 7, order: 10 },
{ key: 'g-pyth', label: 'Пифагор', subject: 'geometry', grade: 8, order: 11 },
{ key: 'g-area', label: 'Площади', subject: 'geometry', grade: 8, order: 12 }
];
var GENERATORS = [
@@ -442,6 +447,149 @@
{ note: 'Переносим {b} вправо:', tex: '-{a}x < {cmb}' },
{ note: 'Делим на отрицательное число (−{a}) — знак неравенства МЕНЯЕТСЯ на противоположный:', tex: 'x > {root}' }
]
},
/* ═══ Тема: Прогрессии (9 класс) ═══ */
/* n-й член арифметической прогрессии */
{
id: 'prog-arith-term', topic: 'progressions', order: 1, subject: 'algebra', grade: 9, kind: 'compute',
title: 'n-й член арифм. прогрессии',
pick: { a: [-10, 20], d: [-8, 8], n: [3, 12] }, require: 'd != 0',
derive: { val: 'a + (n - 1)*d' },
lhs: 'x', rhs: '{a} + ({n} - 1)*{d}', display: 'Арифметическая прогрессия: a₁ = {a}, d = {d}. Найдите {n}-й член.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Формула n-го члена арифметической прогрессии: aₙ = a₁ + (n − 1)·d. Подставляем:', tex: 'x = {a} + ({n} - 1)*{d}' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* n-й член геометрической прогрессии */
{
id: 'prog-geom-term', topic: 'progressions', order: 2, subject: 'algebra', grade: 9, kind: 'compute',
title: 'n-й член геом. прогрессии',
pick: { b: [1, 5], q: [2, 3], n: [2, 4] },
derive: { val: 'b * q^(n - 1)' },
lhs: 'x', rhs: '{b} * {q}^({n} - 1)', display: 'Геометрическая прогрессия: b₁ = {b}, q = {q}. Найдите {n}-й член.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Формула n-го члена геометрической прогрессии: bₙ = b₁·q^(n−1). Подставляем:', tex: 'x = {b} * {q}^({n} - 1)' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* ═══ Тема: Углы (геометрия, 7 класс) ═══ */
/* третий угол треугольника */
{
id: 'ang-triangle', topic: 'g-angles', order: 1, subject: 'geometry', grade: 7, kind: 'compute',
title: 'Третий угол треугольника',
pick: { a: [20, 80], b: [20, 80] }, derive: { val: '180 - a - b' }, require: 'val >= 15 && val <= 150',
lhs: 'x', rhs: '180 - {a} - {b}', display: 'В треугольнике два угла равны {a}° и {b}°. Найдите третий угол (в градусах).',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Сумма углов треугольника равна 180°. Значит третий угол:', tex: 'x = 180 - {a} - {b}' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* смежный угол */
{
id: 'ang-adjacent', topic: 'g-angles', order: 2, subject: 'geometry', grade: 7, kind: 'compute',
title: 'Смежный угол',
pick: { a: [25, 155] }, derive: { val: '180 - a' },
lhs: 'x', rhs: '180 - {a}', display: 'Один из смежных углов равен {a}°. Найдите другой смежный с ним угол.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Сумма смежных углов равна 180°. Значит:', tex: 'x = 180 - {a}' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* внешний угол треугольника */
{
id: 'ang-exterior', topic: 'g-angles', order: 3, subject: 'geometry', grade: 7, kind: 'compute',
title: 'Внешний угол треугольника',
pick: { a: [20, 80], b: [20, 80] }, derive: { val: 'a + b' }, require: 'val <= 160',
lhs: 'x', rhs: '{a} + {b}', display: 'Внешний угол треугольника равен сумме двух не смежных с ним внутренних углов. Эти углы равны {a}° и {b}°. Найдите внешний угол.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Внешний угол треугольника равен сумме двух не смежных с ним внутренних углов:', tex: 'x = {a} + {b}' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* ═══ Тема: Теорема Пифагора (геометрия, 8 класс) ═══ */
/* гипотенуза по катетам (пифагорова тройка m,n) */
{
id: 'pyth-hyp', topic: 'g-pyth', order: 1, subject: 'geometry', grade: 8, kind: 'compute',
title: 'Гипотенуза (Пифагор)',
pick: { m: [2, 5], n: [1, 4] }, constraint: 'm > n',
derive: { a: 'm*m - n*n', b: '2*m*n', c: 'm*m + n*n' },
lhs: 'x', rhs: 'sqrt({a}^2 + {b}^2)', display: 'Катеты прямоугольного треугольника равны {a} и {b}. Найдите гипотенузу.',
answerVar: 'x', answer: 'c', integerAnswer: true,
solution: [
{ note: 'По теореме Пифагора c² = a² + b², значит c = √(a² + b²):', tex: 'x = sqrt({a}^2 + {b}^2)' },
{ note: 'Считаем (выходит целое — это пифагорова тройка):', tex: 'x = {ans}' }
]
},
/* катет по гипотенузе и катету */
{
id: 'pyth-leg', topic: 'g-pyth', order: 2, subject: 'geometry', grade: 8, kind: 'compute',
title: 'Катет (Пифагор)',
pick: { m: [2, 5], n: [1, 4] }, constraint: 'm > n',
derive: { a: 'm*m - n*n', b: '2*m*n', c: 'm*m + n*n' },
lhs: 'x', rhs: 'sqrt({c}^2 - {a}^2)', display: 'Гипотенуза прямоугольного треугольника {c}, один катет {a}. Найдите второй катет.',
answerVar: 'x', answer: 'b', integerAnswer: true,
solution: [
{ note: 'По теореме Пифагора b² = c² − a², значит b = √(c² a²):', tex: 'x = sqrt({c}^2 - {a}^2)' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* ═══ Тема: Площади (геометрия, 8 класс) ═══ */
/* площадь прямоугольника */
{
id: 'area-rect', topic: 'g-area', order: 1, subject: 'geometry', grade: 8, kind: 'compute',
title: 'Площадь прямоугольника',
pick: { a: [2, 16], b: [2, 16] }, derive: { val: 'a*b' },
lhs: 'x', rhs: '{a}*{b}', display: 'Стороны прямоугольника {a} и {b}. Найдите его площадь.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Площадь прямоугольника — произведение его сторон:', tex: 'x = {a}*{b}' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* площадь треугольника */
{
id: 'area-triangle', topic: 'g-area', order: 2, subject: 'geometry', grade: 8, kind: 'compute',
title: 'Площадь треугольника',
pick: { a: [2, 16], h: [2, 16] }, require: 'mod(a*h, 2) == 0',
derive: { val: 'a*h/2' },
lhs: 'x', rhs: '{a}*{h}/2', display: 'Основание треугольника {a}, высота к нему {h}. Найдите площадь.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Площадь треугольника — половина произведения основания на высоту:', tex: 'x = {a}*{h}/2' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
},
/* площадь квадрата */
{
id: 'area-square', topic: 'g-area', order: 3, subject: 'geometry', grade: 8, kind: 'compute',
title: 'Площадь квадрата',
pick: { a: [2, 20] }, derive: { val: 'a*a' },
lhs: 'x', rhs: '{a}^2', display: 'Сторона квадрата {a}. Найдите его площадь.',
answerVar: 'x', answer: 'val', integerAnswer: true,
solution: [
{ note: 'Площадь квадрата — сторона, возведённая в квадрат:', tex: 'x = {a}^2' },
{ note: 'Считаем:', tex: 'x = {ans}' }
]
}
];
+44 -4
View File
@@ -39,9 +39,16 @@
}
.tr-wrap { max-width: 740px; margin: 0 auto; padding: 34px 20px 90px; }
@keyframes trUp { from { opacity: 0; transform: translateY(12px); } to { opacity: 1; transform: none; } }
.tr-head, .tr-overall, .tr-mode, .tr-topbar, .tr-skillpanel, .tr-card { animation: trUp .5s var(--ease) both; }
.tr-head, .tr-overall, .tr-mode, .tr-subjects, .tr-topbar, .tr-skillpanel, .tr-card { animation: trUp .5s var(--ease) both; }
.tr-overall { animation-delay: .04s; } .tr-mode { animation-delay: .06s; }
.tr-topbar { animation-delay: .1s; } .tr-skillpanel { animation-delay: .13s; } .tr-card { animation-delay: .16s; }
.tr-subjects { animation-delay: .08s; } .tr-topbar { animation-delay: .1s; } .tr-skillpanel { animation-delay: .13s; } .tr-card { animation-delay: .16s; }
/* ── фильтр по предмету (Алгебра / Геометрия) ── */
.tr-subjects { display: flex; gap: 7px; margin-bottom: 14px; }
.tr-subjects:empty { display: none; }
.tr-subbtn { font: inherit; font-size: .86rem; font-weight: 800; cursor: pointer; padding: 8px 18px; border-radius: 99px; border: 1px solid rgba(99,102,241,.2); background: #fff; color: var(--ink-soft); transition: .16s var(--ease); }
.tr-subbtn:hover { border-color: var(--g1); color: var(--accent-ink); }
.tr-subbtn.on { color: #fff; border-color: transparent; background: linear-gradient(135deg, var(--g1), var(--g2)); box-shadow: 0 8px 20px rgba(99,102,241,.3); }
.tr-head { margin-bottom: 20px; }
.tr-h1 {
@@ -335,6 +342,7 @@
</div>
</div>
<div class="tr-subjects" id="tr-subjects"></div>
<div class="tr-topbar">
<span class="tr-nav-eyebrow">Тема</span>
<div class="tr-topics" id="tr-topics"></div>
@@ -470,6 +478,7 @@
var topics = (TG.topics ? TG.topics() : [{ key: null, label: 'Задачи' }]).concat([{ key: 'word', label: 'Текстовые задачи', word: true }]);
var isTeacher = !!(ip && ip.isTeacher);
var isAdmin = !!(ip && ip.isAdmin);
var curSubject = 'algebra'; // фильтр предмета (Алгебра/Геометрия)
var customGens = []; // пользовательские генераторы (P13), тема «Авторские»
function skillKey(g) { return g.skill || g.id; }
function skillsOf(topicKey) {
@@ -561,13 +570,29 @@
return '';
}
function topicVisible(t) { return !!(t && (t.word || t.custom || (t.subject || 'algebra') === curSubject)); }
function renderTopics() {
$('tr-topics').innerHTML = topics.map(function (t, i) {
if (!topicVisible(t)) return '';
var done = topicMastered(t.key) ? '<span class="tr-badge" title="Тема освоена">' + ICON.star + '</span>' : '';
var gr = t.grade ? '<span class="tr-grade" title="' + t.grade + ' класс">' + t.grade + '</span>' : '';
return '<button class="tr-chip' + (t.key === curTopic ? ' on' : '') + '" type="button" data-ti="' + i + '">' + esc(t.label) + gr + done + '</button>';
}).join('');
}
function presentSubjects() {
var seen = {}, out = [];
topics.forEach(function (t) { if (t.subject && !seen[t.subject]) { seen[t.subject] = 1; out.push(t.subject); } });
return out;
}
function renderSubjects() {
var el = $('tr-subjects'); if (!el) return;
var subs = presentSubjects();
if (subs.length <= 1) { el.innerHTML = ''; return; }
var LBL = { algebra: 'Алгебра', geometry: 'Геометрия' };
el.innerHTML = subs.map(function (s) {
return '<button class="tr-subbtn' + (s === curSubject ? ' on' : '') + '" type="button" data-sub="' + s + '">' + esc(LBL[s] || s) + '</button>';
}).join('');
}
function skillPanelHeader() {
var hd = $('tr-skillpanel-hd'); if (!hd) return;
var t = topics.filter(function (x) { return x.key === curTopic; })[0];
@@ -778,7 +803,7 @@
var last = (lastSkill !== undefined) ? lastSkill : (curGen ? skillKey(curGen) : null);
var id = TA.nextSkill({ ordered: ordered, progress: prog, queue: reviewQ, answered: sessAnswered, last: last });
var g = id ? gens.filter(function (x) { return skillKey(x) === id; })[0] : null;
if (g) { curGen = g; curTopic = g.topic; renderTopics(); renderSkills(); }
if (g) { curGen = g; curTopic = g.topic; if (g.subject) curSubject = g.subject; renderSubjects(); renderTopics(); renderSkills(); }
}
function recordAnswer(correct) {
var sk = currentSkill();
@@ -983,6 +1008,20 @@
$('tr-teacher').addEventListener('click', function (e) { if (e.target === $('tr-teacher')) $('tr-teacher').style.display = 'none'; });
$('tr-analytics-btn').addEventListener('click', openAnalytics);
$('tr-builder-btn').addEventListener('click', function () { location.href = '/trainer-builder'; });
$('tr-subjects').addEventListener('click', function (e) {
var b = e.target.closest('.tr-subbtn'); if (!b) return;
curSubject = b.getAttribute('data-sub');
renderSubjects();
var ct = topics.filter(function (x) { return x.key === curTopic; })[0];
if (ct && topicVisible(ct)) { renderTopics(); return; } // текущая тема видна — просто перерисовать
var first = topics.filter(function (x) { return (x.subject || 'algebra') === curSubject; })[0];
if (first) {
curTopic = first.key;
var ss = skillsOf(curTopic); curGen = ss[0] || curGen;
for (var i = 0; i < ss.length; i++) { var p = prog[skillKey(ss[i])]; if (!(p && p.mastered)) { curGen = ss[i]; break; } }
}
renderTopics(); renderSkills(); newProblem();
});
$('tr-an-close').addEventListener('click', function () { $('tr-analytics').style.display = 'none'; });
$('tr-analytics').addEventListener('click', function (e) { if (e.target === $('tr-analytics')) $('tr-analytics').style.display = 'none'; });
$('tr-an-body').addEventListener('click', function (e) {
@@ -993,6 +1032,7 @@
var b = e.target.closest('.tr-chip'); if (!b) return;
var t = topics[+b.getAttribute('data-ti')]; if (!t) return;
curTopic = t.key;
if (t.subject) { curSubject = t.subject; renderSubjects(); }
renderTopics();
if (t.word) { renderSkills(); loadWordPool(function () { serveWordProblem(); }); return; }
var ss = skillsOf(curTopic);
@@ -1051,7 +1091,7 @@
curGen = ss[0] || gens[0];
for (var si = 0; si < ss.length; si++) { var p = prog[skillKey(ss[si])]; if (!(p && p.mastered)) { curGen = ss[si]; break; } }
if (smart) pickNext(null); // адаптивный первый навык (last=null — можно взять текущий)
renderTopics(); renderSkills(); updateSession(); updateOverall(); newProblem();
renderSubjects(); renderTopics(); renderSkills(); updateSession(); updateOverall(); newProblem();
if (isTeacher) $('tr-analytics-btn').style.display = '';
if (isAdmin) $('tr-builder-btn').style.display = '';
}
+10
View File
@@ -62,6 +62,16 @@ solved-форме `x=c` → общий `onSolved` (засчитывается к
(T15 неравенства, T16 степени/формулы). **Осталось (стретч):** системы 2 ур-ний
(пара-ответ), дроби 5–6, явная привязка к таксономии ЦТ.
**Дополнено (контент по программе учебников + геометрия):** классы тем выровнены по
нашим учебникам (степени/формулы/упрощение/неравенства → 7; пропорции/проценты → 6;
квадратные → 8; прогрессии → 9). Добавлены **Прогрессии** (арифм./геом. n-й член, 9 кл)
и **ГЕОМЕТРИЯ** (subject 'geometry'): Углы (сумма углов треугольника, смежные, внешний —
7 кл), Пифагор (гипотенуза/катет через пифагоровы тройки — 8 кл), Площади (прямоугольник/
треугольник/квадрат — 8 кл) — всё kind compute (числовой ответ, проверка подстановкой,
sqrt в SimExpr). **36 генераторов, 12 тем.** На странице — **фильтр предмета
Алгебра/Геометрия** + бейдж класса на темах (Предмет → Тема → Навык). Смоук движка 572/572.
**Осталось:** системы 2 ур-ний (новый kind), дробно-рациональные, тригонометрия (9 кл геом).
Расширить охват и связать с подготовкой к ЦТ/ЦЭ.
- Новые темы: арифметика/дроби/десятичные (5–6), степени, формулы сокр. умножения,
разложение на множители, **линейные неравенства** (новый тип ответа: парсинг и