Files
dockhand/lib/components/WhatsNewModal.svelte
Jarek Krochmalski 62e3c6439e Initial commit
2025-12-28 21:16:03 +01:00

82 lines
2.3 KiB
Svelte

<script lang="ts">
import * as Dialog from '$lib/components/ui/dialog';
import { Button } from '$lib/components/ui/button';
import { Sparkles, Bug, Zap, CheckCircle, ScrollText } from 'lucide-svelte';
import { compareVersions } from '$lib/utils/version';
interface ChangelogEntry {
version: string;
date: string;
changes: Array<{ type: string; text: string }>;
imageTag?: string;
}
interface Props {
open: boolean;
version: string;
lastSeenVersion: string | null;
changelog: ChangelogEntry[];
onDismiss: () => void;
}
let { open = $bindable(), version, changelog, lastSeenVersion, onDismiss }: Props = $props();
// Filter to show versions newer than lastSeenVersion, limited to 3 most recent
const missedReleases = $derived(
changelog
.filter((r) => {
if (!lastSeenVersion) return true; // Show all if first time
return compareVersions(r.version, lastSeenVersion.replace(/^v/, '')) > 0;
})
.slice(0, 3)
);
function getChangeIcon(type: string) {
switch (type) {
case 'feature':
return { icon: Sparkles, class: 'text-green-500' };
case 'fix':
return { icon: Bug, class: 'text-amber-500' };
case 'improvement':
return { icon: Zap, class: 'text-green-500' };
default:
return { icon: CheckCircle, class: 'text-muted-foreground' };
}
}
</script>
<Dialog.Root bind:open>
<Dialog.Content class="max-w-3xl">
<Dialog.Header>
<Dialog.Title class="flex items-center gap-2">
<ScrollText class="w-5 h-5 text-muted-foreground" />
Dockhand has been updated to {version}
</Dialog.Title>
</Dialog.Header>
<div class="py-4 max-h-[60vh] overflow-y-auto space-y-6">
{#each missedReleases as release}
<div>
<h3 class="font-semibold text-sm mb-2 flex items-center gap-2">
<span>v{release.version}</span>
<span class="text-muted-foreground font-normal">({release.date})</span>
</h3>
<div class="space-y-1.5 ml-1">
{#each release.changes as change}
{@const { icon: Icon, class: iconClass } = getChangeIcon(change.type)}
<div class="flex items-start gap-2">
<Icon class="w-4 h-4 mt-0.5 shrink-0 {iconClass}" />
<span class="text-sm">{change.text}</span>
</div>
{/each}
</div>
</div>
{/each}
</div>
<Dialog.Footer>
<Button onclick={onDismiss}>Got it</Button>
</Dialog.Footer>
</Dialog.Content>
</Dialog.Root>