Дорабоки интерфейса системы тестирования. Раздел 1 Шапка+Верхний brick
This commit is contained in:
@@ -25,14 +25,74 @@
|
||||
const saveStatusEl = $('#save-status');
|
||||
const aiStatusEl = $('#ai-status');
|
||||
const chainActiveEl = $('#chain-active');
|
||||
const aiTopicEl = $('#ai-topic');
|
||||
const aiQCountEl = $('#ai-q-count');
|
||||
const aiOCountEl = $('#ai-o-count');
|
||||
const introUpdatedEl = $('#intro-updated');
|
||||
const introForkBannerEl = $('#intro-fork-banner');
|
||||
const versionsListEl = $('#versions-list');
|
||||
const attemptsListEl = $('#attempts-list');
|
||||
const visibilityBtn = $('#btn-toggle-visibility');
|
||||
const assignSearchEl = $('#assign-search');
|
||||
const assignDeptEl = $('#assign-dept');
|
||||
const assignClinicEl = $('#assign-clinic');
|
||||
const assignListEl = $('#assign-list');
|
||||
const assignSelectAllBtn = $('#assign-select-all');
|
||||
const assignSubmitBtn = $('#assign-submit');
|
||||
const assignStatusEl = $('#assign-status');
|
||||
|
||||
const tplQ = $('#tpl-question');
|
||||
const tplO = $('#tpl-option');
|
||||
|
||||
let chainActive = true;
|
||||
let assignPeople = [];
|
||||
let assignSelected = new Set();
|
||||
let hasAnyAttempts = false;
|
||||
let hasForkRisk = Boolean(initial?.test?.hasForkRisk)
|
||||
|| (introForkBannerEl && introForkBannerEl.dataset.forkRisk === '1');
|
||||
let baselineDraftKey = '';
|
||||
let dirtyCheckQueued = false;
|
||||
|
||||
function currentDraftKey() {
|
||||
return JSON.stringify(collectPayload());
|
||||
}
|
||||
|
||||
function isDirty() {
|
||||
return baselineDraftKey !== '' && baselineDraftKey !== currentDraftKey();
|
||||
}
|
||||
|
||||
function updateForkBanner() {
|
||||
if (!introForkBannerEl) return;
|
||||
introForkBannerEl.style.display = (hasForkRisk && isDirty()) ? '' : 'none';
|
||||
}
|
||||
|
||||
function scheduleDirtyCheck() {
|
||||
if (dirtyCheckQueued) return;
|
||||
dirtyCheckQueued = true;
|
||||
requestAnimationFrame(() => {
|
||||
dirtyCheckQueued = false;
|
||||
updateForkBanner();
|
||||
});
|
||||
}
|
||||
|
||||
function resetBaselineDraft() {
|
||||
baselineDraftKey = currentDraftKey();
|
||||
updateForkBanner();
|
||||
}
|
||||
|
||||
// ─── render ─────────────────────────────────────────────────────────
|
||||
|
||||
function syncOptionInputTypes(qNode) {
|
||||
const isMulti = $('.q-multi', qNode).checked;
|
||||
const qName = `q-correct-${Math.random().toString(36).slice(2)}`;
|
||||
$$('.opt-correct', qNode).forEach((input) => {
|
||||
input.type = isMulti ? 'checkbox' : 'radio';
|
||||
if (isMulti) input.removeAttribute('name');
|
||||
else input.setAttribute('name', qName);
|
||||
input.classList.add('question-option-row__mark');
|
||||
});
|
||||
}
|
||||
|
||||
function renderQuestion(q) {
|
||||
const node = tplQ.content.firstElementChild.cloneNode(true);
|
||||
node._q = { id: q.id || null };
|
||||
@@ -43,6 +103,7 @@
|
||||
(q.options || []).forEach((o) => optsEl.appendChild(renderOption(o)));
|
||||
|
||||
bindQuestionEvents(node);
|
||||
syncOptionInputTypes(node);
|
||||
return node;
|
||||
}
|
||||
|
||||
@@ -61,41 +122,86 @@
|
||||
if (!confirm('Удалить вопрос?')) return;
|
||||
node.remove();
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
});
|
||||
$('.q-up', node).addEventListener('click', () => {
|
||||
if (node.previousElementSibling) {
|
||||
node.parentNode.insertBefore(node, node.previousElementSibling);
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
}
|
||||
});
|
||||
$('.q-down', node).addEventListener('click', () => {
|
||||
if (node.nextElementSibling) {
|
||||
node.parentNode.insertBefore(node.nextElementSibling, node);
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
}
|
||||
});
|
||||
$('.q-add-option', node).addEventListener('click', () => {
|
||||
$('.q-options', node).appendChild(renderOption({ text: '', isCorrect: false }));
|
||||
syncOptionInputTypes(node);
|
||||
scheduleDirtyCheck();
|
||||
});
|
||||
$('.q-ai', node).addEventListener('click', () => aiGenerateQuestion(node));
|
||||
$('.q-multi', node).addEventListener('change', () => {
|
||||
syncOptionInputTypes(node);
|
||||
scheduleDirtyCheck();
|
||||
});
|
||||
}
|
||||
|
||||
function renumber() {
|
||||
$$('#questions .q-item').forEach((li, i) => {
|
||||
$('.q-num', li).textContent = `Вопрос #${i + 1}`;
|
||||
});
|
||||
qCountEl.textContent = $$('#questions .q-item').length;
|
||||
const n = $$('#questions .q-item').length;
|
||||
if (qCountEl) qCountEl.textContent = n;
|
||||
const mirror = document.getElementById('q-count-mirror');
|
||||
if (mirror) mirror.textContent = n;
|
||||
}
|
||||
|
||||
function autoResize(el) {
|
||||
if (!el) return;
|
||||
el.style.height = 'auto';
|
||||
el.style.height = el.scrollHeight + 'px';
|
||||
}
|
||||
|
||||
function loadInitial() {
|
||||
titleEl.value = initial.test.title || '';
|
||||
descEl.value = initial.test.description || '';
|
||||
autoResize(titleEl);
|
||||
autoResize(descEl);
|
||||
if (titleEl && titleEl.tagName === 'TEXTAREA') {
|
||||
titleEl.addEventListener('input', () => autoResize(titleEl));
|
||||
titleEl.addEventListener('keydown', (e) => {
|
||||
if (e.key === 'Enter') e.preventDefault();
|
||||
});
|
||||
}
|
||||
if (descEl) descEl.addEventListener('input', () => autoResize(descEl));
|
||||
thresholdEl.value =
|
||||
initial.test.passingThreshold == null ? '' : Number(initial.test.passingThreshold);
|
||||
|
||||
questionsEl.innerHTML = '';
|
||||
(initial.questions || []).forEach((q) => questionsEl.appendChild(renderQuestion(q)));
|
||||
renumber();
|
||||
if (aiTopicEl && !aiTopicEl.value.trim()) {
|
||||
aiTopicEl.value = initial.test.title || '';
|
||||
}
|
||||
}
|
||||
|
||||
function fmtDt(iso) {
|
||||
if (!iso) return '—';
|
||||
try {
|
||||
return new Date(iso).toLocaleString('ru-RU', { dateStyle: 'short', timeStyle: 'short' });
|
||||
} catch {
|
||||
return '—';
|
||||
}
|
||||
}
|
||||
|
||||
function escHtml(s) {
|
||||
return String(s == null ? '' : s)
|
||||
.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')
|
||||
.replace(/"/g, '"').replace(/'/g, ''');
|
||||
}
|
||||
|
||||
// ─── collect ───────────────────────────────────────────────────────
|
||||
@@ -144,6 +250,7 @@
|
||||
}),
|
||||
);
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
});
|
||||
|
||||
$('#save-draft').addEventListener('click', async () => {
|
||||
@@ -167,6 +274,7 @@
|
||||
saveStatusEl.textContent = data.forked
|
||||
? 'Сохранено (создана новая версия — есть попытки прохождения).'
|
||||
: 'Сохранено.';
|
||||
resetBaselineDraft();
|
||||
setTimeout(() => (saveStatusEl.textContent = ''), 4000);
|
||||
} catch (e) {
|
||||
saveStatusEl.textContent = '';
|
||||
@@ -175,18 +283,24 @@
|
||||
});
|
||||
|
||||
$('#ai-generate-test').addEventListener('click', async () => {
|
||||
const shape = collectShape();
|
||||
if (!shape.length) {
|
||||
alert('Сначала добавьте хотя бы один вопрос (его настройки определяют сетку).');
|
||||
const topic = (aiTopicEl?.value || titleEl.value || '').trim();
|
||||
if (!topic) {
|
||||
alert('Укажите тему.');
|
||||
return;
|
||||
}
|
||||
const nQ = Math.min(30, Math.max(1, Number(aiQCountEl?.value || 7) || 7));
|
||||
const nO = Math.min(8, Math.max(2, Number(aiOCountEl?.value || 3) || 3));
|
||||
const shape = Array.from({ length: nQ }, () => ({
|
||||
optionsCount: nO,
|
||||
hasMultipleAnswers: false,
|
||||
}));
|
||||
aiStatusEl.textContent = 'Генерируем…';
|
||||
try {
|
||||
const r = await fetch(`/api/tests/${TEST_ID}/ai/generate-test`, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({
|
||||
testTitle: titleEl.value,
|
||||
testTitle: topic,
|
||||
testDescription: descEl.value,
|
||||
shape,
|
||||
}),
|
||||
@@ -194,11 +308,15 @@
|
||||
const data = await r.json();
|
||||
if (!r.ok) throw new Error(data.error || 'AI: ошибка.');
|
||||
const draft = data.draft;
|
||||
if (draft.title) titleEl.value = draft.title;
|
||||
if (draft.title) {
|
||||
titleEl.value = draft.title;
|
||||
if (aiTopicEl) aiTopicEl.value = draft.title;
|
||||
}
|
||||
if (draft.description) descEl.value = draft.description;
|
||||
questionsEl.innerHTML = '';
|
||||
(draft.questions || []).forEach((q) => questionsEl.appendChild(renderQuestion(q)));
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
aiStatusEl.textContent = `Готово: ${draft.questions?.length || 0} вопросов.`;
|
||||
setTimeout(() => (aiStatusEl.textContent = ''), 4000);
|
||||
} catch (e) {
|
||||
@@ -242,6 +360,7 @@
|
||||
questionsEl.innerHTML = '';
|
||||
(draft.questions || []).forEach((q) => questionsEl.appendChild(renderQuestion(q)));
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
aiStatusEl.textContent = `Применено: ${draft.questions?.length || 0} вопросов.`;
|
||||
setTimeout(() => (aiStatusEl.textContent = ''), 4000);
|
||||
} catch (e) {
|
||||
@@ -263,12 +382,6 @@
|
||||
alert(msg);
|
||||
}
|
||||
|
||||
function escHtml(s) {
|
||||
return String(s == null ? '' : s)
|
||||
.replace(/&/g, '&').replace(/</g, '<').replace(/>/g, '>')
|
||||
.replace(/"/g, '"').replace(/'/g, ''');
|
||||
}
|
||||
|
||||
const modal = $('#ai-modal');
|
||||
const modalTitle = $('#ai-modal-title');
|
||||
const modalBody = $('#ai-modal-body');
|
||||
@@ -290,7 +403,8 @@
|
||||
modal.showModal();
|
||||
}
|
||||
|
||||
$('#ai-generate-by-title').addEventListener('click', async () => {
|
||||
const aiGenerateByTitleBtn = $('#ai-generate-by-title');
|
||||
if (aiGenerateByTitleBtn) aiGenerateByTitleBtn.addEventListener('click', async () => {
|
||||
const title = titleEl.value.trim();
|
||||
if (!title) {
|
||||
alert('Сначала заполните название теста.');
|
||||
@@ -334,6 +448,7 @@
|
||||
questionsEl.innerHTML = '';
|
||||
(draft.questions || []).forEach((q) => questionsEl.appendChild(renderQuestion(q)));
|
||||
renumber();
|
||||
scheduleDirtyCheck();
|
||||
aiStatusEl.textContent = `Применено: ${draft.questions.length} вопросов.`;
|
||||
setTimeout(() => (aiStatusEl.textContent = ''), 4000);
|
||||
} catch (e) {
|
||||
@@ -342,7 +457,8 @@
|
||||
}
|
||||
});
|
||||
|
||||
$('#ai-check').addEventListener('click', async () => {
|
||||
const aiCheckBtn = $('#ai-check');
|
||||
if (aiCheckBtn) aiCheckBtn.addEventListener('click', async () => {
|
||||
const payload = collectPayload();
|
||||
if (!payload.questions.length) {
|
||||
alert('В тесте нет вопросов — нечего проверять.');
|
||||
@@ -394,7 +510,8 @@
|
||||
}
|
||||
});
|
||||
|
||||
$('#ai-improve').addEventListener('click', async () => {
|
||||
const aiImproveBtn = $('#ai-improve');
|
||||
if (aiImproveBtn) aiImproveBtn.addEventListener('click', async () => {
|
||||
const payload = collectPayload();
|
||||
if (!payload.questions.length) {
|
||||
alert('В тесте нет вопросов — нечего улучшать.');
|
||||
@@ -480,6 +597,7 @@
|
||||
it.suggested.options.forEach((o) => optsEl.appendChild(renderOption(o)));
|
||||
});
|
||||
modal.close();
|
||||
scheduleDirtyCheck();
|
||||
aiStatusEl.textContent = 'Изменения применены. Не забудьте сохранить.';
|
||||
setTimeout(() => (aiStatusEl.textContent = ''), 5000);
|
||||
},
|
||||
@@ -517,6 +635,7 @@
|
||||
data.options.forEach((o) => optsEl.appendChild(renderOption(o)));
|
||||
$('.q-multi', node).checked = !!data.hasMultipleAnswers;
|
||||
}
|
||||
scheduleDirtyCheck();
|
||||
aiStatusEl.textContent = data.mode === 'full' ? 'AI: вопрос сгенерирован.' : 'AI: формулировка обновлена.';
|
||||
setTimeout(() => (aiStatusEl.textContent = ''), 4000);
|
||||
} catch (e) {
|
||||
@@ -542,5 +661,195 @@
|
||||
chainActiveEl.checked = true;
|
||||
});
|
||||
|
||||
function renderVersions(rows) {
|
||||
if (!versionsListEl) return;
|
||||
versionsListEl.innerHTML = '';
|
||||
(rows || []).forEach((r) => {
|
||||
const li = document.createElement('li');
|
||||
li.className = 'surface-card version-card-list__item';
|
||||
li.innerHTML = `
|
||||
<div class="version-card-list__row">
|
||||
<div class="version-card-list__main">
|
||||
<div class="version-card-list__title-line">
|
||||
<span class="font-headline" style="font-size:1rem;">v${r.version}</span>
|
||||
${r.is_active ? '<span class="code-inline" style="font-size:0.7rem;">текущая</span>' : ''}
|
||||
</div>
|
||||
<p class="muted mono" style="margin:.4rem 0 0; font-size:.8rem;">${fmtDt(r.created_at)}</p>
|
||||
<p class="muted" style="margin:.2rem 0 0; font-size:.8rem;">Активна: ${r.is_active ? 'да' : 'нет'}</p>
|
||||
</div>
|
||||
</div>`;
|
||||
versionsListEl.appendChild(li);
|
||||
});
|
||||
}
|
||||
|
||||
function renderAttempts(rows) {
|
||||
if (!attemptsListEl) return;
|
||||
attemptsListEl.innerHTML = '';
|
||||
(rows || []).forEach((a) => {
|
||||
const when = a.completedAt ? fmtDt(a.completedAt) : (a.startedAt ? fmtDt(a.startedAt) : '—');
|
||||
const result = a.status === 'completed' && a.totalQuestions != null
|
||||
? `${a.correctCount}/${a.totalQuestions}${a.passed ? ' · зачёт' : ' · незачёт'}`
|
||||
: a.status;
|
||||
const li = document.createElement('li');
|
||||
li.className = 'surface-card attempts-card-list__item';
|
||||
li.innerHTML = `
|
||||
<div class="attempts-card-list__row">
|
||||
<div class="attempts-card-list__main">
|
||||
<p class="muted mono" style="margin:0; font-size:.8rem;">${when}</p>
|
||||
<p style="margin:.35rem 0 0; font-weight:600;">${escHtml(a.attempterName || '—')}
|
||||
${a.attempterLogin ? `<span class="code-inline" style="font-size:11px; margin-left:6px;">${escHtml(a.attempterLogin)}</span>` : ''}
|
||||
</p>
|
||||
<p class="muted" style="margin:.25rem 0 0; font-size:.85rem;">v${a.testVersion} · ${escHtml(result)}</p>
|
||||
</div>
|
||||
${a.status === 'completed'
|
||||
? `<a class="btn btn-ghost btn--sm attempts-card-list__action" href="/tests/${TEST_ID}/attempts/${a.id}/review">Разбор</a>`
|
||||
: ''}
|
||||
</div>`;
|
||||
attemptsListEl.appendChild(li);
|
||||
});
|
||||
}
|
||||
|
||||
function renderAssignList() {
|
||||
if (!assignListEl) return;
|
||||
assignListEl.innerHTML = '';
|
||||
assignPeople.forEach((p) => {
|
||||
const row = document.createElement('label');
|
||||
row.className = `assign-row${assignSelected.has(String(p.staffId)) ? ' assign-row--selected' : ''}`;
|
||||
row.innerHTML = `
|
||||
<input type="checkbox" ${assignSelected.has(String(p.staffId)) ? 'checked' : ''} />
|
||||
<span class="assign-row__text">
|
||||
<span class="assign-row__fio">${escHtml(p.fio || '—')}</span>
|
||||
${p.webLogin ? `<span class="assign-row__login">${escHtml(p.webLogin)}</span>` : ''}
|
||||
<span class="assign-row__meta">${escHtml(p.department || '—')}${p.clinicUserId ? ' · есть учётка' : ' · нет учётки (создадим при назначении)'}</span>
|
||||
</span>`;
|
||||
const cb = row.querySelector('input');
|
||||
cb.addEventListener('change', () => {
|
||||
const k = String(p.staffId);
|
||||
if (cb.checked) assignSelected.add(k); else assignSelected.delete(k);
|
||||
row.classList.toggle('assign-row--selected', cb.checked);
|
||||
});
|
||||
assignListEl.appendChild(row);
|
||||
});
|
||||
if (!assignPeople.length) assignListEl.innerHTML = '<p class="muted" style="padding:.75rem;">Никого не найдено.</p>';
|
||||
}
|
||||
|
||||
async function loadDirectory() {
|
||||
if (!assignListEl) return;
|
||||
assignStatusEl.textContent = 'Загружаем…';
|
||||
try {
|
||||
const params = new URLSearchParams();
|
||||
if (assignSearchEl.value.trim()) params.set('q', assignSearchEl.value.trim());
|
||||
if (assignDeptEl.value && assignDeptEl.value !== '__all__') params.set('department', assignDeptEl.value);
|
||||
params.set('clinic', assignClinicEl.value || 'all');
|
||||
const r = await fetch(`/api/auth/dev/assignment-directory?${params.toString()}`);
|
||||
const data = await r.json();
|
||||
if (!r.ok) throw new Error(data.error || 'Не удалось загрузить сотрудников');
|
||||
assignPeople = data.people || [];
|
||||
const depts = data.departments || [];
|
||||
if (assignDeptEl.options.length <= 1) {
|
||||
depts.forEach((d) => {
|
||||
const o = document.createElement('option');
|
||||
o.value = d;
|
||||
o.textContent = d;
|
||||
assignDeptEl.appendChild(o);
|
||||
});
|
||||
}
|
||||
assignSelected = new Set();
|
||||
renderAssignList();
|
||||
assignStatusEl.textContent = '';
|
||||
} catch (e) {
|
||||
assignStatusEl.textContent = e.message || 'Ошибка загрузки';
|
||||
}
|
||||
}
|
||||
|
||||
if (assignSearchEl) {
|
||||
let t = null;
|
||||
assignSearchEl.addEventListener('input', () => {
|
||||
clearTimeout(t);
|
||||
t = setTimeout(loadDirectory, 350);
|
||||
});
|
||||
assignDeptEl.addEventListener('change', loadDirectory);
|
||||
assignClinicEl.addEventListener('change', loadDirectory);
|
||||
assignSelectAllBtn.addEventListener('click', () => {
|
||||
assignPeople.forEach((p) => assignSelected.add(String(p.staffId)));
|
||||
renderAssignList();
|
||||
});
|
||||
assignSubmitBtn.addEventListener('click', async () => {
|
||||
const selectedRows = assignPeople.filter((p) => assignSelected.has(String(p.staffId)));
|
||||
const userIds = selectedRows.filter((p) => p.clinicUserId).map((p) => p.clinicUserId);
|
||||
const staffIds = selectedRows.filter((p) => !p.clinicUserId && p.staffId != null).map((p) => p.staffId);
|
||||
if (!userIds.length && !staffIds.length) return;
|
||||
assignStatusEl.textContent = 'Назначаем…';
|
||||
try {
|
||||
const r = await fetch(`/api/tests/${TEST_ID}/assign`, {
|
||||
method: 'POST',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ userIds, staffIds }),
|
||||
});
|
||||
const data = await r.json();
|
||||
if (!r.ok) throw new Error(data.error || 'Ошибка назначения');
|
||||
assignStatusEl.textContent = `Назначено: ${data.count ?? selectedRows.length}`;
|
||||
} catch (e) {
|
||||
assignStatusEl.textContent = e.message || 'Ошибка назначения';
|
||||
}
|
||||
});
|
||||
loadDirectory();
|
||||
}
|
||||
|
||||
if (visibilityBtn) {
|
||||
visibilityBtn.addEventListener('click', async () => {
|
||||
const next = !chainActiveEl.checked;
|
||||
try {
|
||||
const r = await fetch(`/api/tests/${TEST_ID}`, {
|
||||
method: 'PATCH',
|
||||
headers: { 'Content-Type': 'application/json' },
|
||||
body: JSON.stringify({ chainActive: next }),
|
||||
});
|
||||
const data = await r.json();
|
||||
if (!r.ok) throw new Error(data.error || 'Ошибка изменения видимости');
|
||||
chainActiveEl.checked = !!next;
|
||||
chainActive = !!next;
|
||||
visibilityBtn.textContent = chainActive ? 'Скрыть из списка' : 'Снова показать в списке';
|
||||
} catch (e) {
|
||||
alert(e.message || 'Ошибка изменения видимости');
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
Promise.all([
|
||||
fetch(`/api/tests/${TEST_ID}/versions`).then((r) => r.json()).catch(() => null),
|
||||
fetch(`/api/tests/${TEST_ID}/attempts`).then((r) => r.json()).catch(() => null),
|
||||
fetch(`/api/tests/${TEST_ID}/summary`).then((r) => r.json()).catch(() => null),
|
||||
]).then(([v, a, s]) => {
|
||||
if (v && Array.isArray(v.versions)) {
|
||||
renderVersions(v.versions);
|
||||
hasForkRisk = hasForkRisk || (v.versions.length > 1);
|
||||
if (typeof v.hasAttempts === 'boolean') {
|
||||
hasAnyAttempts = hasAnyAttempts || v.hasAttempts;
|
||||
hasForkRisk = hasForkRisk || v.hasAttempts;
|
||||
}
|
||||
}
|
||||
if (a && Array.isArray(a.attempts)) {
|
||||
renderAttempts(a.attempts);
|
||||
hasAnyAttempts = a.attempts.length > 0;
|
||||
}
|
||||
if (s && s.test) {
|
||||
if (introUpdatedEl) introUpdatedEl.textContent = fmtDt(s.test.updated_at || s.test.updatedAt);
|
||||
const versionEl = document.getElementById('intro-version');
|
||||
if (versionEl && s.test.version != null) versionEl.textContent = s.test.version;
|
||||
if (typeof s.test.hasAttempts === 'boolean') {
|
||||
hasAnyAttempts = hasAnyAttempts || s.test.hasAttempts;
|
||||
hasForkRisk = hasForkRisk || s.test.hasAttempts;
|
||||
}
|
||||
if (typeof s.test.versionCount === 'number') {
|
||||
hasForkRisk = hasForkRisk || s.test.versionCount > 1;
|
||||
}
|
||||
}
|
||||
updateForkBanner();
|
||||
});
|
||||
|
||||
loadInitial();
|
||||
resetBaselineDraft();
|
||||
root.addEventListener('input', scheduleDirtyCheck);
|
||||
root.addEventListener('change', scheduleDirtyCheck);
|
||||
})();
|
||||
|
||||
Reference in New Issue
Block a user