mirror of
https://github.com/immich-app/immich.git
synced 2025-06-22 15:00:52 -04:00
120 lines
3.2 KiB
Svelte
120 lines
3.2 KiB
Svelte
<script lang="ts" module>
|
|
export type Size = 'full' | 'sm' | 'md' | 'lg' | 'xl' | 'xxl' | 'xxxl';
|
|
</script>
|
|
|
|
<script lang="ts">
|
|
import { getProfileImageUrl } from '$lib/utils';
|
|
import { type UserAvatarColor } from '@immich/sdk';
|
|
import { t } from 'svelte-i18n';
|
|
|
|
interface User {
|
|
id: string;
|
|
name: string;
|
|
email: string;
|
|
profileImagePath: string;
|
|
avatarColor: UserAvatarColor;
|
|
profileChangedAt: string;
|
|
}
|
|
|
|
interface Props {
|
|
user: User;
|
|
color?: UserAvatarColor | undefined;
|
|
size?: Size;
|
|
rounded?: boolean;
|
|
interactive?: boolean;
|
|
showTitle?: boolean;
|
|
showProfileImage?: boolean;
|
|
label?: string | undefined;
|
|
}
|
|
|
|
let {
|
|
user,
|
|
color = undefined,
|
|
size = 'full',
|
|
rounded = true,
|
|
interactive = false,
|
|
showTitle = true,
|
|
showProfileImage = true,
|
|
label = undefined,
|
|
}: Props = $props();
|
|
|
|
let img: HTMLImageElement | undefined = $state();
|
|
let showFallback = $state(true);
|
|
|
|
const tryLoadImage = async () => {
|
|
try {
|
|
await img?.decode();
|
|
showFallback = false;
|
|
} catch {
|
|
showFallback = true;
|
|
}
|
|
};
|
|
|
|
const colorClasses: Record<UserAvatarColor, string> = {
|
|
primary: 'bg-immich-primary dark:bg-immich-dark-primary text-immich-dark-fg dark:text-immich-fg',
|
|
pink: 'bg-pink-400 text-immich-bg',
|
|
red: 'bg-red-500 text-immich-bg',
|
|
yellow: 'bg-yellow-500 text-immich-bg',
|
|
blue: 'bg-blue-500 text-immich-bg',
|
|
green: 'bg-green-600 text-immich-bg',
|
|
purple: 'bg-purple-600 text-immich-bg',
|
|
orange: 'bg-orange-600 text-immich-bg',
|
|
gray: 'bg-gray-600 text-immich-bg',
|
|
amber: 'bg-amber-600 text-immich-bg',
|
|
};
|
|
|
|
const sizeClasses: Record<Size, string> = {
|
|
full: 'w-full h-full',
|
|
sm: 'w-7 h-7',
|
|
md: 'w-10 h-10',
|
|
lg: 'w-12 h-12',
|
|
xl: 'w-16 h-16',
|
|
xxl: 'w-24 h-24',
|
|
xxxl: 'w-28 h-28',
|
|
};
|
|
|
|
$effect(() => {
|
|
if (img && user) {
|
|
tryLoadImage().catch(console.error);
|
|
}
|
|
});
|
|
|
|
let colorClass = $derived(colorClasses[color || user.avatarColor]);
|
|
let sizeClass = $derived(sizeClasses[size]);
|
|
let title = $derived(label ?? `${user.name} (${user.email})`);
|
|
let interactiveClass = $derived(
|
|
interactive
|
|
? 'border-2 border-immich-primary hover:border-immich-dark-primary dark:hover:border-immich-primary dark:border-immich-dark-primary transition-colors'
|
|
: '',
|
|
);
|
|
</script>
|
|
|
|
<figure
|
|
class="{sizeClass} {colorClass} {interactiveClass} overflow-hidden shadow-md"
|
|
class:rounded-full={rounded}
|
|
title={showTitle ? title : undefined}
|
|
>
|
|
{#if showProfileImage && user.profileImagePath}
|
|
<img
|
|
bind:this={img}
|
|
src={getProfileImageUrl(user)}
|
|
alt={$t('profile_image_of_user', { values: { user: title } })}
|
|
class="h-full w-full object-cover"
|
|
class:hidden={showFallback}
|
|
draggable="false"
|
|
/>
|
|
{/if}
|
|
{#if showFallback}
|
|
<span
|
|
class="flex h-full w-full select-none items-center justify-center font-medium"
|
|
class:text-xs={size === 'sm'}
|
|
class:text-lg={size === 'lg'}
|
|
class:text-xl={size === 'xl'}
|
|
class:text-2xl={size === 'xxl'}
|
|
class:text-3xl={size === 'xxxl'}
|
|
>
|
|
{(user.name[0] || '').toUpperCase()}
|
|
</span>
|
|
{/if}
|
|
</figure>
|