- Tabbed Wallet Settings modal (Cashu + Fedimint) and dual-balance wallet card - Buy a peer's paid file (ecash / node Lightning / on-chain / external QR) - Recovery-phrase reveal + backup section; onboarding seed retry resilience - NetBird HTTPS launch, remote-control two-finger scroll + external-open - Shared BackButton, single-v version label, mesh Bitcoin header toggles Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
205 lines
7.7 KiB
Vue
205 lines
7.7 KiB
Vue
<template>
|
|
<div class="h-[100dvh] flex items-center justify-center p-3 sm:p-4 md:p-6">
|
|
<div class="max-w-[800px] w-full max-h-full relative z-10 path-glass-container onb-scroll-container flex flex-col">
|
|
<!-- Header -->
|
|
<div class="text-center flex-shrink-0 px-3 sm:px-4 pt-4 sm:pt-6 pb-2 sm:pb-3">
|
|
<h1 class="text-xl sm:text-2xl md:text-[26px] font-semibold text-white/96 mb-1.5 drop-shadow-[0_2px_6px_rgba(0,0,0,0.4)]">
|
|
Your Recovery Seed
|
|
</h1>
|
|
<p class="text-xs sm:text-sm md:text-base text-white/75 leading-relaxed max-w-[600px] mx-auto">
|
|
Write down these 24 words in order. They are the only way to recover your node.
|
|
</p>
|
|
</div>
|
|
|
|
<!-- Scrollable Content -->
|
|
<div class="flex-1 overflow-y-auto overflow-x-hidden px-6 sm:px-8 min-h-0">
|
|
<div class="flex flex-col items-center gap-3 sm:gap-4 py-3">
|
|
<!-- Loading State -->
|
|
<div v-if="loading" class="text-center py-8">
|
|
<div class="flex justify-center mb-4">
|
|
<div class="w-16 h-16 rounded-full bg-white/10 flex items-center justify-center onb-lock-spin">
|
|
<svg class="w-8 h-8 text-orange-400" fill="none" stroke="currentColor" viewBox="0 0 24 24" stroke-width="2">
|
|
<path stroke-linecap="round" stroke-linejoin="round" d="M12 15v2m-6 4h12a2 2 0 002-2v-6a2 2 0 00-2-2H6a2 2 0 00-2 2v6a2 2 0 002 2zm10-10V7a4 4 0 00-8 0v4h8z" />
|
|
</svg>
|
|
</div>
|
|
</div>
|
|
<div v-if="waitingForServer" class="flex items-center justify-center gap-3 mb-2">
|
|
<p class="text-lg text-white/80">Server starting up</p>
|
|
<span class="text-sm text-white/40 font-mono tabular-nums">{{ elapsedDisplay }}</span>
|
|
</div>
|
|
<p v-if="waitingForServer" class="text-sm text-white/50">This usually takes 1-3 minutes after first boot</p>
|
|
<p v-else class="text-lg text-white/80">Generating your seed phrase...</p>
|
|
</div>
|
|
|
|
<!-- Error (genuine failure — server-starting hiccups retry silently) -->
|
|
<div v-if="errorMessage" class="text-center">
|
|
<p class="text-red-400 text-sm mb-3">{{ errorMessage }}</p>
|
|
<button
|
|
@click="generateSeed"
|
|
class="path-action-button path-action-button--continue mx-auto"
|
|
>
|
|
Try again
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Word Grid -->
|
|
<div v-if="words.length > 0" class="w-full max-w-[600px]">
|
|
<div class="grid grid-cols-2 sm:grid-cols-4 gap-1 sm:gap-1.5">
|
|
<div
|
|
v-for="(word, i) in words"
|
|
:key="i"
|
|
class="bg-black/60 rounded-lg px-2.5 py-1 sm:py-1.5 border border-white/10"
|
|
>
|
|
<span class="text-white/40 text-sm font-mono mr-1">{{ i + 1 }}.</span>
|
|
<span class="text-white/95 text-[1.05rem] font-mono">{{ word }}</span>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Warning -->
|
|
<div class="mt-3 bg-orange-500/10 border border-orange-500/20 rounded-lg px-3 py-2.5">
|
|
<p class="text-xs sm:text-sm text-orange-300/90">
|
|
Never share these words. Anyone with them controls your node, identities, and Bitcoin wallet.
|
|
</p>
|
|
</div>
|
|
|
|
<!-- Confirmation Checkbox -->
|
|
<label class="flex items-center justify-center gap-3 mt-3 cursor-pointer select-none">
|
|
<input
|
|
v-model="confirmed"
|
|
type="checkbox"
|
|
class="w-5 h-5 rounded border-white/20 bg-black/40 accent-orange-400"
|
|
/>
|
|
<span class="text-xs sm:text-sm text-white/80">I have written down these 24 words in a safe place</span>
|
|
</label>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Fixed Footer -->
|
|
<div v-if="words.length > 0" class="flex-shrink-0 flex justify-center px-3 sm:px-4 pt-3 pb-4 sm:pb-6">
|
|
<button
|
|
ref="continueButton"
|
|
@click="proceed"
|
|
:disabled="!confirmed"
|
|
class="path-action-button path-action-button--continue disabled:opacity-50"
|
|
>
|
|
Continue
|
|
</button>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, onMounted, onUnmounted, watch, nextTick } from 'vue'
|
|
import { useRouter } from 'vue-router'
|
|
import { rpcClient } from '@/api/rpc-client'
|
|
import { playNavSound } from '@/composables/useNavSounds'
|
|
|
|
const router = useRouter()
|
|
const continueButton = ref<HTMLButtonElement | null>(null)
|
|
const words = ref<string[]>([])
|
|
const confirmed = ref(false)
|
|
const loading = ref(false)
|
|
const waitingForServer = ref(false)
|
|
const errorMessage = ref('')
|
|
const elapsedDisplay = ref('0:00')
|
|
let retryTimer: ReturnType<typeof setTimeout> | null = null
|
|
let elapsedTimer: ReturnType<typeof setInterval> | null = null
|
|
let startTime = 0
|
|
|
|
function startElapsedTimer() {
|
|
startTime = Date.now()
|
|
elapsedTimer = setInterval(() => {
|
|
const secs = Math.floor((Date.now() - startTime) / 1000)
|
|
const m = Math.floor(secs / 60)
|
|
const s = secs % 60
|
|
elapsedDisplay.value = `${m}:${s.toString().padStart(2, '0')}`
|
|
}, 1000)
|
|
}
|
|
|
|
function stopTimers() {
|
|
if (retryTimer) { clearTimeout(retryTimer); retryTimer = null }
|
|
if (elapsedTimer) { clearInterval(elapsedTimer); elapsedTimer = null }
|
|
}
|
|
|
|
// Transient errors mean the backend is still booting (slow first boot) — we
|
|
// retry these silently. Anything else is a genuine failure the user should see.
|
|
function isServerStartingError(err: unknown): boolean {
|
|
const msg = err instanceof Error ? err.message : String(err)
|
|
return /502|503|504|timeout|fetch|network|Failed to fetch|Request failed/i.test(msg)
|
|
}
|
|
|
|
async function generateSeed() {
|
|
loading.value = true
|
|
errorMessage.value = ''
|
|
|
|
try {
|
|
// seed.generate is idempotent server-side, so retries are safe. Use a
|
|
// longer timeout than the default 15s — key derivation + disk writes can be
|
|
// slow on first boot, and we don't want a spurious abort to look like a
|
|
// failure.
|
|
const res = await rpcClient.call<{ words: string[] }>({ method: 'seed.generate', timeout: 30000 })
|
|
stopTimers()
|
|
words.value = res.words
|
|
loading.value = false
|
|
waitingForServer.value = false
|
|
} catch (err) {
|
|
loading.value = false
|
|
if (isServerStartingError(err)) {
|
|
// Backend not ready yet — keep waiting, retry silently.
|
|
if (!waitingForServer.value) {
|
|
waitingForServer.value = true
|
|
startElapsedTimer()
|
|
}
|
|
retryTimer = setTimeout(generateSeed, 4000)
|
|
} else {
|
|
// Genuine failure — stop the silent loop and surface it with a manual retry.
|
|
stopTimers()
|
|
waitingForServer.value = false
|
|
errorMessage.value = err instanceof Error ? err.message : 'Failed to generate seed'
|
|
}
|
|
}
|
|
}
|
|
|
|
watch(confirmed, (val) => {
|
|
if (val) {
|
|
nextTick(() => {
|
|
setTimeout(() => continueButton.value?.focus({ preventScroll: true }), 100)
|
|
})
|
|
}
|
|
})
|
|
|
|
onMounted(() => {
|
|
// Restore previously generated seed if navigating back (don't regenerate)
|
|
const saved = sessionStorage.getItem('_seed_words')
|
|
if (saved) {
|
|
try {
|
|
const parsed = JSON.parse(saved)
|
|
if (Array.isArray(parsed) && parsed.length === 24) {
|
|
words.value = parsed
|
|
return
|
|
}
|
|
} catch { /* regenerate */ }
|
|
}
|
|
generateSeed()
|
|
})
|
|
onUnmounted(() => { stopTimers() })
|
|
|
|
function proceed() {
|
|
playNavSound('action')
|
|
sessionStorage.setItem('_seed_words', JSON.stringify(words.value))
|
|
router.push('/onboarding/seed-verify').catch(() => {})
|
|
}
|
|
</script>
|
|
|
|
<style scoped>
|
|
.onb-lock-spin {
|
|
animation: onb-lock-pulse 1.2s ease-in-out infinite;
|
|
}
|
|
@keyframes onb-lock-pulse {
|
|
0%, 100% { transform: scale(1); opacity: 1; }
|
|
50% { transform: scale(1.08); opacity: 0.7; }
|
|
}
|
|
</style>
|