jedmund-svelte/src/lib/components/admin/AutoSaveStatus.svelte
Devin AI 62263e5785 lint: fix unused expressions by adding void operator (26 fixes)
Co-Authored-By: Justin Edmund <justin@jedmund.com>
2025-11-24 05:39:21 +00:00

117 lines
2.8 KiB
Svelte

<script lang="ts">
import type { AutoSaveStatus } from '$lib/admin/autoSave'
import { formatTimeAgo } from '$lib/utils/time'
interface Props {
statusStore?: { subscribe: (run: (v: AutoSaveStatus) => void) => () => void }
errorStore?: { subscribe: (run: (v: string | null) => void) => () => void }
status?: AutoSaveStatus
error?: string | null
lastSavedAt?: Date | string | null
showTimestamp?: boolean
compact?: boolean
}
let {
statusStore,
errorStore,
status: statusProp,
error: errorProp,
lastSavedAt,
showTimestamp = true,
compact = true
}: Props = $props()
// Support both old subscription-based stores and new reactive values
let status = $state<AutoSaveStatus>('idle')
let errorText = $state<string | null>(null)
let refreshKey = $state(0) // Used to force re-render for time updates
$effect(() => {
// If using direct props (new runes-based store)
if (statusProp !== undefined) {
status = statusProp
errorText = errorProp ?? null
return
}
// Otherwise use subscriptions (old store)
if (!statusStore) return
const unsub = statusStore.subscribe((v) => (status = v))
let unsubErr: (() => void) | null = null
if (errorStore) unsubErr = errorStore.subscribe((v) => (errorText = v))
return () => {
unsub()
if (unsubErr) unsubErr()
}
})
// Auto-refresh timestamp every 30 seconds
$effect(() => {
if (!lastSavedAt || !showTimestamp) return
const interval = setInterval(() => {
refreshKey++
}, 30000)
return () => clearInterval(interval)
})
const label = $derived.by(() => {
// Force dependency on refreshKey to trigger re-computation
void refreshKey
switch (status) {
case 'saving':
return 'Saving…'
case 'saved':
case 'idle':
return lastSavedAt && showTimestamp
? `Saved ${formatTimeAgo(lastSavedAt)}`
: 'All changes saved'
case 'offline':
return 'Offline'
case 'error':
return errorText ? `Error — ${errorText}` : 'Save failed'
default:
return ''
}
})
</script>
{#if label}
<div class="autosave-status" class:compact>
{#if status === 'saving'}
<span class="spinner" aria-hidden="true"></span>
{/if}
<span class="text">{label}</span>
</div>
{/if}
<style lang="scss">
.autosave-status {
display: inline-flex;
align-items: center;
gap: 6px;
color: $gray-40;
font-size: 0.875rem;
&.compact {
font-size: 0.75rem;
}
}
.spinner {
width: 12px;
height: 12px;
border: 2px solid $gray-80;
border-top-color: $gray-40;
border-radius: 50%;
animation: spin 0.9s linear infinite;
}
@keyframes spin {
to { transform: rotate(360deg); }
}
</style>