/* ═══════════════════════════════════════════════════════ CHANGELOG — Mission Update Log Controller Groups entries by date into unified cards ═══════════════════════════════════════════════════════ */ (function () { 'use strict'; const root = document.getElementById('changelogRoot'); if (!root) return; function esc(s) { const d = document.createElement('div'); d.textContent = s || ''; return d.innerHTML; } function formatDate(dateStr) { const d = new Date(dateStr + 'T00:00:00'); const day = d.getDate().toString().padStart(2, '0'); const month = (d.getMonth() + 1).toString().padStart(2, '0'); const year = d.getFullYear(); return `${day}/${month}/${year}`; } function render(data) { const entries = data.entries || []; const totalChanges = entries.reduce((sum, e) => sum + (e.changes ? e.changes.length : 0), 0); const latestVersion = entries.length > 0 ? entries[0].version : '0.0.0'; // Group entries by date const grouped = []; const dateMap = {}; for (const entry of entries) { if (!dateMap[entry.date]) { dateMap[entry.date] = { date: entry.date, entries: [] }; grouped.push(dateMap[entry.date]); } dateMap[entry.date].entries.push(entry); } let html = ''; // Header html += `
`; html += `
MAINTENANCE LOG
`; html += `
// ALL MODIFICATIONS — LOGGED & VERIFIED
`; html += `
`; html += `CURRENT BUILD: v${esc(latestVersion)}`; html += `UPDATES: ${entries.length}`; html += `TOTAL CHANGES: ${totalChanges}`; html += `
`; // Timeline — grouped by date html += `
`; for (const group of grouped) { // Version range for this date const versions = group.entries.map(e => e.version); const versionRange = versions.length > 1 ? `v${esc(versions[versions.length - 1])} — v${esc(versions[0])}` : `v${esc(versions[0])}`; html += `
`; html += `
`; // Date header html += `
`; html += `${formatDate(group.date)}`; html += `${versionRange}`; html += `
`; // Each sub-entry within this date for (const entry of group.entries) { const badge = entry.category || 'update'; html += `
`; html += `
`; html += `v${esc(entry.version)}`; html += `${esc(badge)}`; html += `${esc(entry.title)}`; html += `
`; if (entry.changes && entry.changes.length > 0) { html += `
    `; for (const change of entry.changes) { html += `
  • ${esc(change)}
  • `; } html += `
`; } html += `
`; } html += `
`; } html += `
`; root.innerHTML = html; } // Load data root.innerHTML = '
LOADING UPDATE LOG...
'; fetch('/api/changelog') .then(res => { if (!res.ok) throw new Error('API ' + res.status); return res.json(); }) .then(data => render(data)) .catch(err => { root.innerHTML = '
⚠ FAILED TO LOAD UPDATE LOG
'; console.error('Changelog load error:', err); }); })();