feat(lab-content-engine): phase 5 завершение — редактор связей в админке + кнопка в учебнике
- lab.js: GET /api/lab/links/all?kind= — пакетный обратный поиск (byRef map), чтобы каталог учебников не делал N+1 запросов - tests/lab-links.test.js: +3 теста для /links/all (group/400/401) -> 21/21 - admin/sections/sims.js: inline-редактор курикулумных связей на карточке симуляции (кнопка «Связи» -> панель: список связей с удалением + выбор учебника + добавить); использует /api/access/catalog, POST/DELETE /links. Без LS.modal (inline-панель) - textbooks.html: кнопка «В лабораторию» на карточке учебника, если есть связанные симуляции (один батч-запрос /links/all при загрузке); deep-link /lab?sim=<id> Двусторонняя навигация sim <-> учебник готова. Иконки .ic, без эмодзи. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
This commit is contained in:
@@ -218,6 +218,35 @@ router.get('/links', (req, res) => {
|
||||
res.json({ sims });
|
||||
});
|
||||
|
||||
/* GET /api/lab/links/all?kind=textbook
|
||||
→ { byRef: { <ref_id>: [{id,title,cat}] } } — пакетный обратный поиск для всех
|
||||
ref_id данного типа за один запрос (избегаем N+1 на странице каталога учебников).
|
||||
Отдаёт только включённые симуляции. */
|
||||
router.get('/links/all', (req, res) => {
|
||||
const kind = String(req.query.kind || '');
|
||||
if (!LINK_KINDS.includes(kind)) {
|
||||
return res.status(400).json({ error: 'неверный kind' });
|
||||
}
|
||||
let rows;
|
||||
try {
|
||||
rows = db.prepare(`
|
||||
SELECT l.ref_id, s.id, s.title, s.cat, s.enabled, s.sort_order
|
||||
FROM lab_sim_links l JOIN lab_sims s ON s.id = l.sim_id
|
||||
WHERE l.kind = ?
|
||||
ORDER BY s.sort_order, s.id
|
||||
`).all(kind);
|
||||
} catch (e) {
|
||||
return res.json({ byRef: {}, needs_migration: true });
|
||||
}
|
||||
const legacyDisabled = readLegacyDisabledIds();
|
||||
const byRef = {};
|
||||
for (const r of rows) {
|
||||
if (!r.enabled || legacyDisabled.has(r.id)) continue;
|
||||
(byRef[r.ref_id] || (byRef[r.ref_id] = [])).push({ id: r.id, title: r.title, cat: r.cat });
|
||||
}
|
||||
res.json({ byRef });
|
||||
});
|
||||
|
||||
/* ── admin: управление связями ─────────────────────────────────────────── */
|
||||
|
||||
/* POST /api/lab/sims/:id/links body: { kind, ref_id, label? } */
|
||||
|
||||
Reference in New Issue
Block a user