2026-01-24 22:59:20 +00:00
|
|
|
<template>
|
|
|
|
|
<div>
|
2026-02-17 15:03:34 +00:00
|
|
|
<div class="mb-8 flex flex-col md:flex-row md:items-center md:justify-between gap-4">
|
|
|
|
|
<div>
|
|
|
|
|
<h1 class="text-3xl font-bold text-white mb-2 drop-shadow-[0_2px_8px_rgba(0,0,0,0.6)]">Settings</h1>
|
|
|
|
|
<p class="text-white/80">Configure your Archipelago experience</p>
|
|
|
|
|
</div>
|
|
|
|
|
<!-- Controller indicator - Mobile only (desktop shows in sidebar) -->
|
|
|
|
|
<div class="md:hidden">
|
|
|
|
|
<ControllerIndicator />
|
|
|
|
|
</div>
|
2026-01-24 22:59:20 +00:00
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- Account Section -->
|
|
|
|
|
<div class="path-option-card cursor-default px-6 py-6 mb-6">
|
|
|
|
|
<h2 class="text-xl font-semibold text-white/96 mb-6">Account</h2>
|
|
|
|
|
|
|
|
|
|
<!-- Info Grid -->
|
|
|
|
|
<div class="grid grid-cols-1 md:grid-cols-2 gap-4 mb-6">
|
|
|
|
|
<!-- Server Name Card -->
|
|
|
|
|
<div class="bg-black/20 rounded-xl px-5 py-4 border border-white/10">
|
|
|
|
|
<div class="flex items-center gap-3 mb-2">
|
|
|
|
|
<svg class="w-5 h-5 text-white/70" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M5 12h14M5 12a2 2 0 01-2-2V6a2 2 0 012-2h14a2 2 0 012 2v4a2 2 0 01-2 2M5 12a2 2 0 00-2 2v4a2 2 0 002 2h14a2 2 0 002-2v-4a2 2 0 00-2-2m-2-4h.01M17 16h.01" />
|
|
|
|
|
</svg>
|
|
|
|
|
<p class="text-xs font-semibold text-white/60 uppercase tracking-wide">Server Name</p>
|
|
|
|
|
</div>
|
|
|
|
|
<p class="text-lg font-semibold text-white/95">{{ serverName }}</p>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- Version Card -->
|
|
|
|
|
<div class="bg-black/20 rounded-xl px-5 py-4 border border-white/10">
|
|
|
|
|
<div class="flex items-center gap-3 mb-2">
|
|
|
|
|
<svg class="w-5 h-5 text-white/70" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M7 7h.01M7 3h5c.512 0 1.024.195 1.414.586l7 7a2 2 0 010 2.828l-7 7a2 2 0 01-2.828 0l-7-7A1.994 1.994 0 013 12V7a4 4 0 014-4z" />
|
|
|
|
|
</svg>
|
|
|
|
|
<p class="text-xs font-semibold text-white/60 uppercase tracking-wide">Version</p>
|
|
|
|
|
</div>
|
|
|
|
|
<p class="text-lg font-semibold text-white/95">{{ version }}</p>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- Session Card -->
|
|
|
|
|
<div class="bg-black/20 rounded-xl px-5 py-4 border border-white/10 md:col-span-2">
|
|
|
|
|
<div class="flex items-center gap-3 mb-2">
|
|
|
|
|
<svg class="w-5 h-5 text-green-400" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M9 12l2 2 4-4m6 2a9 9 0 11-18 0 9 9 0 0118 0z" />
|
|
|
|
|
</svg>
|
|
|
|
|
<p class="text-xs font-semibold text-white/60 uppercase tracking-wide">Session Status</p>
|
|
|
|
|
</div>
|
|
|
|
|
<p class="text-base font-medium text-white/90">Currently logged in</p>
|
|
|
|
|
</div>
|
2026-02-17 15:03:34 +00:00
|
|
|
|
|
|
|
|
<!-- Identity Card: DID + Tor Address (onion below DID, with copy) -->
|
|
|
|
|
<div v-if="userDid || serverTorAddress" class="bg-black/20 rounded-xl px-5 py-4 border border-white/10 md:col-span-2 space-y-4">
|
|
|
|
|
<!-- DID -->
|
|
|
|
|
<div v-if="userDid">
|
|
|
|
|
<div class="flex items-center gap-3 mb-2">
|
|
|
|
|
<svg class="w-5 h-5 text-amber-400 shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 7a2 2 0 012 2m4 0a6 6 0 01-7.743 5.743L11 17H9v2H7v2H4a1 1 0 01-1-1v-2.586a1 1 0 01.293-.707l5.964-5.964A6 6 0 1121 9z" />
|
|
|
|
|
</svg>
|
|
|
|
|
<p class="text-xs font-semibold text-white/60 uppercase tracking-wide">Your DID</p>
|
|
|
|
|
</div>
|
|
|
|
|
<p class="text-sm font-mono text-white/90 break-all" :title="userDid">{{ userDid }}</p>
|
|
|
|
|
<p class="text-xs text-white/50 mt-1">Decentralized identifier for passwordless auth</p>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- Tor / Onion Address (below DID, with copy button) -->
|
|
|
|
|
<div v-if="serverTorAddress" :class="userDid ? 'pt-4 border-t border-white/10' : ''">
|
|
|
|
|
<div class="flex items-center justify-between gap-2 mb-2">
|
|
|
|
|
<div class="flex items-center gap-3 min-w-0">
|
|
|
|
|
<svg class="w-5 h-5 text-amber-400 shrink-0" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M21 12a9 9 0 01-9 9m9-9a9 9 0 00-9-9m9 9H3m9 9a9 9 0 01-9-9m9 9c1.657 0 3-4.03 3-9s-1.343-9-3-9m0 18c-1.657 0-3-4.03-3-9s1.343-9 3-9m-9 9a9 9 0 019-9" />
|
|
|
|
|
</svg>
|
|
|
|
|
<p class="text-xs font-semibold text-white/60 uppercase tracking-wide">Node .onion Address</p>
|
|
|
|
|
</div>
|
|
|
|
|
<button
|
|
|
|
|
@click="copyOnionAddress"
|
|
|
|
|
class="shrink-0 px-3 py-1.5 rounded-lg glass-button glass-button-sm text-xs font-medium text-white/90 hover:text-white transition-colors flex items-center gap-1.5"
|
|
|
|
|
>
|
|
|
|
|
<svg v-if="!copiedOnion" class="w-4 h-4" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M8 16H6a2 2 0 01-2-2V6a2 2 0 012-2h8a2 2 0 012 2v2m-6 12h8a2 2 0 002-2v-8a2 2 0 00-2-2h-8a2 2 0 00-2 2v8a2 2 0 002 2z" />
|
|
|
|
|
</svg>
|
|
|
|
|
<span v-else class="text-green-400 text-xs">Copied</span>
|
|
|
|
|
<span v-if="!copiedOnion">Copy</span>
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
<p class="text-sm font-mono text-amber-400/90 break-all" :title="serverTorAddress">{{ serverTorAddress }}</p>
|
|
|
|
|
<p class="text-xs text-white/50 mt-1">Onion address for node interface and peer discovery over Tor</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
<!-- Change Password -->
|
2026-02-18 11:29:05 +00:00
|
|
|
<div data-controller-container tabindex="0" class="mb-6">
|
2026-02-17 15:03:34 +00:00
|
|
|
<button
|
|
|
|
|
@click="showChangePasswordModal = true"
|
|
|
|
|
class="w-full flex items-center justify-center gap-2 mb-4 px-4 py-2 rounded-lg border border-orange-500/50 text-orange-400 font-medium hover:bg-orange-500/10 transition-colors"
|
|
|
|
|
>
|
|
|
|
|
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 7a2 2 0 012 2m4 0a6 6 0 01-7.743 5.743L11 17H9v2H7v2H4a1 1 0 01-1-1v-2.586a1 1 0 01.293-.707l5.964-5.964A6 6 0 1121 9z" />
|
|
|
|
|
</svg>
|
|
|
|
|
<span>Change Password</span>
|
|
|
|
|
</button>
|
2026-01-24 22:59:20 +00:00
|
|
|
</div>
|
|
|
|
|
|
2026-02-17 15:03:34 +00:00
|
|
|
<!-- Change Password Modal -->
|
|
|
|
|
<Teleport to="body">
|
|
|
|
|
<div
|
|
|
|
|
v-if="showChangePasswordModal"
|
|
|
|
|
class="fixed inset-0 z-50 flex items-center justify-center p-4 bg-black/60 backdrop-blur-sm"
|
2026-02-17 22:10:38 +00:00
|
|
|
@click.self="closeChangePasswordModal()"
|
2026-02-17 15:03:34 +00:00
|
|
|
>
|
2026-02-17 22:10:38 +00:00
|
|
|
<div ref="changePasswordModalRef" class="glass-card p-6 max-w-md w-full">
|
2026-02-17 15:03:34 +00:00
|
|
|
<h3 class="text-lg font-semibold text-white mb-4">Change Password</h3>
|
|
|
|
|
<p class="text-white/70 text-sm mb-4">Updates both web login and SSH access. Use a strong password (12+ chars, upper, lower, digit, special).</p>
|
|
|
|
|
<form @submit.prevent="handleChangePassword" class="space-y-4">
|
|
|
|
|
<div>
|
|
|
|
|
<label class="block text-sm font-medium text-white/80 mb-2">Current Password</label>
|
|
|
|
|
<input
|
|
|
|
|
v-model="changePasswordForm.currentPassword"
|
|
|
|
|
type="password"
|
|
|
|
|
required
|
|
|
|
|
autocomplete="current-password"
|
|
|
|
|
class="w-full px-3 py-2 rounded-lg bg-white/10 text-white border border-white/20 focus:border-orange-500 focus:ring-1 focus:ring-orange-500"
|
|
|
|
|
placeholder="Enter current password"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
<div>
|
|
|
|
|
<label class="block text-sm font-medium text-white/80 mb-2">New Password</label>
|
|
|
|
|
<input
|
|
|
|
|
v-model="changePasswordForm.newPassword"
|
|
|
|
|
type="password"
|
|
|
|
|
required
|
|
|
|
|
autocomplete="new-password"
|
|
|
|
|
class="w-full px-3 py-2 rounded-lg bg-white/10 text-white border border-white/20 focus:border-orange-500 focus:ring-1 focus:ring-orange-500"
|
|
|
|
|
placeholder="12+ chars, upper, lower, digit, special"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
<div>
|
|
|
|
|
<label class="block text-sm font-medium text-white/80 mb-2">Confirm New Password</label>
|
|
|
|
|
<input
|
|
|
|
|
v-model="changePasswordForm.confirmPassword"
|
|
|
|
|
type="password"
|
|
|
|
|
required
|
|
|
|
|
autocomplete="new-password"
|
|
|
|
|
class="w-full px-3 py-2 rounded-lg bg-white/10 text-white border border-white/20 focus:border-orange-500 focus:ring-1 focus:ring-orange-500"
|
|
|
|
|
placeholder="Re-enter new password"
|
|
|
|
|
/>
|
|
|
|
|
</div>
|
|
|
|
|
<label class="flex items-center gap-2 text-sm text-white/80">
|
|
|
|
|
<input v-model="changePasswordForm.alsoChangeSsh" type="checkbox" class="rounded border-white/30" />
|
|
|
|
|
Also update SSH password (recommended)
|
|
|
|
|
</label>
|
|
|
|
|
<p v-if="changePasswordError" class="text-sm text-red-400">{{ changePasswordError }}</p>
|
|
|
|
|
<p v-if="changePasswordSuccess" class="text-sm text-green-400">{{ changePasswordSuccess }}</p>
|
|
|
|
|
<div class="flex gap-3 pt-2">
|
|
|
|
|
<button
|
|
|
|
|
type="submit"
|
|
|
|
|
:disabled="changingPassword"
|
|
|
|
|
class="flex-1 px-4 py-2 rounded-lg bg-orange-500 text-white font-medium hover:bg-orange-600 disabled:opacity-50 disabled:cursor-not-allowed transition-colors"
|
|
|
|
|
>
|
|
|
|
|
{{ changingPassword ? 'Updating...' : 'Update Password' }}
|
|
|
|
|
</button>
|
|
|
|
|
<button
|
|
|
|
|
type="button"
|
|
|
|
|
@click="closeChangePasswordModal"
|
|
|
|
|
class="px-4 py-2 rounded-lg bg-white/10 text-white font-medium hover:bg-white/20 transition-colors"
|
|
|
|
|
>
|
|
|
|
|
Cancel
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</form>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</Teleport>
|
|
|
|
|
|
2026-01-24 22:59:20 +00:00
|
|
|
<!-- Logout Button -->
|
|
|
|
|
<button
|
|
|
|
|
@click="handleLogout"
|
|
|
|
|
class="w-full path-action-button path-action-button--continue flex items-center justify-center gap-2"
|
|
|
|
|
>
|
|
|
|
|
<svg class="w-5 h-5" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M17 16l4-4m0 0l-4-4m4 4H7m6 4v1a3 3 0 01-3 3H6a3 3 0 01-3-3V7a3 3 0 013-3h4a3 3 0 013 3v1" />
|
|
|
|
|
</svg>
|
|
|
|
|
<span>Logout</span>
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
|
2026-03-04 07:09:31 +00:00
|
|
|
<!-- Interface Mode Section -->
|
|
|
|
|
<div class="path-option-card cursor-default px-6 py-6 mb-6">
|
|
|
|
|
<h2 class="text-xl font-semibold text-white/96 mb-2">Interface Mode</h2>
|
|
|
|
|
<p class="text-sm text-white/60 mb-6">Choose how you want to interact with your node.</p>
|
|
|
|
|
|
|
|
|
|
<div class="grid grid-cols-1 md:grid-cols-3 gap-4">
|
|
|
|
|
<button
|
|
|
|
|
v-for="m in interfaceModes"
|
|
|
|
|
:key="m.id"
|
|
|
|
|
@click="uiMode.setMode(m.id)"
|
|
|
|
|
class="path-option-card text-left p-5"
|
|
|
|
|
:class="{ 'path-option-card--selected': uiMode.mode === m.id }"
|
|
|
|
|
>
|
|
|
|
|
<div class="flex items-center gap-3 mb-3">
|
|
|
|
|
<svg class="w-6 h-6 text-white/70" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path
|
|
|
|
|
v-for="(path, index) in m.iconPaths"
|
|
|
|
|
:key="index"
|
|
|
|
|
stroke-linecap="round"
|
|
|
|
|
stroke-linejoin="round"
|
|
|
|
|
stroke-width="2"
|
|
|
|
|
:d="path"
|
|
|
|
|
/>
|
|
|
|
|
</svg>
|
|
|
|
|
<h3 class="text-lg font-semibold text-white/96">{{ m.label }}</h3>
|
|
|
|
|
</div>
|
|
|
|
|
<p class="text-sm text-white/60 leading-relaxed">{{ m.description }}</p>
|
|
|
|
|
</button>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
|
2026-01-24 22:59:20 +00:00
|
|
|
<!-- System Section -->
|
|
|
|
|
<div class="path-option-card cursor-default px-6 py-6">
|
|
|
|
|
<h2 class="text-xl font-semibold text-white/96 mb-4">System</h2>
|
|
|
|
|
<div class="text-center py-8">
|
|
|
|
|
<svg class="w-12 h-12 mx-auto text-white/40 mb-3" fill="none" stroke="currentColor" viewBox="0 0 24 24">
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z" />
|
|
|
|
|
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 12a3 3 0 11-6 0 3 3 0 016 0z" />
|
|
|
|
|
</svg>
|
|
|
|
|
<p class="text-white/70">Additional settings coming soon</p>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</div>
|
|
|
|
|
</template>
|
|
|
|
|
|
|
|
|
|
<script setup lang="ts">
|
2026-02-17 15:03:34 +00:00
|
|
|
import { computed, ref, onMounted } from 'vue'
|
2026-01-24 22:59:20 +00:00
|
|
|
import { useRouter } from 'vue-router'
|
|
|
|
|
import { useAppStore } from '../stores/app'
|
2026-03-04 07:09:31 +00:00
|
|
|
import { useUIModeStore } from '@/stores/uiMode'
|
2026-02-17 15:03:34 +00:00
|
|
|
import ControllerIndicator from '@/components/ControllerIndicator.vue'
|
|
|
|
|
import { rpcClient } from '@/api/rpc-client'
|
2026-02-17 22:10:38 +00:00
|
|
|
import { useModalKeyboard } from '@/composables/useModalKeyboard'
|
2026-03-04 07:09:31 +00:00
|
|
|
import type { UIMode } from '@/types/api'
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
const router = useRouter()
|
|
|
|
|
const store = useAppStore()
|
2026-03-04 07:09:31 +00:00
|
|
|
const uiMode = useUIModeStore()
|
|
|
|
|
|
|
|
|
|
const interfaceModes: { id: UIMode; label: string; description: string; iconPaths: string[] }[] = [
|
|
|
|
|
{
|
|
|
|
|
id: 'easy',
|
|
|
|
|
label: 'Easy',
|
|
|
|
|
description: 'Goal-based interface. Choose what you want to do, and the system handles the rest.',
|
|
|
|
|
iconPaths: ['M14.828 14.828a4 4 0 01-5.656 0M9 10h.01M15 10h.01M21 12a9 9 0 11-18 0 9 9 0 0118 0z'],
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
id: 'gamer',
|
|
|
|
|
label: 'Pro',
|
|
|
|
|
description: 'Full control over all services. Configure everything manually with all technical details.',
|
|
|
|
|
iconPaths: ['M10.325 4.317c.426-1.756 2.924-1.756 3.35 0a1.724 1.724 0 002.573 1.066c1.543-.94 3.31.826 2.37 2.37a1.724 1.724 0 001.065 2.572c1.756.426 1.756 2.924 0 3.35a1.724 1.724 0 00-1.066 2.573c.94 1.543-.826 3.31-2.37 2.37a1.724 1.724 0 00-2.572 1.065c-.426 1.756-2.924 1.756-3.35 0a1.724 1.724 0 00-2.573-1.066c-1.543.94-3.31-.826-2.37-2.37a1.724 1.724 0 00-1.065-2.572c-1.756-.426-1.756-2.924 0-3.35a1.724 1.724 0 001.066-2.573c-.94-1.543.826-3.31 2.37-2.37.996.608 2.296.07 2.572-1.065z', 'M15 12a3 3 0 11-6 0 3 3 0 016 0z'],
|
|
|
|
|
},
|
|
|
|
|
{
|
|
|
|
|
id: 'chat',
|
|
|
|
|
label: 'Chat',
|
|
|
|
|
description: 'Conversational AI interface. Manage your node through natural language. Coming soon.',
|
|
|
|
|
iconPaths: ['M8 12h.01M12 12h.01M16 12h.01M21 12c0 4.418-4.03 8-9 8a9.863 9.863 0 01-4.255-.949L3 20l1.395-3.72C3.512 15.042 3 13.574 3 12c0-4.418 4.03-8 9-8s9 3.582 9 8z'],
|
|
|
|
|
},
|
|
|
|
|
]
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
const serverName = computed(() => store.serverName)
|
|
|
|
|
const version = computed(() => store.serverInfo?.version || '0.0.0')
|
2026-02-17 15:03:34 +00:00
|
|
|
const serverTorAddressFromStore = computed(() => store.serverInfo?.['tor-address'] || null)
|
|
|
|
|
const torAddressFromRpc = ref<string | null>(null)
|
|
|
|
|
const serverTorAddress = computed(() => serverTorAddressFromStore.value || torAddressFromRpc.value)
|
|
|
|
|
const userDid = computed(() => {
|
|
|
|
|
try {
|
|
|
|
|
return localStorage.getItem('neode_did') || null
|
|
|
|
|
} catch {
|
|
|
|
|
return null
|
|
|
|
|
}
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
const copiedOnion = ref(false)
|
|
|
|
|
const showChangePasswordModal = ref(false)
|
2026-02-17 22:10:38 +00:00
|
|
|
const changePasswordModalRef = ref<HTMLElement | null>(null)
|
|
|
|
|
const changePasswordRestoreFocusRef = ref<HTMLElement | null>(null)
|
|
|
|
|
useModalKeyboard(changePasswordModalRef, showChangePasswordModal, closeChangePasswordModal, { restoreFocusRef: changePasswordRestoreFocusRef })
|
2026-02-17 15:03:34 +00:00
|
|
|
const changingPassword = ref(false)
|
|
|
|
|
const changePasswordError = ref('')
|
|
|
|
|
const changePasswordSuccess = ref('')
|
|
|
|
|
const changePasswordForm = ref({
|
|
|
|
|
currentPassword: '',
|
|
|
|
|
newPassword: '',
|
|
|
|
|
confirmPassword: '',
|
|
|
|
|
alsoChangeSsh: true,
|
|
|
|
|
})
|
|
|
|
|
|
|
|
|
|
function validatePasswordStrength(pw: string): string | null {
|
|
|
|
|
if (pw.length < 12) return 'Password must be at least 12 characters'
|
|
|
|
|
if (!/[A-Z]/.test(pw)) return 'Password must contain at least one uppercase letter'
|
|
|
|
|
if (!/[a-z]/.test(pw)) return 'Password must contain at least one lowercase letter'
|
|
|
|
|
if (!/\d/.test(pw)) return 'Password must contain at least one digit'
|
|
|
|
|
if (!/[^A-Za-z0-9]/.test(pw)) return 'Password must contain at least one special character (!@#$%^&* etc.)'
|
|
|
|
|
return null
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
async function handleChangePassword() {
|
|
|
|
|
changePasswordError.value = ''
|
|
|
|
|
changePasswordSuccess.value = ''
|
|
|
|
|
const { currentPassword, newPassword, confirmPassword, alsoChangeSsh } = changePasswordForm.value
|
|
|
|
|
if (!currentPassword || !newPassword || !confirmPassword) {
|
|
|
|
|
changePasswordError.value = 'All fields are required'
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if (newPassword !== confirmPassword) {
|
|
|
|
|
changePasswordError.value = 'New passwords do not match'
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
const strengthError = validatePasswordStrength(newPassword)
|
|
|
|
|
if (strengthError) {
|
|
|
|
|
changePasswordError.value = strengthError
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
changingPassword.value = true
|
|
|
|
|
try {
|
|
|
|
|
await rpcClient.changePassword({
|
|
|
|
|
currentPassword,
|
|
|
|
|
newPassword,
|
|
|
|
|
alsoChangeSsh,
|
|
|
|
|
})
|
|
|
|
|
changePasswordSuccess.value = 'Password updated successfully. Use the new password for login and SSH.'
|
|
|
|
|
changePasswordForm.value = { currentPassword: '', newPassword: '', confirmPassword: '', alsoChangeSsh: true }
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
closeChangePasswordModal()
|
|
|
|
|
}, 2000)
|
|
|
|
|
} catch (e) {
|
|
|
|
|
changePasswordError.value = e instanceof Error ? e.message : 'Failed to change password'
|
|
|
|
|
} finally {
|
|
|
|
|
changingPassword.value = false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-01 18:07:35 +00:00
|
|
|
let copiedTimer: ReturnType<typeof setTimeout> | null = null
|
|
|
|
|
|
2026-02-17 15:03:34 +00:00
|
|
|
async function copyOnionAddress() {
|
|
|
|
|
const addr = serverTorAddress.value
|
|
|
|
|
if (!addr) return
|
|
|
|
|
try {
|
|
|
|
|
await navigator.clipboard.writeText(addr)
|
|
|
|
|
} catch {
|
|
|
|
|
const ta = document.createElement('textarea')
|
|
|
|
|
ta.value = addr
|
2026-03-01 18:07:35 +00:00
|
|
|
ta.style.position = 'fixed'
|
|
|
|
|
ta.style.opacity = '0'
|
2026-02-17 15:03:34 +00:00
|
|
|
document.body.appendChild(ta)
|
|
|
|
|
ta.select()
|
|
|
|
|
document.execCommand('copy')
|
|
|
|
|
document.body.removeChild(ta)
|
|
|
|
|
}
|
2026-03-01 18:07:35 +00:00
|
|
|
copiedOnion.value = true
|
|
|
|
|
if (copiedTimer) clearTimeout(copiedTimer)
|
|
|
|
|
copiedTimer = setTimeout(() => { copiedOnion.value = false }, 2000)
|
2026-02-17 15:03:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
function closeChangePasswordModal() {
|
2026-02-17 22:10:38 +00:00
|
|
|
changePasswordRestoreFocusRef.value?.focus?.()
|
2026-02-17 15:03:34 +00:00
|
|
|
showChangePasswordModal.value = false
|
|
|
|
|
changePasswordError.value = ''
|
|
|
|
|
changePasswordSuccess.value = ''
|
|
|
|
|
changePasswordForm.value = { currentPassword: '', newPassword: '', confirmPassword: '', alsoChangeSsh: true }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
onMounted(async () => {
|
|
|
|
|
if (!serverTorAddressFromStore.value) {
|
|
|
|
|
try {
|
|
|
|
|
const res = await rpcClient.getTorAddress()
|
|
|
|
|
torAddressFromRpc.value = res.tor_address ?? null
|
|
|
|
|
} catch {
|
|
|
|
|
// Ignore - tor address may not be available yet
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
})
|
2026-01-24 22:59:20 +00:00
|
|
|
|
|
|
|
|
async function handleLogout() {
|
2026-03-01 18:07:35 +00:00
|
|
|
try { await store.logout() } catch { /* proceed */ }
|
|
|
|
|
router.push('/login').catch(() => { window.location.href = '/login' })
|
2026-01-24 22:59:20 +00:00
|
|
|
}
|
|
|
|
|
</script>
|