mirror of
https://github.com/immich-app/immich
synced 2025-11-07 17:27:20 +00:00
158 lines
5.8 KiB
Svelte
158 lines
5.8 KiB
Svelte
<script lang="ts">
|
|
import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte';
|
|
import FullScreenModal from '$lib/components/shared-components/full-screen-modal.svelte';
|
|
import {
|
|
AlbumUserRole,
|
|
getMyUser,
|
|
removeUserFromAlbum,
|
|
updateAlbumUser,
|
|
type AlbumResponseDto,
|
|
type UserResponseDto,
|
|
} from '@immich/sdk';
|
|
import { mdiDotsVertical } from '@mdi/js';
|
|
import { onMount } from 'svelte';
|
|
import { t } from 'svelte-i18n';
|
|
import { handleError } from '../../utils/handle-error';
|
|
import MenuOption from '../shared-components/context-menu/menu-option.svelte';
|
|
import ConfirmDialog from '../shared-components/dialog/confirm-dialog.svelte';
|
|
import { NotificationType, notificationController } from '../shared-components/notification/notification';
|
|
import UserAvatar from '../shared-components/user-avatar.svelte';
|
|
|
|
interface Props {
|
|
album: AlbumResponseDto;
|
|
onClose: () => void;
|
|
onRemove: (userId: string) => void;
|
|
onRefreshAlbum: () => void;
|
|
}
|
|
|
|
let { album, onClose, onRemove, onRefreshAlbum }: Props = $props();
|
|
|
|
let currentUser: UserResponseDto | undefined = $state();
|
|
let selectedRemoveUser: UserResponseDto | null = $state(null);
|
|
|
|
let isOwned = $derived(currentUser?.id == album.ownerId);
|
|
|
|
onMount(async () => {
|
|
try {
|
|
currentUser = await getMyUser();
|
|
} catch (error) {
|
|
handleError(error, $t('errors.unable_to_refresh_user'));
|
|
}
|
|
});
|
|
|
|
const handleMenuRemove = (user: UserResponseDto) => {
|
|
selectedRemoveUser = user;
|
|
};
|
|
|
|
const handleRemoveUser = async () => {
|
|
if (!selectedRemoveUser) {
|
|
return;
|
|
}
|
|
|
|
const userId = selectedRemoveUser.id === currentUser?.id ? 'me' : selectedRemoveUser.id;
|
|
|
|
try {
|
|
await removeUserFromAlbum({ id: album.id, userId });
|
|
onRemove(userId);
|
|
const message =
|
|
userId === 'me'
|
|
? $t('album_user_left', { values: { album: album.albumName } })
|
|
: $t('album_user_removed', { values: { user: selectedRemoveUser.name } });
|
|
notificationController.show({ type: NotificationType.Info, message });
|
|
} catch (error) {
|
|
handleError(error, $t('errors.unable_to_remove_album_users'));
|
|
} finally {
|
|
selectedRemoveUser = null;
|
|
}
|
|
};
|
|
|
|
const handleSetReadonly = async (user: UserResponseDto, role: AlbumUserRole) => {
|
|
try {
|
|
await updateAlbumUser({ id: album.id, userId: user.id, updateAlbumUserDto: { role } });
|
|
const message = $t('user_role_set', {
|
|
values: { user: user.name, role: role == AlbumUserRole.Viewer ? $t('role_viewer') : $t('role_editor') },
|
|
});
|
|
onRefreshAlbum();
|
|
notificationController.show({ type: NotificationType.Info, message });
|
|
} catch (error) {
|
|
handleError(error, $t('errors.unable_to_change_album_user_role'));
|
|
} finally {
|
|
selectedRemoveUser = null;
|
|
}
|
|
};
|
|
</script>
|
|
|
|
{#if !selectedRemoveUser}
|
|
<FullScreenModal title={$t('options')} {onClose}>
|
|
<section class="immich-scrollbar max-h-[400px] overflow-y-auto pb-4">
|
|
<div class="flex w-full place-items-center justify-between gap-4 p-5">
|
|
<div class="flex place-items-center gap-4">
|
|
<UserAvatar user={album.owner} size="md" />
|
|
<p class="text-sm font-medium">{album.owner.name}</p>
|
|
</div>
|
|
|
|
<div id="icon-{album.owner.id}" class="flex place-items-center">
|
|
<p class="text-sm">{$t('owner')}</p>
|
|
</div>
|
|
</div>
|
|
{#each album.albumUsers as { user, role } (user.id)}
|
|
<div
|
|
class="flex w-full place-items-center justify-between gap-4 p-5 rounded-xl transition-colors hover:bg-gray-50 dark:hover:bg-gray-700"
|
|
>
|
|
<div class="flex place-items-center gap-4">
|
|
<UserAvatar {user} size="md" />
|
|
<p class="text-sm font-medium">{user.name}</p>
|
|
</div>
|
|
|
|
<div id="icon-{user.id}" class="flex place-items-center gap-2 text-sm">
|
|
<div>
|
|
{#if role === AlbumUserRole.Viewer}
|
|
{$t('role_viewer')}
|
|
{:else}
|
|
{$t('role_editor')}
|
|
{/if}
|
|
</div>
|
|
{#if isOwned}
|
|
<ButtonContextMenu icon={mdiDotsVertical} size="20" title={$t('options')}>
|
|
{#if role === AlbumUserRole.Viewer}
|
|
<MenuOption onClick={() => handleSetReadonly(user, AlbumUserRole.Editor)} text={$t('allow_edits')} />
|
|
{:else}
|
|
<MenuOption
|
|
onClick={() => handleSetReadonly(user, AlbumUserRole.Viewer)}
|
|
text={$t('disallow_edits')}
|
|
/>
|
|
{/if}
|
|
<MenuOption onClick={() => handleMenuRemove(user)} text={$t('remove')} />
|
|
</ButtonContextMenu>
|
|
{:else if user.id == currentUser?.id}
|
|
<button
|
|
type="button"
|
|
onclick={() => (selectedRemoveUser = user)}
|
|
class="text-sm font-medium text-immich-primary transition-colors hover:text-immich-primary/75 dark:text-immich-dark-primary"
|
|
>{$t('leave')}</button
|
|
>
|
|
{/if}
|
|
</div>
|
|
</div>
|
|
{/each}
|
|
</section>
|
|
</FullScreenModal>
|
|
{/if}
|
|
|
|
{#if selectedRemoveUser && selectedRemoveUser?.id === currentUser?.id}
|
|
<ConfirmDialog
|
|
title={$t('album_leave')}
|
|
prompt={$t('album_leave_confirmation', { values: { album: album.albumName } })}
|
|
confirmText={$t('leave')}
|
|
onClose={(confirmed) => (confirmed ? handleRemoveUser() : (selectedRemoveUser = null))}
|
|
/>
|
|
{/if}
|
|
|
|
{#if selectedRemoveUser && selectedRemoveUser?.id !== currentUser?.id}
|
|
<ConfirmDialog
|
|
title={$t('album_remove_user')}
|
|
prompt={$t('album_remove_user_confirmation', { values: { user: selectedRemoveUser.name } })}
|
|
confirmText={$t('remove_user')}
|
|
onClose={(confirmed) => (confirmed ? handleRemoveUser() : (selectedRemoveUser = null))}
|
|
/>
|
|
{/if}
|