mirror of
https://github.com/immich-app/immich.git
synced 2025-04-28 13:50:12 +02:00
- ability to add custom hover colors - migrate activity menu to ButtonContextMenu component - onClick callbacks rather than events for menu options - remove slots - configurable menu option colors - improve menu option layout
325 lines
12 KiB
Svelte
325 lines
12 KiB
Svelte
<script lang="ts">
|
|
import Icon from '$lib/components/elements/icon.svelte';
|
|
import { AppRoute, timeBeforeShowLoadingSpinner } from '$lib/constants';
|
|
import { getAssetThumbnailUrl, handlePromiseError } from '$lib/utils';
|
|
import { getAssetType } from '$lib/utils/asset-utils';
|
|
import { autoGrowHeight } from '$lib/actions/autogrow';
|
|
import { handleError } from '$lib/utils/handle-error';
|
|
import { isTenMinutesApart } from '$lib/utils/timesince';
|
|
import {
|
|
ReactionType,
|
|
createActivity,
|
|
deleteActivity,
|
|
getActivities,
|
|
type ActivityResponseDto,
|
|
type AssetTypeEnum,
|
|
type UserResponseDto,
|
|
} from '@immich/sdk';
|
|
import { mdiClose, mdiDotsVertical, mdiHeart, mdiSend, mdiDeleteOutline } from '@mdi/js';
|
|
import * as luxon from 'luxon';
|
|
import { createEventDispatcher, onMount } from 'svelte';
|
|
import CircleIconButton from '../elements/buttons/circle-icon-button.svelte';
|
|
import LoadingSpinner from '../shared-components/loading-spinner.svelte';
|
|
import { NotificationType, notificationController } from '../shared-components/notification/notification';
|
|
import UserAvatar from '../shared-components/user-avatar.svelte';
|
|
import { locale } from '$lib/stores/preferences.store';
|
|
import { shortcut } from '$lib/actions/shortcut';
|
|
import { t } from 'svelte-i18n';
|
|
import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte';
|
|
import MenuOption from '$lib/components/shared-components/context-menu/menu-option.svelte';
|
|
|
|
const units: Intl.RelativeTimeFormatUnit[] = ['year', 'month', 'week', 'day', 'hour', 'minute', 'second'];
|
|
|
|
const shouldGroup = (currentDate: string, nextDate: string): boolean => {
|
|
const currentDateTime = luxon.DateTime.fromISO(currentDate, { locale: $locale });
|
|
const nextDateTime = luxon.DateTime.fromISO(nextDate, { locale: $locale });
|
|
|
|
return currentDateTime.hasSame(nextDateTime, 'hour') || currentDateTime.toRelative() === nextDateTime.toRelative();
|
|
};
|
|
|
|
const timeSince = (dateTime: luxon.DateTime) => {
|
|
const diff = dateTime.diffNow().shiftTo(...units);
|
|
const unit = units.find((unit) => diff.get(unit) !== 0) || 'second';
|
|
|
|
const relativeFormatter = new Intl.RelativeTimeFormat('en', {
|
|
numeric: 'auto',
|
|
});
|
|
return relativeFormatter.format(Math.trunc(diff.as(unit)), unit);
|
|
};
|
|
|
|
export let reactions: ActivityResponseDto[];
|
|
export let user: UserResponseDto;
|
|
export let assetId: string | undefined = undefined;
|
|
export let albumId: string;
|
|
export let assetType: AssetTypeEnum | undefined = undefined;
|
|
export let albumOwnerId: string;
|
|
export let disabled: boolean;
|
|
export let isLiked: ActivityResponseDto | null;
|
|
|
|
let textArea: HTMLTextAreaElement;
|
|
let innerHeight: number;
|
|
let activityHeight: number;
|
|
let chatHeight: number;
|
|
let divHeight: number;
|
|
let previousAssetId: string | undefined = assetId;
|
|
let message = '';
|
|
let isSendingMessage = false;
|
|
|
|
const dispatch = createEventDispatcher<{
|
|
deleteComment: void;
|
|
deleteLike: void;
|
|
addComment: void;
|
|
close: void;
|
|
}>();
|
|
|
|
$: {
|
|
if (innerHeight && activityHeight) {
|
|
divHeight = innerHeight - activityHeight;
|
|
}
|
|
}
|
|
|
|
$: {
|
|
if (assetId && previousAssetId != assetId) {
|
|
handlePromiseError(getReactions());
|
|
previousAssetId = assetId;
|
|
}
|
|
}
|
|
onMount(async () => {
|
|
await getReactions();
|
|
});
|
|
|
|
const getReactions = async () => {
|
|
try {
|
|
reactions = await getActivities({ assetId, albumId });
|
|
} catch (error) {
|
|
handleError(error, $t('errors.unable_to_load_asset_activity'));
|
|
}
|
|
};
|
|
|
|
const timeOptions = {
|
|
year: 'numeric',
|
|
month: '2-digit',
|
|
day: '2-digit',
|
|
hour: '2-digit',
|
|
minute: '2-digit',
|
|
hour12: false,
|
|
} as Intl.DateTimeFormatOptions;
|
|
|
|
const handleDeleteReaction = async (reaction: ActivityResponseDto, index: number) => {
|
|
try {
|
|
await deleteActivity({ id: reaction.id });
|
|
reactions.splice(index, 1);
|
|
reactions = reactions;
|
|
if (isLiked && reaction.type === 'like' && reaction.id == isLiked.id) {
|
|
dispatch('deleteLike');
|
|
} else {
|
|
dispatch('deleteComment');
|
|
}
|
|
notificationController.show({
|
|
message: `${reaction.type} deleted`,
|
|
type: NotificationType.Info,
|
|
});
|
|
} catch (error) {
|
|
handleError(error, $t('errors.unable_to_remove_reaction'));
|
|
}
|
|
};
|
|
|
|
const handleSendComment = async () => {
|
|
if (!message) {
|
|
return;
|
|
}
|
|
const timeout = setTimeout(() => (isSendingMessage = true), timeBeforeShowLoadingSpinner);
|
|
try {
|
|
const data = await createActivity({
|
|
activityCreateDto: { albumId, assetId, type: ReactionType.Comment, comment: message },
|
|
});
|
|
reactions.push(data);
|
|
textArea.style.height = '18px';
|
|
message = '';
|
|
dispatch('addComment');
|
|
// Re-render the activity feed
|
|
reactions = reactions;
|
|
} catch (error) {
|
|
handleError(error, $t('errors.unable_to_add_comment'));
|
|
} finally {
|
|
clearTimeout(timeout);
|
|
}
|
|
isSendingMessage = false;
|
|
};
|
|
</script>
|
|
|
|
<div class="overflow-y-hidden relative h-full" bind:offsetHeight={innerHeight}>
|
|
<div class="dark:bg-immich-dark-bg dark:text-immich-dark-fg w-full h-full">
|
|
<div
|
|
class="flex w-full h-fit dark:bg-immich-dark-bg dark:text-immich-dark-fg p-2 bg-white"
|
|
bind:clientHeight={activityHeight}
|
|
>
|
|
<div class="flex place-items-center gap-2">
|
|
<CircleIconButton on:click={() => dispatch('close')} icon={mdiClose} title={$t('close')} />
|
|
|
|
<p class="text-lg text-immich-fg dark:text-immich-dark-fg">{$t('activity')}</p>
|
|
</div>
|
|
</div>
|
|
{#if innerHeight}
|
|
<div
|
|
class="overflow-y-auto immich-scrollbar relative w-full px-2"
|
|
style="height: {divHeight}px;padding-bottom: {chatHeight}px"
|
|
>
|
|
{#each reactions as reaction, index (reaction.id)}
|
|
{#if reaction.type === 'comment'}
|
|
<div class="flex dark:bg-gray-800 bg-gray-200 py-3 pl-3 mt-3 rounded-lg gap-4 justify-start">
|
|
<div class="flex items-center">
|
|
<UserAvatar user={reaction.user} size="sm" />
|
|
</div>
|
|
|
|
<div class="w-full leading-4 overflow-hidden self-center break-words text-sm">{reaction.comment}</div>
|
|
{#if assetId === undefined && reaction.assetId}
|
|
<a class="aspect-square w-[75px] h-[75px]" href="{AppRoute.ALBUMS}/{albumId}/photos/{reaction.assetId}">
|
|
<img
|
|
class="rounded-lg w-[75px] h-[75px] object-cover"
|
|
src={getAssetThumbnailUrl(reaction.assetId)}
|
|
alt="Profile picture of {reaction.user.name}, who commented on this asset"
|
|
/>
|
|
</a>
|
|
{/if}
|
|
{#if reaction.user.id === user.id || albumOwnerId === user.id}
|
|
<div class="mr-4">
|
|
<ButtonContextMenu
|
|
icon={mdiDotsVertical}
|
|
title={$t('comment_options')}
|
|
align="top-right"
|
|
direction="left"
|
|
size="16"
|
|
>
|
|
<MenuOption
|
|
activeColor="bg-red-200"
|
|
icon={mdiDeleteOutline}
|
|
text={$t('remove')}
|
|
onClick={() => handleDeleteReaction(reaction, index)}
|
|
/>
|
|
</ButtonContextMenu>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
|
|
{#if (index != reactions.length - 1 && !shouldGroup(reactions[index].createdAt, reactions[index + 1].createdAt)) || index === reactions.length - 1}
|
|
<div
|
|
class="pt-1 px-2 text-right w-full text-sm text-gray-500 dark:text-gray-300"
|
|
title={new Date(reaction.createdAt).toLocaleDateString(undefined, timeOptions)}
|
|
>
|
|
{timeSince(luxon.DateTime.fromISO(reaction.createdAt, { locale: $locale }))}
|
|
</div>
|
|
{/if}
|
|
{:else if reaction.type === 'like'}
|
|
<div class="relative">
|
|
<div class="flex py-3 pl-3 mt-3 gap-4 items-center text-sm">
|
|
<div class="text-red-600"><Icon path={mdiHeart} size={20} /></div>
|
|
|
|
<div class="w-full" title={`${reaction.user.name} (${reaction.user.email})`}>
|
|
{`${reaction.user.name} liked ${assetType ? `this ${getAssetType(assetType).toLowerCase()}` : 'it'}`}
|
|
</div>
|
|
{#if assetId === undefined && reaction.assetId}
|
|
<a
|
|
class="aspect-square w-[75px] h-[75px]"
|
|
href="{AppRoute.ALBUMS}/{albumId}/photos/{reaction.assetId}"
|
|
>
|
|
<img
|
|
class="rounded-lg w-[75px] h-[75px] object-cover"
|
|
src={getAssetThumbnailUrl(reaction.assetId)}
|
|
alt="Profile picture of {reaction.user.name}, who liked this asset"
|
|
/>
|
|
</a>
|
|
{/if}
|
|
{#if reaction.user.id === user.id || albumOwnerId === user.id}
|
|
<div class="mr-4">
|
|
<ButtonContextMenu
|
|
icon={mdiDotsVertical}
|
|
title={$t('reaction_options')}
|
|
align="top-right"
|
|
direction="left"
|
|
size="16"
|
|
>
|
|
<MenuOption
|
|
activeColor="bg-red-200"
|
|
icon={mdiDeleteOutline}
|
|
text={$t('remove')}
|
|
onClick={() => handleDeleteReaction(reaction, index)}
|
|
/>
|
|
</ButtonContextMenu>
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
{#if (index != reactions.length - 1 && isTenMinutesApart(reactions[index].createdAt, reactions[index + 1].createdAt)) || index === reactions.length - 1}
|
|
<div
|
|
class="pt-1 px-2 text-right w-full text-sm text-gray-500 dark:text-gray-300"
|
|
title={new Date(reaction.createdAt).toLocaleDateString(navigator.language, timeOptions)}
|
|
>
|
|
{timeSince(luxon.DateTime.fromISO(reaction.createdAt, { locale: $locale }))}
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
{/if}
|
|
{/each}
|
|
</div>
|
|
{/if}
|
|
</div>
|
|
|
|
<div class="absolute w-full bottom-0">
|
|
<div class="flex items-center justify-center p-2" bind:clientHeight={chatHeight}>
|
|
<div class="flex p-2 gap-4 h-fit bg-gray-200 text-immich-dark-gray rounded-3xl w-full">
|
|
<div>
|
|
<UserAvatar {user} size="md" showTitle={false} />
|
|
</div>
|
|
<form class="flex w-full max-h-56 gap-1" on:submit|preventDefault={() => handleSendComment()}>
|
|
<div class="flex w-full items-center gap-4">
|
|
<textarea
|
|
{disabled}
|
|
bind:this={textArea}
|
|
bind:value={message}
|
|
use:autoGrowHeight={'5px'}
|
|
placeholder={disabled ? $t('comments_are_disabled') : $t('say_something')}
|
|
on:input={() => autoGrowHeight(textArea, '5px')}
|
|
use:shortcut={{
|
|
shortcut: { key: 'Enter' },
|
|
onShortcut: () => handleSendComment(),
|
|
}}
|
|
class="h-[18px] {disabled
|
|
? 'cursor-not-allowed'
|
|
: ''} w-full max-h-56 pr-2 items-center overflow-y-auto leading-4 outline-none resize-none bg-gray-200"
|
|
/>
|
|
</div>
|
|
{#if isSendingMessage}
|
|
<div class="flex items-end place-items-center pb-2 ml-0">
|
|
<div class="flex w-full place-items-center">
|
|
<LoadingSpinner />
|
|
</div>
|
|
</div>
|
|
{:else if message}
|
|
<div class="flex items-end w-fit ml-0">
|
|
<CircleIconButton
|
|
title={$t('send_message')}
|
|
size="15"
|
|
icon={mdiSend}
|
|
class="dark:text-immich-dark-gray"
|
|
on:click={() => handleSendComment()}
|
|
/>
|
|
</div>
|
|
{/if}
|
|
</form>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<style>
|
|
::placeholder {
|
|
color: rgb(60, 60, 60);
|
|
opacity: 0.6;
|
|
}
|
|
|
|
::-ms-input-placeholder {
|
|
/* Edge 12 -18 */
|
|
color: white;
|
|
}
|
|
</style>
|