73 lines
2.6 KiB
Vue
73 lines
2.6 KiB
Vue
<template>
|
|
<div class="card overflow-hidden">
|
|
<div
|
|
class="border-b border-gray-100 bg-gradient-to-r from-primary-500/10 to-primary-600/5 px-6 py-5 dark:border-dark-700 dark:from-primary-500/20 dark:to-primary-600/10"
|
|
>
|
|
<div class="flex items-center gap-4">
|
|
<div
|
|
class="flex h-16 w-16 items-center justify-center overflow-hidden rounded-2xl bg-gradient-to-br from-primary-500 to-primary-600 text-2xl font-bold text-white shadow-lg shadow-primary-500/20"
|
|
>
|
|
<img
|
|
v-if="avatarUrl"
|
|
:src="avatarUrl"
|
|
:alt="displayName"
|
|
class="h-full w-full object-cover"
|
|
>
|
|
<span v-else>{{ avatarInitial }}</span>
|
|
</div>
|
|
<div class="min-w-0 flex-1">
|
|
<h2 class="truncate text-lg font-semibold text-gray-900 dark:text-white">
|
|
{{ displayName }}
|
|
</h2>
|
|
<p class="mt-1 truncate text-sm text-gray-500 dark:text-gray-400">
|
|
{{ user?.email }}
|
|
</p>
|
|
<div class="mt-1 flex items-center gap-2">
|
|
<span :class="['badge', user?.role === 'admin' ? 'badge-primary' : 'badge-gray']">
|
|
{{ user?.role === 'admin' ? t('profile.administrator') : t('profile.user') }}
|
|
</span>
|
|
<span
|
|
:class="['badge', user?.status === 'active' ? 'badge-success' : 'badge-danger']"
|
|
>
|
|
{{ user?.status }}
|
|
</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
<div class="px-6 py-4">
|
|
<div class="space-y-3">
|
|
<div class="flex items-center gap-3 text-sm text-gray-600 dark:text-gray-400">
|
|
<Icon name="mail" size="sm" class="text-gray-400 dark:text-gray-500" />
|
|
<span class="truncate">{{ user?.email }}</span>
|
|
</div>
|
|
<div
|
|
v-if="user?.username"
|
|
class="flex items-center gap-3 text-sm text-gray-600 dark:text-gray-400"
|
|
>
|
|
<Icon name="user" size="sm" class="text-gray-400 dark:text-gray-500" />
|
|
<span class="truncate">{{ user.username }}</span>
|
|
</div>
|
|
</div>
|
|
|
|
</div>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { computed } from 'vue'
|
|
import { useI18n } from 'vue-i18n'
|
|
import Icon from '@/components/icons/Icon.vue'
|
|
import type { User } from '@/types'
|
|
|
|
const props = defineProps<{
|
|
user: User | null
|
|
}>()
|
|
|
|
const { t } = useI18n()
|
|
|
|
const avatarUrl = computed(() => props.user?.avatar_url?.trim() || '')
|
|
const displayName = computed(() => props.user?.username?.trim() || props.user?.email?.trim() || 'User')
|
|
const avatarInitial = computed(() => displayName.value.charAt(0).toUpperCase() || 'U')
|
|
</script>
|