Files
frangipane-client/src/pages/AccountPage.vue

157 lines
3.6 KiB
Vue

<template>
<div class="account-page">
<h1>{{ $t('account-title') }}</h1>
<UpdateAccountModal v-if="showUpdateModal" :user="user" @close="showUpdateModal = false" @updated="fetchUserData" />
<div v-if="user" class="info-card">
<button class="update-btn" @click="showUpdateModal = true">{{ $t('account-update-btn') }}</button>
<p><strong>{{ $t('account-label-username') }}</strong> {{ user.username }}</p>
<p><strong>{{ $t('account-label-email') }}</strong> {{ user.email }}</p>
</div>
<div v-else class="loading-state">
<p>{{ $t('account-loading') }}</p>
</div>
<div class="input-group">
<span>{{ $t('account-language') }}</span>
<div class="lang-grid">
<button v-for="lang in languages" :key="lang.code" class="lang-btn"
:class="{ active: currentLang === lang.code }" @click="changeLanguage(lang.code)">
{{ lang.name }}
</button>
</div>
</div>
<button class="logout-btn" @click="logout">
<i class="fa-solid fa-right-from-bracket"></i>
<span>{{ $t('account-logout-btn') }}</span>
</button>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, computed } from 'vue'
import { useRouter } from 'vue-router'
import { logout as authLogout } from '../store.ts'
import { getAuthData } from "../authStore.ts"
import type { User } from "../types"
import UpdateAccountModal from '../components/UpdateAccountModal.vue'
import { useFluent } from 'fluent-vue'
import { saveLocalePreference, getLocalePreference } from "../authStore.ts"
import { getSupportedLanguagesMetadata, setLanguage } from '../i18n'
const router = useRouter()
const user = ref<User | null>(null)
const showUpdateModal = ref(false)
const { $t } = useFluent()
const currentLang = ref('')
const languages = computed(() => getSupportedLanguagesMetadata())
async function fetchUserData() {
try {
const auth = await getAuthData()
user.value = auth.user
} catch (err) {
console.error("Failed to load user data:", err)
}
}
onMounted(async () => {
const pref = await getLocalePreference()
// Synchronize the UI state with the actual active language
currentLang.value = pref || (navigator.language.split('-')[0])
fetchUserData()
})
async function changeLanguage(code: string) {
const actual = setLanguage(code)
currentLang.value = actual
await saveLocalePreference(actual)
}
function logout() {
authLogout()
router.push('/login')
}
</script>
<style scoped>
.account-page {
max-width: 720px;
margin: 0 auto;
padding: 2rem 1.5rem;
display: flex;
flex-direction: column;
gap: 2rem;
}
.info-card {
position: relative;
background: var(--panel);
border: 1px solid var(--border);
border-radius: var(--radius);
padding: 1rem;
}
.update-btn {
position: absolute;
right: 10px;
top: 10px;
}
.lang-grid {
display: grid;
grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
gap: 0.5rem;
margin-top: 0.5rem;
}
.lang-btn {
background: var(--panel);
border: 1px solid var(--border);
color: var(--text);
margin: 0;
}
.lang-btn.active {
background: var(--accent);
color: #000;
border-color: var(--accent);
}
.logout-btn {
cursor: pointer;
display: flex;
align-items: center;
gap: 10px;
padding: 10px 20px;
color: var(--text);
background-color: transparent;
border: 1px solid #ff5050;
border-radius: 8px;
width: fit-content;
transition: all 0.2s ease;
}
.logout-btn:hover {
color: rgba(255, 80, 80, 0.8);
}
.logout-btn:hover i {
color: rgba(255, 80, 80, 0.8);
}
.logout-btn i {
font-size: 1.25rem;
}
.loading-state {
padding: 2rem;
text-align: center;
color: var(--muted);
}
</style>