Files
claude-mnemonic/ui/src/components/Header.vue
T

171 lines
6.4 KiB
Vue

<script setup lang="ts">
import { ref } from 'vue'
import type { UpdateInfo, UpdateStatus } from '@/composables/useUpdate'
defineProps<{
isConnected: boolean
isProcessing: boolean
updateInfo: UpdateInfo | null
updateStatus: UpdateStatus
isUpdating: boolean
}>()
const emit = defineEmits<{
refresh: []
applyUpdate: []
}>()
const showUpdateModal = ref(false)
const isRestarting = ref(false)
const restartWorker = async () => {
isRestarting.value = true
try {
await fetch('/api/update/restart', { method: 'POST' })
// Wait a moment for the new process to start, then reload
setTimeout(() => {
globalThis.location.reload()
}, 2000)
} catch (error) {
console.error('Failed to restart:', error)
isRestarting.value = false
}
}
</script>
<template>
<header class="glass border-b border-white/10 sticky top-0 z-50">
<div class="max-w-7xl mx-auto px-4 py-4">
<div class="flex items-center justify-between">
<!-- Logo & Title -->
<div class="flex items-center gap-3">
<div class="w-10 h-10 rounded-xl bg-gradient-to-br from-claude-500 to-claude-700 flex items-center justify-center shadow-lg">
<i class="fas fa-brain text-xl text-white" />
</div>
<div>
<h1 class="text-xl font-bold text-white">Claude <span class="text-claude-400">Mnemonic</span></h1>
<p class="text-xs text-slate-400">Persistent Memory System</p>
</div>
</div>
<!-- Status & Actions -->
<div class="flex items-center gap-4">
<!-- Update Available Indicator -->
<div v-if="updateInfo?.available && !isUpdating && updateStatus.state === 'idle'" class="relative">
<button
class="flex items-center gap-2 px-3 py-1.5 rounded-lg bg-amber-500/20 border border-amber-500/50 text-amber-400 hover:bg-amber-500/30 transition-colors text-sm"
@click="showUpdateModal = true"
>
<i class="fas fa-arrow-circle-up" />
<span>v{{ updateInfo.latest_version }}</span>
</button>
</div>
<!-- Update In Progress -->
<div v-else-if="isUpdating" class="flex items-center gap-2 text-amber-400 text-sm">
<i class="fas fa-spinner animate-spin" />
<span>{{ updateStatus.message || 'Updating...' }}</span>
<span class="text-slate-500">{{ Math.round(updateStatus.progress * 100) }}%</span>
</div>
<!-- Update Complete -->
<div v-else-if="updateStatus.state === 'done'" class="flex items-center gap-2">
<button
class="flex items-center gap-2 px-3 py-1.5 rounded-lg bg-green-500/20 border border-green-500/50 text-green-400 hover:bg-green-500/30 transition-colors text-sm"
:disabled="isRestarting"
@click="restartWorker"
>
<i :class="isRestarting ? 'fas fa-spinner animate-spin' : 'fas fa-check-circle'" />
<span>{{ isRestarting ? 'Restarting...' : 'Restart' }}</span>
</button>
</div>
<!-- Update Error -->
<div v-else-if="updateStatus.state === 'error'" class="flex items-center gap-2 text-red-400 text-sm" :title="updateStatus.error">
<i class="fas fa-exclamation-circle" />
<span>Update failed</span>
</div>
<!-- Connection Status -->
<div class="flex items-center gap-2">
<span
class="w-2 h-2 rounded-full"
:class="[
isConnected ? 'bg-green-500' : 'bg-red-500',
isProcessing ? 'animate-pulse' : ''
]"
/>
<span class="text-sm text-slate-400">
{{ isConnected ? (isProcessing ? 'Processing' : 'Connected') : 'Disconnected' }}
</span>
</div>
<!-- Refresh Button -->
<button
class="p-2 rounded-lg bg-white/5 hover:bg-white/10 transition-colors text-slate-400 hover:text-white"
title="Refresh"
@click="emit('refresh')"
>
<i class="fas fa-rotate" />
</button>
</div>
</div>
</div>
<!-- Update Modal -->
<Teleport to="body">
<div v-if="showUpdateModal" class="fixed inset-0 z-50 flex items-center justify-center p-4">
<!-- Backdrop -->
<div class="absolute inset-0 bg-black/60 backdrop-blur-sm" @click="showUpdateModal = false" />
<!-- Modal -->
<div class="relative glass border border-white/10 rounded-2xl p-6 max-w-md w-full shadow-2xl">
<button
class="absolute top-4 right-4 text-slate-400 hover:text-white"
@click="showUpdateModal = false"
>
<i class="fas fa-times" />
</button>
<div class="text-center mb-4">
<div class="w-16 h-16 mx-auto mb-4 rounded-2xl bg-gradient-to-br from-amber-500 to-orange-600 flex items-center justify-center">
<i class="fas fa-arrow-circle-up text-3xl text-white" />
</div>
<h3 class="text-xl font-bold text-white mb-1">Update Available</h3>
<p class="text-slate-400 text-sm">
v{{ updateInfo?.current_version }} v{{ updateInfo?.latest_version }}
</p>
</div>
<!-- Release Notes -->
<div v-if="updateInfo?.release_notes" class="mb-4 max-h-48 overflow-y-auto">
<p class="text-slate-500 text-xs uppercase tracking-wide mb-2">What's new</p>
<div class="bg-slate-800/50 rounded-lg p-3 text-sm text-slate-300 whitespace-pre-wrap leading-relaxed">
{{ updateInfo.release_notes }}
</div>
</div>
<div class="space-y-3">
<button
class="w-full py-3 rounded-xl bg-amber-500 text-slate-900 font-semibold hover:bg-amber-400 transition-colors"
@click="emit('applyUpdate'); showUpdateModal = false"
>
Update Now
</button>
<button
class="w-full py-3 rounded-xl bg-white/5 text-slate-400 hover:bg-white/10 hover:text-white transition-colors"
@click="showUpdateModal = false"
>
Later
</button>
</div>
<p class="text-center text-slate-500 text-xs mt-4">
Updates are verified with cosign signatures
</p>
</div>
</div>
</Teleport>
</header>
</template>