feat: show new-version changelog in the update banner

The updater already fetched the release notes (update.body) but never
displayed them. The banner now has a 'What's new' toggle that expands a
panel showing only the newest version's section (sliced out of the
cumulative release body), rendered as markdown, with a 'Full changelog
on GitHub' link to the releases page.
This commit is contained in:
Jure
2026-06-15 21:43:54 +02:00
parent 2e0af4e6bf
commit 08f231fb7e
+77 -34
View File
@@ -32,49 +32,92 @@ import { useCanSign } from "./stores/user";
import { getTheme, applyTheme } from "./lib/themes";
import { useUpdater } from "./hooks/useUpdater";
import { useKeyboardShortcuts } from "./hooks/useKeyboardShortcuts";
import { renderMarkdown } from "./lib/markdown";
// The updater's release body is a cumulative changelog: a `## Vega vX` header
// followed by a `### vX — title` section per version, newest first. For the
// "what's new" panel we want only the newest version's section. Returns the
// whole body unchanged if the format isn't recognized.
function latestChangelogSection(body: string): string {
const lines = body.split("\n");
const headings = lines.reduce<number[]>((acc, line, i) => {
if (/^###\s+v/i.test(line.trim())) acc.push(i);
return acc;
}, []);
if (headings.length === 0) return body.trim();
const start = headings[0];
const end = headings.length > 1 ? headings[1] : lines.length;
return lines.slice(start, end).join("\n").trim();
}
function UpdateBanner() {
const { available, version, installing, error, canSelfUpdate, kind, install, dismiss } = useUpdater();
const { available, version, body, installing, error, canSelfUpdate, kind, install, dismiss } = useUpdater();
const [copied, setCopied] = useState(false);
const [showChangelog, setShowChangelog] = useState(false);
if (!available) return null;
const aurCmd = "yay -S vega-nostr-git";
const changelog = body ? latestChangelogSection(body) : "";
return (
<div className="flex items-center justify-between px-4 py-2 bg-accent/10 border-b border-accent/30 text-[12px] shrink-0">
<span className="text-text">
Vega {version} is available.
{!canSelfUpdate && kind === "pacman" && (
<> Update with <code className="text-accent bg-bg-raised px-1 rounded-sm">{aurCmd}</code></>
)}
{error && <span className="text-danger ml-1">{error}</span>}
</span>
<div className="flex items-center gap-3">
{canSelfUpdate ? (
<button
onClick={install}
disabled={installing}
className="text-accent hover:text-accent-hover transition-colors disabled:opacity-50"
>
{installing ? "Installing…" : "Update & restart"}
</button>
) : kind === "pacman" ? (
<button
onClick={() => navigator.clipboard.writeText(aurCmd).then(() => setCopied(true)).catch(() => {})}
className="text-accent hover:text-accent-hover transition-colors"
>
{copied ? "Copied" : "Copy command"}
</button>
) : (
<button
onClick={() => openUrl("https://github.com/hoornet/vega/releases/latest").catch(() => {})}
className="text-accent hover:text-accent-hover transition-colors"
>
View release
</button>
)}
<button onClick={dismiss} aria-label="Dismiss update" className="text-text-dim hover:text-text transition-colors">×</button>
<div className="bg-accent/10 border-b border-accent/30 text-[12px] shrink-0">
<div className="flex items-center justify-between px-4 py-2">
<span className="text-text">
Vega {version} is available.
{!canSelfUpdate && kind === "pacman" && (
<> Update with <code className="text-accent bg-bg-raised px-1 rounded-sm">{aurCmd}</code></>
)}
{error && <span className="text-danger ml-1">{error}</span>}
</span>
<div className="flex items-center gap-3">
{changelog && (
<button
onClick={() => setShowChangelog((v) => !v)}
aria-expanded={showChangelog}
className="text-accent hover:text-accent-hover transition-colors"
>
{showChangelog ? "Hide changes" : "What's new"}
</button>
)}
{canSelfUpdate ? (
<button
onClick={install}
disabled={installing}
className="text-accent hover:text-accent-hover transition-colors disabled:opacity-50"
>
{installing ? "Installing…" : "Update & restart"}
</button>
) : kind === "pacman" ? (
<button
onClick={() => navigator.clipboard.writeText(aurCmd).then(() => setCopied(true)).catch(() => {})}
className="text-accent hover:text-accent-hover transition-colors"
>
{copied ? "Copied" : "Copy command"}
</button>
) : (
<button
onClick={() => openUrl("https://github.com/hoornet/vega/releases/latest").catch(() => {})}
className="text-accent hover:text-accent-hover transition-colors"
>
View release
</button>
)}
<button onClick={dismiss} aria-label="Dismiss update" className="text-text-dim hover:text-text transition-colors">×</button>
</div>
</div>
{/* What's-new panel — only the newest version's changes (issue: update changelog) */}
{showChangelog && changelog && (
<div className="px-4 pb-3 pt-1 max-h-72 overflow-y-auto border-t border-accent/20">
<div className="prose-article" dangerouslySetInnerHTML={{ __html: renderMarkdown(changelog) }} />
<button
onClick={() => openUrl("https://github.com/hoornet/vega/releases").catch(() => {})}
className="mt-3 text-accent hover:text-accent-hover transition-colors text-[11px]"
>
Full changelog on GitHub
</button>
</div>
)}
</div>
);
}