diff --git a/backend/tests/math6-page.test.js b/backend/tests/math6-page.test.js index 52b035f..fc0b97f 100644 --- a/backend/tests/math6-page.test.js +++ b/backend/tests/math6-page.test.js @@ -53,6 +53,14 @@ const CHAPTERS = [ { file: 'math_6_ch6.html', cards: 6 } ]; +test('engine: init() вызывается ПОСЛЕ экспортов (guard от sync-defer бага makeCard)', () => { + const src = readF('frontend/js/math6_engine.js'); + const exportIdx = src.indexOf('window.makeCard = makeCard'); + const initCallIdx = src.lastIndexOf('else init();'); + assert.ok(exportIdx > 0, 'есть экспорт window.makeCard'); + assert.ok(initCallIdx > exportIdx, 'else init() должен идти ПОСЛЕ window.makeCard = makeCard (иначе билдеры упадут с ReferenceError при defer-старте)'); +}); + for (const ch of CHAPTERS) { test(`${ch.file}: SPA без ошибок, ${ch.cards} карточек, активен § 1`, async () => { const { doc, errors } = await loadDom(ch.file); diff --git a/frontend/js/math6_engine.js b/frontend/js/math6_engine.js index 1d82f37..2982aa9 100644 --- a/frontend/js/math6_engine.js +++ b/frontend/js/math6_engine.js @@ -165,10 +165,11 @@ function buildParaSelector() { var BUILT = new Set(); function ensureBuilt(id) { if (BUILT.has(id)) return; - var fn = M6.builders && M6.builders[id]; + BUILT.add(id); + var cfg = window.M6 || M6; /* читаем актуальный конфиг из window */ + var fn = cfg.builders && cfg.builders[id]; if (fn) { try { fn(); } catch (e) { placeholder(id); } } else placeholder(id); - BUILT.add(id); } function placeholder(id) { var box = document.getElementById(id + '-body'); if (!box) return; @@ -368,9 +369,6 @@ function init() { window.LS.xp.load().then(function (s) { if (s && s.xp > STATE.xp) { STATE.xp = s.xp; STATE.level = calcLevel(STATE.xp); saveProgress(); refreshProgressUI(); if (STATE.current) buildSidebar(STATE.current); } }).catch(function () {}); } } -if (document.readyState === 'loading') document.addEventListener('DOMContentLoaded', init); -else init(); - /* ============================================================ EXPORTS (для inline-билдеров) */ window.goTo = goTo; window.makeCard = makeCard; @@ -386,4 +384,11 @@ window.setupSorter = setupSorter; window.confetti = confetti; window.M6icon = M6icon; window.M6engine = { goTo: goTo, ensureBuilt: ensureBuilt, refreshProgressUI: refreshProgressUI, buildSidebar: buildSidebar }; + +/* Запуск init — СТРОГО ПОСЛЕ экспортов в window. Иначе при defer-старте + (readyState='interactive') синхронная ветка else init() вызовет билдеры, + которые обращаются к makeCard/secNav/feedback ДО их экспорта → ReferenceError + → перехват в ensureBuilt → заглушка. */ +if (document.readyState === 'loading') document.addEventListener('DOMContentLoaded', init); +else init(); })();