Files
kestrelos/app/pages/account.vue
Keli Grubb 0aab29ea72
All checks were successful
ci/woodpecker/push/push Pipeline was successful
minor: new nav system (#5)
Co-authored-by: Madison Grubb <madison@elastiflow.com>
Reviewed-on: #5
2026-02-15 04:04:54 +00:00

258 lines
7.0 KiB
Vue

<template>
<div class="p-6">
<h2 class="kestrel-page-heading mb-4">
Account
</h2>
<section
v-if="user"
class="mb-8"
>
<h3 class="kestrel-section-label">
Profile photo
</h3>
<div class="kestrel-card flex items-center gap-4 p-4">
<div class="flex h-16 w-16 shrink-0 overflow-hidden rounded-full border border-kestrel-border bg-kestrel-border">
<img
v-if="user.avatar_url"
:src="`${user.avatar_url}${avatarBust ? `?t=${avatarBust}` : ''}`"
alt=""
class="h-full w-full object-cover"
>
<span
v-else
class="flex h-full w-full items-center justify-center text-lg font-medium text-kestrel-text"
>
{{ accountInitials }}
</span>
</div>
<div class="flex flex-wrap gap-2">
<label class="kestrel-btn-secondary cursor-pointer">
<input
type="file"
accept="image/jpeg,image/png"
class="sr-only"
:disabled="avatarLoading"
@change="onAvatarFileChange"
>
{{ avatarLoading ? 'Uploading…' : 'Upload' }}
</label>
<button
type="button"
class="kestrel-btn-secondary disabled:opacity-50"
:disabled="avatarLoading || !user.avatar_url"
@click="onRemoveAvatar"
>
Remove
</button>
</div>
</div>
</section>
<section class="mb-8">
<h3 class="kestrel-section-label">
Profile
</h3>
<div class="kestrel-card p-4">
<template v-if="user">
<dl class="space-y-2 text-sm">
<div>
<dt class="text-kestrel-muted">
Identifier
</dt>
<dd class="mt-0.5 font-medium text-kestrel-text">
{{ user.identifier }}
</dd>
</div>
<div>
<dt class="text-kestrel-muted">
Role
</dt>
<dd class="mt-0.5 font-medium text-kestrel-text">
{{ user.role }}
</dd>
</div>
<div>
<dt class="text-kestrel-muted">
Sign-in method
</dt>
<dd class="mt-0.5 font-medium text-kestrel-text">
{{ user.auth_provider === 'oidc' ? 'OIDC' : 'Local' }}
</dd>
</div>
</dl>
<p class="mt-3 text-xs text-kestrel-muted">
Admins can manage all users on the Members page.
</p>
</template>
<p
v-else
class="text-sm text-kestrel-muted"
>
Sign in to see your profile.
</p>
</div>
</section>
<section
v-if="user?.auth_provider === 'local'"
class="mb-8"
>
<h3 class="kestrel-section-label">
Change password
</h3>
<div class="kestrel-card p-4">
<p
v-if="passwordSuccess"
class="mb-3 text-sm text-green-400"
>
Password updated.
</p>
<p
v-if="passwordError"
class="mb-3 text-sm text-red-400"
>
{{ passwordError }}
</p>
<form
class="space-y-3"
@submit.prevent="onChangePassword"
>
<div>
<label
for="account-current-password"
class="kestrel-label"
>Current password</label>
<input
id="account-current-password"
v-model="currentPassword"
type="password"
autocomplete="current-password"
class="kestrel-input"
>
</div>
<div>
<label
for="account-new-password"
class="kestrel-label"
>New password</label>
<input
id="account-new-password"
v-model="newPassword"
type="password"
autocomplete="new-password"
class="kestrel-input"
>
</div>
<div>
<label
for="account-confirm-password"
class="kestrel-label"
>Confirm new password</label>
<input
id="account-confirm-password"
v-model="confirmPassword"
type="password"
autocomplete="new-password"
class="kestrel-input"
>
</div>
<button
type="submit"
class="rounded bg-kestrel-accent px-4 py-2 text-sm font-medium text-kestrel-bg transition-opacity hover:opacity-90 disabled:opacity-50"
:disabled="passwordLoading"
>
{{ passwordLoading ? 'Updating…' : 'Update password' }}
</button>
</form>
</div>
</section>
</div>
</template>
<script setup>
const { user, refresh } = useUser()
const avatarBust = ref(0)
const avatarLoading = ref(false)
const currentPassword = ref('')
const newPassword = ref('')
const confirmPassword = ref('')
const passwordLoading = ref(false)
const passwordSuccess = ref(false)
const passwordError = ref('')
const accountInitials = computed(() => {
const id = user.value?.identifier ?? ''
const parts = id.trim().split(/\s+/)
if (parts.length >= 2) return (parts[0][0] + parts[1][0]).toUpperCase()
return id.slice(0, 2).toUpperCase() || '?'
})
async function onAvatarFileChange(e) {
const file = e.target.files?.[0]
if (!file) return
avatarLoading.value = true
try {
const form = new FormData()
form.append('avatar', file, file.name)
await $fetch('/api/me/avatar', { method: 'PUT', body: form, credentials: 'include' })
avatarBust.value = Date.now()
await refresh()
}
catch {
// Error surfaced by refresh or network
}
finally {
avatarLoading.value = false
e.target.value = ''
}
}
async function onRemoveAvatar() {
avatarLoading.value = true
try {
await $fetch('/api/me/avatar', { method: 'DELETE', credentials: 'include' })
avatarBust.value = Date.now()
await refresh()
}
finally {
avatarLoading.value = false
}
}
async function onChangePassword() {
passwordError.value = ''
passwordSuccess.value = false
if (newPassword.value !== confirmPassword.value) {
passwordError.value = 'New password and confirmation do not match.'
return
}
if (newPassword.value.length < 1) {
passwordError.value = 'New password cannot be empty.'
return
}
passwordLoading.value = true
try {
await $fetch('/api/me/password', {
method: 'PUT',
body: {
currentPassword: currentPassword.value,
newPassword: newPassword.value,
},
credentials: 'include',
})
currentPassword.value = ''
newPassword.value = ''
confirmPassword.value = ''
passwordSuccess.value = true
}
catch (e) {
passwordError.value = e.data?.message ?? e.message ?? 'Failed to update password.'
}
finally {
passwordLoading.value = false
}
}
</script>