/* discrepancy.js — page-scoped UI for /discrepancy.php */ (function () { 'use strict'; const els = {}; let lang = window.DBN_TOOLS_LANG || localStorage.getItem('dbn-ui-lang') || 'en'; let fileA = null; let fileB = null; let lastResult = null; const SLICE_DEFS = [ { id: 'child_welfare', label: 'Child Welfare' }, { id: 'echr', label: 'ECHR' }, { id: 'family_core', label: 'Family Law Core' }, { id: 'bufdir_guidance', label: 'Bufdir Guidance' }, { id: 'norwegian_courts', label: 'Norwegian Courts' }, { id: 'broader_legal', label: 'Broader Legal' }, ]; const STEP_LABELS = [ 'Classify documents', 'Extract parties', 'Build timelines', 'Cross-reference parties', 'Cross-reference timelines', 'Research questions', 'Retrieve legal context', 'Synthesize report', ]; const stepKeyToIndex = { doc_classify: 0, party_extract: 1, timeline_extract: 2, cross_parties: 3, cross_timelines: 4, sub_question_gen: 5, retrieval: 6, synthesis: 7, }; document.addEventListener('DOMContentLoaded', () => { if (!document.body.dataset.activeTool || document.body.dataset.activeTool !== 'discrepancy') return; Object.assign(els, { form: document.getElementById('dcForm'), status: document.getElementById('dcStatus'), runButton: document.getElementById('dcRunButton'), results: document.getElementById('dcResults'), traceList: document.getElementById('traceList'), langButtons: Array.from(document.querySelectorAll('#dcLangSwitcher .lang-btn')), tierRadios: Array.from(document.querySelectorAll('input[name="dcTier"]')), slices: Array.from(document.querySelectorAll('.adv-slice')), // File A zoneA: document.getElementById('dcZoneA'), inputA: document.getElementById('dcInputA'), promptA: document.getElementById('dcPromptA'), fileInfoA: document.getElementById('dcFileInfoA'), fileNameA: document.getElementById('dcFileNameA'), clearA: document.getElementById('dcClearA'), // File B zoneB: document.getElementById('dcZoneB'), inputB: document.getElementById('dcInputB'), promptB: document.getElementById('dcPromptB'), fileInfoB: document.getElementById('dcFileInfoB'), fileNameB: document.getElementById('dcFileNameB'), clearB: document.getElementById('dcClearB'), // Source modal modal: document.getElementById('dcSourceModal'), modalClose: document.getElementById('dcSourceModalClose'), modalTitle: document.getElementById('dcSourceModalTitle'), modalEyebrow: document.getElementById('dcSourceModalEyebrow'), modalMeta: document.getElementById('dcSourceModalMeta'), modalText: document.getElementById('dcSourceModalText'), }); if (!els.form) return; bindLang(); bindSlices(); bindUploadZone('A'); bindUploadZone('B'); bindModal(); els.form.addEventListener('submit', onSubmit); renderTrace(STEP_LABELS.map((label) => ({ label, detail: 'Waiting…', status: 'idle' }))); }); // ── Language ─────────────────────────────────────────────────────────────── function bindLang() { els.langButtons.forEach((b) => { b.classList.toggle('is-active', b.dataset.lang === lang); b.addEventListener('click', () => { els.langButtons.forEach((x) => x.classList.remove('is-active')); b.classList.add('is-active'); lang = b.dataset.lang || 'en'; localStorage.setItem('dbn-ui-lang', lang); }); }); } // ── Corpus slice toggles ─────────────────────────────────────────────────── function bindSlices() { els.slices.forEach((btn) => { btn.addEventListener('click', () => { const isOn = btn.classList.toggle('is-on'); btn.setAttribute('aria-pressed', isOn ? 'true' : 'false'); const badge = btn.querySelector('.dr-slice__badge'); if (badge) badge.textContent = isOn ? 'on' : 'off'; }); }); } function getSelectedSlices() { const out = {}; SLICE_DEFS.forEach((s) => { const btn = els.slices.find((b) => b.dataset.slice === s.id); out[s.id] = !!(btn && btn.classList.contains('is-on')); }); return out; } // ── File upload zones ────────────────────────────────────────────────────── function bindUploadZone(slot) { const zone = els['zone' + slot]; const input = els['input' + slot]; const prompt = els['prompt' + slot]; const info = els['fileInfo' + slot]; const nameEl = els['fileName' + slot]; const clearEl = els['clear' + slot]; if (!zone) return; const accept = (file) => { if (!file) return; if (file.size > 8 * 1024 * 1024) { setStatus(`${file.name} exceeds the 8 MB limit.`, 'error'); return; } const ext = (file.name.split('.').pop() || '').toLowerCase(); if (!['pdf', 'docx', 'txt'].includes(ext)) { setStatus(`${file.name} is not a supported file type (PDF, DOCX, TXT).`, 'error'); return; } if (slot === 'A') fileA = file; else fileB = file; nameEl.textContent = file.name; prompt.classList.add('is-hidden'); info.classList.remove('is-hidden'); zone.classList.remove('is-drop'); setStatus('', ''); }; input.addEventListener('change', (e) => { if (e.target.files && e.target.files[0]) accept(e.target.files[0]); }); zone.addEventListener('dragover', (e) => { e.preventDefault(); zone.classList.add('is-drop'); }); zone.addEventListener('dragleave', () => zone.classList.remove('is-drop')); zone.addEventListener('drop', (e) => { e.preventDefault(); zone.classList.remove('is-drop'); const f = e.dataTransfer?.files?.[0]; if (f) accept(f); }); clearEl?.addEventListener('click', () => { if (slot === 'A') fileA = null; else fileB = null; input.value = ''; info.classList.add('is-hidden'); prompt.classList.remove('is-hidden'); }); } // ── Form submission ──────────────────────────────────────────────────────── async function onSubmit(e) { e.preventDefault(); if (!fileA) { setStatus('Upload Document A (the earlier/original document) before running.', 'error'); return; } if (!fileB) { setStatus('Upload Document B (the later/comparison document) before running.', 'error'); return; } const tier = (els.tierRadios.find((r) => r.checked) || {}).value || 'quick'; const slices = getSelectedSlices(); const expectedDuration = tier === 'pro' ? '2-3 minutes' : '60-90 seconds'; setStatus(`Comparing documents… (${expectedDuration})`, 'busy'); els.runButton.disabled = true; els.results.innerHTML = `

Analysing…

Classifying both documents, extracting parties and timelines, then cross-referencing for discrepancies. Expect ${expectedDuration}.

`; const stepState = STEP_LABELS.map((label) => ({ label, detail: 'Queued', status: 'idle' })); renderTrace(stepState); const payload = { tier, language: lang, slices, use_my_case: (typeof window.dbnGetUseMyCase === 'function') ? window.dbnGetUseMyCase() : false, }; const form = new FormData(); form.append('payload', JSON.stringify(payload)); form.append('file_a', fileA); form.append('file_b', fileB); let response; try { response = await fetch('api/discrepancy.php', { method: 'POST', body: form, credentials: 'same-origin' }); } catch (err) { setStatus(`Network error: ${err.message || err}`, 'error'); els.runButton.disabled = false; return; } if (!response.ok || !response.body) { if (response.status === 402 || response.status === 429) { const d = await response.json().catch(() => ({})); if (typeof window.dbnFreeTierError === 'function') window.dbnFreeTierError(response.status, d); } else { setStatus(`Request failed (${response.status}).`, 'error'); } els.runButton.disabled = false; return; } const creditsRemaining = response.headers.get('X-Credits-Remaining'); if (creditsRemaining !== null && typeof window.dbnUpdateCredits === 'function') { window.dbnUpdateCredits(parseInt(creditsRemaining, 10)); } const reader = response.body.getReader(); const decoder = new TextDecoder('utf-8'); let buffer = ''; let finalResult = null; let errorEvent = null; // State for progressive rendering let metaARendered = false; let metaBRendered = false; let partiesARendered = false; let partiesBRendered = false; let tlARendered = false; let tlBRendered = false; function handleStreamEvent(evt) { if (!evt || !evt.event) return; if (evt.event === 'progress') { if (evt.detail) setStatus(evt.detail, 'busy'); return; } if (evt.event === 'start') { setStatus(`Comparing ${escapeHtml(evt.file_a || 'A')} ↔ ${escapeHtml(evt.file_b || 'B')}…`, 'busy'); return; } if (evt.event === 'step') { const idx = stepKeyToIndex[evt.step]; if (idx !== undefined) { if (evt.status === 'running' && stepState[idx].status !== 'running') { stepState[idx] = { label: evt.label || stepState[idx].label, detail: evt.detail || 'Running…', status: 'running' }; } else if (evt.status !== 'running') { stepState[idx] = { label: evt.label || stepState[idx].label, detail: evt.detail || stepState[idx].detail, status: evt.status || stepState[idx].status }; } renderTrace(stepState); } return; } if (evt.event === 'doc_a_meta' && !metaARendered) { renderDocMetaCard('A', evt.result || {}); metaARendered = true; return; } if (evt.event === 'doc_b_meta' && !metaBRendered) { renderDocMetaCard('B', evt.result || {}); metaBRendered = true; return; } if (evt.event === 'parties_a' && !partiesARendered && Array.isArray(evt.parties)) { renderPartiesPreview('A', evt.parties); partiesARendered = true; return; } if (evt.event === 'parties_b' && !partiesBRendered && Array.isArray(evt.parties)) { renderPartiesPreview('B', evt.parties); partiesBRendered = true; return; } if (evt.event === 'timeline_a' && !tlARendered && Array.isArray(evt.events)) { renderTimelinePreview('A', evt.events); tlARendered = true; return; } if (evt.event === 'timeline_b' && !tlBRendered && Array.isArray(evt.events)) { renderTimelinePreview('B', evt.events); tlBRendered = true; return; } if (evt.event === 'subq') { setStatus(`Retrieving ${evt.index}/${evt.total}: ${String(evt.question || '').slice(0, 80)}…`, 'busy'); return; } if (evt.event === 'final') { finalResult = evt.result; return; } if (evt.event === 'error') { errorEvent = evt; return; } } while (true) { let chunk; try { chunk = await reader.read(); } catch (err) { setStatus(`Stream error: ${err.message || err}`, 'error'); els.runButton.disabled = false; return; } const { done, value } = chunk; if (value) { buffer += decoder.decode(value, { stream: true }); const lines = buffer.split('\n'); buffer = lines.pop(); for (const line of lines) { const trimmed = line.trim(); if (!trimmed) continue; let evt; try { evt = JSON.parse(trimmed); } catch (_) { continue; } handleStreamEvent(evt); } } if (done) break; } if (errorEvent) { setStatus(`${errorEvent.code}: ${errorEvent.message}`, 'error'); els.runButton.disabled = false; const runningIdx = stepState.findIndex((s) => s.status === 'running'); if (runningIdx >= 0) { stepState[runningIdx] = { ...stepState[runningIdx], status: 'error', detail: errorEvent.message }; renderTrace(stepState); } return; } if (!finalResult) { setStatus('Stream ended without a final result.', 'error'); els.runButton.disabled = false; return; } lastResult = finalResult; if (typeof finalResult.balance === 'number' && typeof window.dbnUpdateCredits === 'function') { window.dbnUpdateCredits(finalResult.balance); } const meta = finalResult.trace_metadata || {}; setStatus( `Done · ${meta.conflict_count || 0} contradictions · ${meta.deleted_count || 0} deletions · ${meta.added_count || 0} additions · ${meta.source_count || 0} sources`, 'ok' ); els.runButton.disabled = false; renderTrace(finalResult.trace || []); renderFinalResults(finalResult); } // ── Progressive rendering ────────────────────────────────────────────────── function ensureResultsReady() { const emptyState = els.results.querySelector('.empty-state'); if (emptyState) emptyState.remove(); // Ensure the doc-meta pair container exists if (!els.results.querySelector('#dcDocMetaPair')) { const pair = document.createElement('div'); pair.id = 'dcDocMetaPair'; pair.className = 'dc-doc-meta-pair'; els.results.insertBefore(pair, els.results.firstChild); } } function renderDocMetaCard(slot, meta) { ensureResultsReady(); const pair = els.results.querySelector('#dcDocMetaPair'); if (!pair) return; const existing = pair.querySelector(`#dcMeta${slot}`); if (existing) existing.remove(); const card = document.createElement('div'); card.id = `dcMeta${slot}`; card.className = 'dc-doc-meta-card'; const fields = [ meta.doc_date ? ['Date', meta.doc_date] : null, meta.issuing_authority ? ['Authority', meta.issuing_authority] : null, meta.reference_number ? ['Ref', meta.reference_number] : null, ].filter(Boolean); card.innerHTML = `
Document ${slot} ${escapeHtml(meta.doc_type || ('Document ' + slot))}
${fields.length ? `
${fields.map(([k, v]) => `${escapeHtml(k)}: ${escapeHtml(String(v))}`).join('')}
` : ''} `; pair.appendChild(card); } function renderPartiesPreview(slot, parties) { if (!parties.length) return; ensureResultsReady(); const pair = els.results.querySelector('#dcDocMetaPair'); if (!pair) return; const metaCard = pair.querySelector(`#dcMeta${slot}`); if (!metaCard) return; const existing = metaCard.querySelector('.dc-parties-preview'); if (existing) existing.remove(); const preview = document.createElement('div'); preview.className = 'dc-parties-preview'; preview.innerHTML = `

${parties.length} party${parties.length === 1 ? '' : 'ies'} identified

${parties.slice(0, 6).map((p) => `${escapeHtml(p.name || p.role || '?')}`).join('')} ${parties.length > 6 ? `+${parties.length - 6} more` : ''}
`; metaCard.appendChild(preview); } function renderTimelinePreview(slot, events) { if (!events.length) return; ensureResultsReady(); const pair = els.results.querySelector('#dcDocMetaPair'); if (!pair) return; const metaCard = pair.querySelector(`#dcMeta${slot}`); if (!metaCard) return; const existing = metaCard.querySelector('.dc-timeline-preview'); if (existing) existing.remove(); const highCount = events.filter((e) => e.significance === 'high').length; const preview = document.createElement('div'); preview.className = 'dc-timeline-preview'; preview.innerHTML = `

${events.length} events · ${highCount} high-significance

`; metaCard.appendChild(preview); } // ── Final render ─────────────────────────────────────────────────────────── function renderFinalResults(data) { const sources = data.sources || []; const discrepancies = Array.isArray(data.critical_discrepancies) ? data.critical_discrepancies : []; const actions = Array.isArray(data.recommended_actions) ? data.recommended_actions : []; const uncertain = Array.isArray(data.what_remains_uncertain) ? data.what_remains_uncertain : []; const partiesDiff = data.parties_diff || {}; const tlDiff = data.timeline_diff || {}; const headline = data.headline_finding || ''; const nameA = data.doc_a_name || 'Document A'; const nameB = data.doc_b_name || 'Document B'; // Remove progressive doc meta pair — we'll re-render from authoritative data els.results.querySelector('#dcDocMetaPair')?.remove(); // Re-render doc meta pair from final data renderDocMetaCard('A', data.doc_a_meta || {}); renderDocMetaCard('B', data.doc_b_meta || {}); if ((data.parties_a || []).length) renderPartiesPreview('A', data.parties_a); if ((data.parties_b || []).length) renderPartiesPreview('B', data.parties_b); if ((data.timeline_a || []).length) renderTimelinePreview('A', data.timeline_a); if ((data.timeline_b || []).length) renderTimelinePreview('B', data.timeline_b); // Build tabs const conflicts = tlDiff.conflicts || []; const deletedEvents = tlDiff.in_a_only || []; const addedEvents = tlDiff.in_b_only || []; const procGaps = tlDiff.procedural_gaps || []; const narrative = tlDiff.narrative_shifts || {}; const pRemoved = partiesDiff.in_a_only || []; const pAdded = partiesDiff.in_b_only || []; const pChanged = partiesDiff.changed_between || []; const totalDiscrepancies = discrepancies.length; const tabCountStr = (n) => n > 0 ? ` ${n}` : ''; const finalHtml = ` ${headline ? `

Key finding

${escapeHtml(headline)}

` : ''}
${renderDiscrepanciesTab(discrepancies, sources)} ${actions.length ? `

Recommended actions

    ${actions.map((a) => `
  1. ${escapeHtml(String(a))}
  2. `).join('')}
` : ''} ${narrative.summary ? `

Narrative shift

${escapeHtml(narrative.summary)}

${(narrative.new_in_b || []).length ? `
New in ${escapeHtml(nameB)}:
    ${(narrative.new_in_b || []).map((s) => `
  • ${escapeHtml(String(s))}
  • `).join('')}
` : ''} ${(narrative.removed_from_b || []).length ? `
Removed from ${escapeHtml(nameB)}:
    ${(narrative.removed_from_b || []).map((s) => `
  • ${escapeHtml(String(s))}
  • `).join('')}
` : ''}
` : ''} ${uncertain.length ? `

What remains uncertain

    ${uncertain.map((u) => `
  • ${escapeHtml(String(u))}
  • `).join('')}
` : ''}
${renderPartiesTab(pRemoved, pAdded, pChanged, nameA, nameB)}
${renderTimelineTab(conflicts, deletedEvents, addedEvents, procGaps, nameA, nameB)}
${renderSourcesTab(sources)}

${escapeHtml(data.disclaimer || 'For legal information and preparation only — not legal advice. Verify all findings with a qualified lawyer.')}

`; const finalContainer = document.createElement('div'); finalContainer.innerHTML = finalHtml; while (finalContainer.firstChild) { els.results.appendChild(finalContainer.firstChild); } // Save-to-corpus button (appended after final results) const saveBtn = document.createElement('button'); saveBtn.type = 'button'; saveBtn.className = 'js-save-corpus secondary-button'; saveBtn.dataset.tool = 'discrepancy'; saveBtn.dataset.contentId = 'dcResults'; saveBtn.dataset.suggestedTitle = 'Discrepancy report'; saveBtn.textContent = 'Save to corpus'; saveBtn.style.marginTop = '16px'; els.results.appendChild(saveBtn); // Bind tabs els.results.querySelectorAll('.dc-tab').forEach((btn) => { btn.addEventListener('click', () => { const tab = btn.dataset.tab; els.results.querySelectorAll('.dc-tab').forEach((b) => b.classList.remove('is-active')); els.results.querySelectorAll('.dc-tab-panel').forEach((p) => p.classList.remove('is-active')); btn.classList.add('is-active'); const panel = els.results.querySelector(`.dc-tab-panel[data-panel="${tab}"]`); if (panel) panel.classList.add('is-active'); }); }); // Bind source card clicks els.results.querySelectorAll('.dr-source-card[data-source-n]').forEach((node) => { node.addEventListener('click', (e) => { if (e.target.closest('a')) return; const n = parseInt(node.dataset.sourceN, 10); const src = sources.find((s) => s.n === n); if (src) openModal(src); }); }); } // ── Tab content renderers ────────────────────────────────────────────────── function renderDiscrepanciesTab(discrepancies, sources) { if (!discrepancies.length) { return '

No critical discrepancies were identified in the synthesis.

'; } const sevClass = (s) => s === 'high' ? 'dc-sev--high' : (s === 'medium' ? 'dc-sev--medium' : 'dc-sev--low'); const catLabel = (c) => ({ timeline_conflict: 'Timeline', narrative_shift: 'Narrative', party_discrepancy: 'Party', procedural_gap: 'Procedure' }[c] || c); return `

Critical discrepancies (${discrepancies.length})

${discrepancies.map((d) => `
${escapeHtml(catLabel(d.category || ''))} ${escapeHtml(d.significance || 'low')}
${escapeHtml(d.title || '')}
Document A

${escapeHtml(d.document_a_says || '—')}

Document B

${escapeHtml(d.document_b_says || '—')}

${d.legal_relevance ? `` : ''}
`).join('')}
`; } function renderPartiesTab(removed, added, changed, nameA, nameB) { if (!removed.length && !added.length && !changed.length) { return '

No party discrepancies identified between the two documents.

'; } let html = '
'; if (removed.length) { html += `

Removed from ${escapeHtml(nameB)} (${removed.length})

${removed.map((p) => `
${escapeHtml(p.name || '?')}
${escapeHtml(p.role_in_a || '')}
${p.significance ? `
${escapeHtml(p.significance)}
` : ''}
`).join('')}
`; } if (added.length) { html += `

Added in ${escapeHtml(nameB)} (${added.length})

${added.map((p) => `
${escapeHtml(p.name || '?')}
${escapeHtml(p.role_in_b || '')}
${p.significance ? `
${escapeHtml(p.significance)}
` : ''}
`).join('')}
`; } if (changed.length) { html += `

Changed between versions (${changed.length})

${changed.map((p) => `
${escapeHtml(p.name || '?')}
${escapeHtml(p.in_a || '')}
${escapeHtml(p.in_b || '')}
${p.significance ? `
${escapeHtml(p.significance)}
` : ''}
`).join('')}
`; } html += '
'; return html; } function renderTimelineTab(conflicts, deleted, added, procGaps, nameA, nameB) { if (!conflicts.length && !deleted.length && !added.length && !procGaps.length) { return '

No timeline discrepancies identified between the two documents.

'; } const sigClass = (s) => `dc-sig--${s === 'high' ? 'high' : (s === 'medium' ? 'medium' : 'low')}`; let html = ''; if (conflicts.length) { html += `

Contradictions (${conflicts.length})

${conflicts.map((c) => `
${escapeHtml(c.significance || 'low')} ${c.date_a || c.date_b ? `${escapeHtml(c.date_a || '?')} / ${escapeHtml(c.date_b || '?')}` : ''}
${escapeHtml(nameA)}

${escapeHtml(c.doc_a_says || '—')}

${escapeHtml(nameB)}

${escapeHtml(c.doc_b_says || '—')}

${c.legal_significance ? `` : ''}
`).join('')}
`; } if (deleted.length) { html += `

Deleted from ${escapeHtml(nameB)} (${deleted.length})

${deleted.map((ev) => `
${escapeHtml(ev.significance || 'low')} ${ev.date ? `${escapeHtml(ev.date)}` : ''} ${ev.actor ? `${escapeHtml(ev.actor)}` : ''}

${escapeHtml(ev.description || '')}

${ev.legal_significance ? `` : ''}
`).join('')}
`; } if (added.length) { html += `

New in ${escapeHtml(nameB)} (${added.length})

${added.map((ev) => `
${escapeHtml(ev.significance || 'low')} ${ev.date ? `${escapeHtml(ev.date)}` : ''} ${ev.actor ? `${escapeHtml(ev.actor)}` : ''}

${escapeHtml(ev.description || '')}

${ev.legal_significance ? `` : ''}
`).join('')}
`; } if (procGaps.length) { html += `

Procedural gaps (${procGaps.length})

`; } return html; } function renderSourcesTab(sources) { if (!sources.length) { return '

No corpus sources retrieved. Enable corpus slices and re-run.

'; } return `

Legal context sources (${sources.length})

Click a card to expand · external link opens original source
${sources.map((s) => renderSourceCard(s)).join('')}
`; } function renderSourceCard(s) { const score = s.reranker_score != null ? s.reranker_score : s.similarity; const link = s.deep_link || s.source_url; const titleHtml = link ? `${escapeHtml(s.title || 'Untitled')} ` : `${escapeHtml(s.title || 'Untitled')}`; return `
${s.n}
${titleHtml}
${s.section ? `
${escapeHtml(s.section)}
` : ''}
${escapeHtml(s.package_or_corpus || 'corpus')} ${s.authority_label ? `${escapeHtml(s.authority_label)}` : ''} ${(s.matched_sub_questions || []).map((q) => `${escapeHtml(q)}`).join('')}

${escapeHtml(truncate(s.excerpt || '', 240))}

score
${score != null ? Number(score).toFixed(2) : '—'}
`; } // ── Source modal ─────────────────────────────────────────────────────────── function bindModal() { els.modalClose?.addEventListener('click', closeModal); els.modal?.addEventListener('click', (e) => { if (e.target === els.modal) closeModal(); }); document.addEventListener('keydown', (e) => { if (e.key === 'Escape' && els.modal && !els.modal.classList.contains('is-hidden')) closeModal(); }); } function closeModal() { els.modal?.classList.add('is-hidden'); } function openModal(source) { if (!source) return; els.modalEyebrow.textContent = 'Corpus source'; els.modalTitle.textContent = source.title || 'Source'; const metaRows = [ ['Number', `[${source.n}]`], source.section ? ['Section', source.section] : null, ['Corpus', source.package_or_corpus || '—'], source.authority_label ? ['Authority', source.authority_label] : null, source.similarity != null ? ['Similarity', String(source.similarity)] : null, source.reranker_score != null ? ['Rerank score', String(source.reranker_score)] : null, ].filter(Boolean); els.modalMeta.innerHTML = '
' + metaRows.map(([k, v]) => `
${escapeHtml(k)}
${escapeHtml(String(v))}
`).join('') + '
'; const chunkText = source.chunk_text || source.excerpt || ''; let html = chunkText ? `` : 'No excerpt available.'; els.modalText.innerHTML = html; const toggle = els.modalText.querySelector('.dr-modal-chunk-toggle'); const div = els.modalText.querySelector('.dr-modal-chunk-text'); toggle?.addEventListener('click', () => { const isHidden = div.classList.toggle('is-hidden'); toggle.textContent = isHidden ? 'Show matching text ▼' : 'Hide matching text ▲'; }); els.modal.classList.remove('is-hidden'); } // ── Trace rendering ──────────────────────────────────────────────────────── function renderTrace(steps) { if (!els.traceList) return; els.traceList.classList.add('is-rich'); els.traceList.innerHTML = steps.map((step, i) => { const statusClass = step.status === 'running' ? 'is-running' : step.status === 'complete' ? 'is-done' : step.status === 'warning' ? 'is-warning' : step.status === 'error' ? 'is-error' : ''; const marker = step.status === 'complete' ? '✓' : step.status === 'warning' ? '!' : step.status === 'error' ? '×' : (i + 1); return `
  • ${marker}
    ${escapeHtml(step.label || '')} ${escapeHtml(step.detail || '')}
  • `; }).join(''); } // ── Utility ──────────────────────────────────────────────────────────────── function setStatus(message, kind) { if (!els.status) return; els.status.textContent = message; els.status.style.color = kind === 'error' ? '#b41e1e' : kind === 'ok' ? 'var(--teal-dark)' : 'var(--muted)'; } function renderInlineCitations(escapedHtml, sources) { return escapedHtml.replace(/\[(\d+(?:\s*[-,]\s*\d+)*)\]/g, (_, group) => { const nums = expandCiteGroup(group); return nums.map((n) => `${n}`).join(''); }); } function expandCiteGroup(group) { const out = []; group.split(',').forEach((part) => { const range = part.trim().match(/^(\d+)\s*-\s*(\d+)$/); if (range) { for (let i = parseInt(range[1], 10); i <= parseInt(range[2], 10); i++) out.push(i); } else { const n = parseInt(part.trim(), 10); if (!Number.isNaN(n)) out.push(n); } }); return Array.from(new Set(out)); } function escapeHtml(s) { return String(s) .replace(/&/g, '&') .replace(//g, '>') .replace(/"/g, '"') .replace(/'/g, '''); } function truncate(s, n) { if (!s || s.length <= n) return s || ''; return s.slice(0, n - 1) + '…'; } })();