123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779 |
- <script lang="ts">
- import { goto } from '$app/navigation';
- import {
- ActivityResponseDto,
- AlbumResponseDto,
- api,
- AssetJobName,
- AssetResponseDto,
- AssetTypeEnum,
- ReactionType,
- SharedLinkResponseDto,
- UserResponseDto,
- } from '@api';
- import { createEventDispatcher, onDestroy, onMount } from 'svelte';
- import { fly } from 'svelte/transition';
- import AlbumSelectionModal from '../shared-components/album-selection-modal.svelte';
- import { notificationController, NotificationType } from '../shared-components/notification/notification';
- import AssetViewerNavBar from './asset-viewer-nav-bar.svelte';
- import DetailPanel from './detail-panel.svelte';
- import PhotoViewer from './photo-viewer.svelte';
- import VideoViewer from './video-viewer.svelte';
- import PanoramaViewer from './panorama-viewer.svelte';
- import { ProjectionType } from '$lib/constants';
- import ConfirmDialogue from '$lib/components/shared-components/confirm-dialogue.svelte';
- import ProfileImageCropper from '../shared-components/profile-image-cropper.svelte';
- import { isShowDetail } from '$lib/stores/preferences.store';
- import { addAssetsToAlbum, downloadFile, getAssetType } from '$lib/utils/asset-utils';
- import NavigationArea from './navigation-area.svelte';
- import { browser } from '$app/environment';
- import { handleError } from '$lib/utils/handle-error';
- import type { AssetStore } from '$lib/stores/assets.store';
- import CircleIconButton from '../elements/buttons/circle-icon-button.svelte';
- import ProgressBar, { ProgressBarStatus } from '../shared-components/progress-bar/progress-bar.svelte';
- import { shouldIgnoreShortcut } from '$lib/utils/shortcut';
- import { featureFlags } from '$lib/stores/server-config.store';
- import {
- mdiChevronLeft,
- mdiHeartOutline,
- mdiHeart,
- mdiCommentOutline,
- mdiChevronRight,
- mdiClose,
- mdiImageBrokenVariant,
- mdiPause,
- mdiPlay,
- } from '@mdi/js';
- import Icon from '$lib/components/elements/icon.svelte';
- import Thumbnail from '../assets/thumbnail/thumbnail.svelte';
- import { stackAssetsStore } from '$lib/stores/stacked-asset.store';
- import ActivityViewer from './activity-viewer.svelte';
- export let assetStore: AssetStore | null = null;
- export let asset: AssetResponseDto;
- export let showNavigation = true;
- export let sharedLink: SharedLinkResponseDto | undefined = undefined;
- $: isTrashEnabled = $featureFlags.trash;
- export let force = false;
- export let withStacked = false;
- export let isShared = true;
- export let user: UserResponseDto | null = null;
- export let album: AlbumResponseDto | null = null;
- let reactions: ActivityResponseDto[] = [];
- const dispatch = createEventDispatcher<{
- archived: AssetResponseDto;
- unarchived: AssetResponseDto;
- favorite: AssetResponseDto;
- unfavorite: AssetResponseDto;
- close: void;
- next: void;
- previous: void;
- unstack: void;
- }>();
- let appearsInAlbums: AlbumResponseDto[] = [];
- let isShowAlbumPicker = false;
- let isShowDeleteConfirmation = false;
- let addToSharedAlbum = true;
- let shouldPlayMotionPhoto = false;
- let isShowProfileImageCrop = false;
- let shouldShowDownloadButton = sharedLink ? sharedLink.allowDownload : !asset.isOffline;
- let shouldShowDetailButton = asset.hasMetadata;
- let canCopyImagesToClipboard: boolean;
- let previewStackedAsset: AssetResponseDto | undefined;
- let isShowActivity = false;
- let isLiked: ActivityResponseDto | null = null;
- let numberOfComments: number;
- $: {
- if (asset.stackCount && asset.stack) {
- $stackAssetsStore = asset.stack;
- $stackAssetsStore = [...$stackAssetsStore, asset].sort(
- (a, b) => new Date(b.fileCreatedAt).getTime() - new Date(a.fileCreatedAt).getTime(),
- );
- }
- if (!$stackAssetsStore.map((a) => a.id).includes(asset.id)) {
- $stackAssetsStore = [];
- }
- }
- const handleFavorite = async () => {
- if (album) {
- try {
- if (isLiked) {
- const activityId = isLiked.id;
- await api.activityApi.deleteActivity({ id: activityId });
- reactions = reactions.filter((reaction) => reaction.id !== activityId);
- isLiked = null;
- } else {
- const { data } = await api.activityApi.createActivity({
- activityCreateDto: { albumId: album.id, assetId: asset.id, type: ReactionType.Like },
- });
- isLiked = data;
- reactions = [...reactions, isLiked];
- }
- } catch (error) {
- handleError(error, "Can't change favorite for asset");
- }
- }
- };
- const getFavorite = async () => {
- if (album && user) {
- try {
- const { data } = await api.activityApi.getActivities({
- userId: user.id,
- assetId: asset.id,
- albumId: album.id,
- type: ReactionType.Like,
- });
- if (data.length > 0) {
- isLiked = data[0];
- }
- } catch (error) {
- handleError(error, "Can't get Favorite");
- }
- }
- };
- const getNumberOfComments = async () => {
- if (album) {
- try {
- const { data } = await api.activityApi.getActivityStatistics({ assetId: asset.id, albumId: album.id });
- numberOfComments = data.comments;
- } catch (error) {
- handleError(error, "Can't get number of comments");
- }
- }
- };
- $: {
- if (isShared && asset.id) {
- getFavorite();
- getNumberOfComments();
- }
- }
- const onKeyboardPress = (keyInfo: KeyboardEvent) => handleKeyboardPress(keyInfo);
- onMount(async () => {
- document.addEventListener('keydown', onKeyboardPress);
- if (!sharedLink) {
- await getAllAlbums();
- }
- // Import hack :( see https://github.com/vadimkorr/svelte-carousel/issues/27#issuecomment-851022295
- // TODO: Move to regular import once the package correctly supports ESM.
- const module = await import('copy-image-clipboard');
- canCopyImagesToClipboard = module.canCopyImagesToClipboard();
- if (asset.stackCount && asset.stack) {
- $stackAssetsStore = asset.stack;
- $stackAssetsStore = [...$stackAssetsStore, asset].sort(
- (a, b) => new Date(a.fileCreatedAt).getTime() - new Date(b.fileCreatedAt).getTime(),
- );
- } else {
- $stackAssetsStore = [];
- }
- });
- onDestroy(() => {
- if (browser) {
- document.removeEventListener('keydown', onKeyboardPress);
- }
- });
- $: asset.id && !sharedLink && getAllAlbums(); // Update the album information when the asset ID changes
- const getAllAlbums = async () => {
- if (api.isSharedLink) {
- return;
- }
- try {
- const { data } = await api.albumApi.getAllAlbums({ assetId: asset.id });
- appearsInAlbums = data;
- } catch (e) {
- console.error('Error getting album that asset belong to', e);
- }
- };
- const handleOpenActivity = () => {
- if ($isShowDetail) {
- $isShowDetail = false;
- }
- isShowActivity = !isShowActivity;
- };
- const handleKeyboardPress = (event: KeyboardEvent) => {
- if (shouldIgnoreShortcut(event)) {
- return;
- }
- const key = event.key;
- const shiftKey = event.shiftKey;
- switch (key) {
- case 'a':
- case 'A':
- if (shiftKey) {
- toggleArchive();
- }
- return;
- case 'ArrowLeft':
- navigateAssetBackward();
- return;
- case 'ArrowRight':
- navigateAssetForward();
- return;
- case 'd':
- case 'D':
- if (shiftKey) {
- downloadFile(asset);
- }
- return;
- case 'Delete':
- trashOrDelete();
- return;
- case 'Escape':
- if (isShowDeleteConfirmation) {
- isShowDeleteConfirmation = false;
- return;
- }
- closeViewer();
- return;
- case 'f':
- toggleFavorite();
- return;
- case 'i':
- isShowActivity = false;
- $isShowDetail = !$isShowDetail;
- return;
- }
- };
- const handleCloseViewer = () => {
- $isShowDetail = false;
- closeViewer();
- };
- const closeViewer = () => dispatch('close');
- const navigateAssetForward = async (e?: Event) => {
- if (isSlideshowMode && assetStore && progressBar) {
- const hasNext = await assetStore.getNextAssetId(asset.id);
- if (hasNext) {
- progressBar.restart(true);
- } else {
- await handleStopSlideshow();
- }
- }
- e?.stopPropagation();
- dispatch('next');
- };
- const navigateAssetBackward = (e?: Event) => {
- if (isSlideshowMode && progressBar) {
- progressBar.restart(true);
- }
- e?.stopPropagation();
- dispatch('previous');
- };
- const showDetailInfoHandler = () => {
- if (isShowActivity) {
- isShowActivity = false;
- }
- $isShowDetail = !$isShowDetail;
- };
- $: trashOrDelete = !(force || !isTrashEnabled)
- ? trashAsset
- : () => {
- isShowDeleteConfirmation = true;
- };
- const trashAsset = async () => {
- try {
- await api.assetApi.deleteAssets({ assetBulkDeleteDto: { ids: [asset.id] } });
- await navigateAssetForward();
- assetStore?.removeAsset(asset.id);
- notificationController.show({
- message: 'Moved to trash',
- type: NotificationType.Info,
- });
- } catch (e) {
- handleError(e, 'Unable to trash asset');
- }
- };
- const deleteAsset = async () => {
- try {
- await api.assetApi.deleteAssets({ assetBulkDeleteDto: { ids: [asset.id], force: true } });
- await navigateAssetForward();
- assetStore?.removeAsset(asset.id);
- notificationController.show({
- message: 'Permanently deleted asset',
- type: NotificationType.Info,
- });
- } catch (e) {
- handleError(e, 'Unable to delete asset');
- } finally {
- isShowDeleteConfirmation = false;
- }
- };
- const toggleFavorite = async () => {
- try {
- const { data } = await api.assetApi.updateAsset({
- id: asset.id,
- updateAssetDto: {
- isFavorite: !asset.isFavorite,
- },
- });
- asset.isFavorite = data.isFavorite;
- assetStore?.updateAsset(data);
- dispatch(data.isFavorite ? 'favorite' : 'unfavorite', data);
- notificationController.show({
- type: NotificationType.Info,
- message: asset.isFavorite ? `Added to favorites` : `Removed from favorites`,
- });
- } catch (error) {
- await handleError(error, `Unable to ${asset.isFavorite ? `add asset to` : `remove asset from`} favorites`);
- }
- };
- const openAlbumPicker = (shared: boolean) => {
- isShowAlbumPicker = true;
- addToSharedAlbum = shared;
- };
- const handleAddToNewAlbum = (event: CustomEvent) => {
- isShowAlbumPicker = false;
- const { albumName }: { albumName: string } = event.detail;
- api.albumApi.createAlbum({ createAlbumDto: { albumName, assetIds: [asset.id] } }).then((response) => {
- const album = response.data;
- goto('/albums/' + album.id);
- });
- };
- const handleAddToAlbum = async (event: CustomEvent<{ album: AlbumResponseDto }>) => {
- isShowAlbumPicker = false;
- const album = event.detail.album;
- await addAssetsToAlbum(album.id, [asset.id]);
- await getAllAlbums();
- };
- const disableKeyDownEvent = () => {
- if (browser) {
- document.removeEventListener('keydown', onKeyboardPress);
- }
- };
- const enableKeyDownEvent = () => {
- if (browser) {
- document.addEventListener('keydown', onKeyboardPress);
- }
- };
- const toggleArchive = async () => {
- try {
- const { data } = await api.assetApi.updateAsset({
- id: asset.id,
- updateAssetDto: {
- isArchived: !asset.isArchived,
- },
- });
- asset.isArchived = data.isArchived;
- assetStore?.updateAsset(data);
- dispatch(data.isArchived ? 'archived' : 'unarchived', data);
- notificationController.show({
- type: NotificationType.Info,
- message: asset.isArchived ? `Added to archive` : `Removed from archive`,
- });
- } catch (error) {
- await handleError(error, `Unable to ${asset.isArchived ? `add asset to` : `remove asset from`} archive`);
- }
- };
- const handleRunJob = async (name: AssetJobName) => {
- try {
- await api.assetApi.runAssetJobs({ assetJobsDto: { assetIds: [asset.id], name } });
- notificationController.show({ type: NotificationType.Info, message: api.getAssetJobMessage(name) });
- } catch (error) {
- await handleError(error, `Unable to submit job`);
- }
- };
- /**
- * Slide show mode
- */
- let isSlideshowMode = false;
- let assetViewerHtmlElement: HTMLElement;
- let progressBar: ProgressBar;
- let progressBarStatus: ProgressBarStatus;
- const handleVideoStarted = () => {
- if (isSlideshowMode) {
- progressBar.restart(false);
- }
- };
- const handleVideoEnded = async () => {
- if (isSlideshowMode) {
- await navigateAssetForward();
- }
- };
- const handlePlaySlideshow = async () => {
- try {
- await assetViewerHtmlElement.requestFullscreen();
- } catch (error) {
- console.error('Error entering fullscreen', error);
- } finally {
- isSlideshowMode = true;
- }
- };
- const handleStopSlideshow = async () => {
- try {
- await document.exitFullscreen();
- } catch (error) {
- console.error('Error exiting fullscreen', error);
- } finally {
- isSlideshowMode = false;
- progressBar.restart(false);
- }
- };
- const handleStackedAssetMouseEvent = (e: CustomEvent<{ isMouseOver: boolean }>, asset: AssetResponseDto) => {
- const { isMouseOver } = e.detail;
- if (isMouseOver) {
- previewStackedAsset = asset;
- } else {
- previewStackedAsset = undefined;
- }
- };
- const handleUnstack = async () => {
- try {
- const ids = $stackAssetsStore.map(({ id }) => id);
- await api.assetApi.updateAssets({ assetBulkUpdateDto: { ids, removeParent: true } });
- for (const child of $stackAssetsStore) {
- child.stackParentId = null;
- assetStore?.addAsset(child);
- }
- asset.stackCount = 0;
- asset.stack = [];
- assetStore?.updateAsset(asset);
- dispatch('unstack');
- notificationController.show({ type: NotificationType.Info, message: 'Un-stacked', timeout: 1500 });
- } catch (error) {
- await handleError(error, `Unable to unstack`);
- }
- };
- </script>
- <section
- id="immich-asset-viewer"
- class="fixed left-0 top-0 z-[1001] grid h-screen w-screen grid-cols-4 grid-rows-[64px_1fr] overflow-y-hidden bg-black"
- bind:this={assetViewerHtmlElement}
- >
- <div class="z-[1000] col-span-4 col-start-1 row-span-1 row-start-1 transition-transform">
- {#if isSlideshowMode}
- <!-- SlideShowController -->
- <div class="flex">
- <div class="m-4 flex gap-2">
- <CircleIconButton icon={mdiClose} on:click={handleStopSlideshow} title="Exit Slideshow" />
- <CircleIconButton
- icon={progressBarStatus === ProgressBarStatus.Paused ? mdiPlay : mdiPause}
- on:click={() => (progressBarStatus === ProgressBarStatus.Paused ? progressBar.play() : progressBar.pause())}
- title={progressBarStatus === ProgressBarStatus.Paused ? 'Play' : 'Pause'}
- />
- <CircleIconButton icon={mdiChevronLeft} on:click={navigateAssetBackward} title="Previous" />
- <CircleIconButton icon={mdiChevronRight} on:click={navigateAssetForward} title="Next" />
- </div>
- <ProgressBar
- autoplay
- bind:this={progressBar}
- bind:status={progressBarStatus}
- on:done={navigateAssetForward}
- duration={5000}
- />
- </div>
- {:else}
- <AssetViewerNavBar
- {asset}
- isMotionPhotoPlaying={shouldPlayMotionPhoto}
- showCopyButton={canCopyImagesToClipboard && asset.type === AssetTypeEnum.Image}
- showZoomButton={asset.type === AssetTypeEnum.Image}
- showMotionPlayButton={!!asset.livePhotoVideoId}
- showDownloadButton={shouldShowDownloadButton}
- showDetailButton={shouldShowDetailButton}
- showSlideshow={!!assetStore}
- hasStackChildern={$stackAssetsStore.length > 0}
- on:goBack={closeViewer}
- on:showDetail={showDetailInfoHandler}
- on:download={() => downloadFile(asset)}
- on:delete={trashOrDelete}
- on:favorite={toggleFavorite}
- on:addToAlbum={() => openAlbumPicker(false)}
- on:addToSharedAlbum={() => openAlbumPicker(true)}
- on:playMotionPhoto={() => (shouldPlayMotionPhoto = true)}
- on:stopMotionPhoto={() => (shouldPlayMotionPhoto = false)}
- on:toggleArchive={toggleArchive}
- on:asProfileImage={() => (isShowProfileImageCrop = true)}
- on:runJob={({ detail: job }) => handleRunJob(job)}
- on:playSlideShow={handlePlaySlideshow}
- on:unstack={handleUnstack}
- />
- {/if}
- </div>
- {#if !isSlideshowMode && showNavigation}
- <div class="column-span-1 z-[999] col-start-1 row-span-1 row-start-2 mb-[60px] justify-self-start">
- <NavigationArea on:click={navigateAssetBackward}><Icon path={mdiChevronLeft} size="36" /></NavigationArea>
- </div>
- {/if}
- <!-- Asset Viewer -->
- <div class="relative col-span-4 col-start-1 row-span-full row-start-1">
- {#if previewStackedAsset}
- {#key previewStackedAsset.id}
- {#if previewStackedAsset.type === AssetTypeEnum.Image}
- <PhotoViewer asset={previewStackedAsset} on:close={closeViewer} haveFadeTransition={false} />
- {:else}
- <VideoViewer
- assetId={previewStackedAsset.id}
- on:close={closeViewer}
- on:onVideoEnded={handleVideoEnded}
- on:onVideoStarted={handleVideoStarted}
- />
- {/if}
- {/key}
- {:else}
- {#key asset.id}
- {#if !asset.resized}
- <div class="flex h-full w-full justify-center">
- <div
- class="px-auto flex aspect-square h-full items-center justify-center bg-gray-100 dark:bg-immich-dark-gray"
- >
- <Icon path={mdiImageBrokenVariant} size="25%" />
- </div>
- </div>
- {:else if asset.type === AssetTypeEnum.Image}
- {#if shouldPlayMotionPhoto && asset.livePhotoVideoId}
- <VideoViewer
- assetId={asset.livePhotoVideoId}
- on:close={closeViewer}
- on:onVideoEnded={() => (shouldPlayMotionPhoto = false)}
- />
- {:else if asset.exifInfo?.projectionType === ProjectionType.EQUIRECTANGULAR || (asset.originalPath && asset.originalPath
- .toLowerCase()
- .endsWith('.insp'))}
- <PanoramaViewer {asset} />
- {:else}
- <PhotoViewer {asset} on:close={closeViewer} />
- {/if}
- {:else}
- <VideoViewer
- assetId={asset.id}
- on:close={closeViewer}
- on:onVideoEnded={handleVideoEnded}
- on:onVideoStarted={handleVideoStarted}
- />
- {/if}
- {#if isShared}
- <div class="z-[9999] absolute bottom-0 right-0 mb-6 mr-6 justify-self-end">
- <div
- class="w-full h-14 flex p-4 text-white items-center justify-center rounded-full gap-4 bg-immich-dark-bg bg-opacity-60"
- >
- <button on:click={handleFavorite}>
- <div class="items-center justify-center">
- <Icon path={isLiked ? mdiHeart : mdiHeartOutline} size={24} />
- </div>
- </button>
- <button on:click={handleOpenActivity}>
- <div class="flex gap-2 items-center justify-center">
- <Icon path={mdiCommentOutline} class="scale-x-[-1]" size={24} />
- {#if numberOfComments}
- <div class="text-xl">{numberOfComments}</div>
- {:else if !isShowActivity && !$isShowDetail}
- <div class="text-lg">Say something</div>
- {/if}
- </div>
- </button>
- </div>
- </div>
- {/if}
- {/key}
- {/if}
- {#if $stackAssetsStore.length > 0 && withStacked}
- <div
- id="stack-slideshow"
- class="z-[1005] flex place-item-center place-content-center absolute bottom-0 w-full col-span-4 col-start-1 mb-1 overflow-x-auto horizontal-scrollbar"
- >
- <div class="relative whitespace-nowrap transition-all">
- {#each $stackAssetsStore as stackedAsset (stackedAsset.id)}
- <div
- class="{stackedAsset.id == asset.id
- ? '-translate-y-[1px]'
- : '-translate-y-0'} inline-block px-1 transition-transform"
- >
- <Thumbnail
- class="{stackedAsset.id == asset.id
- ? 'bg-transparent border-2 border-white'
- : 'bg-gray-700/40'} inline-block hover:bg-transparent"
- asset={stackedAsset}
- on:click={() => (asset = stackedAsset)}
- on:mouse-event={(e) => handleStackedAssetMouseEvent(e, stackedAsset)}
- readonly
- thumbnailSize={stackedAsset.id == asset.id ? 65 : 60}
- showStackedIcon={false}
- />
- {#if stackedAsset.id == asset.id}
- <div class="w-full flex place-items-center place-content-center">
- <div class="w-2 h-2 bg-white rounded-full flex mt-[2px]" />
- </div>
- {/if}
- </div>
- {/each}
- </div>
- </div>
- {/if}
- </div>
- <!-- Stack & Stack Controller -->
- {#if !isSlideshowMode && showNavigation}
- <div class="z-[999] col-span-1 col-start-4 row-span-1 row-start-2 mb-[60px] justify-self-end">
- <NavigationArea on:click={navigateAssetForward}><Icon path={mdiChevronRight} size="36" /></NavigationArea>
- </div>
- {/if}
- {#if !isSlideshowMode && $isShowDetail}
- <div
- transition:fly={{ duration: 150 }}
- id="detail-panel"
- class="z-[1002] row-start-1 row-span-5 w-[360px] overflow-y-auto bg-immich-bg transition-all dark:border-l dark:border-l-immich-dark-gray dark:bg-immich-dark-bg"
- translate="yes"
- >
- <DetailPanel
- {asset}
- albums={appearsInAlbums}
- on:close={() => ($isShowDetail = false)}
- on:close-viewer={handleCloseViewer}
- on:description-focus-in={disableKeyDownEvent}
- on:description-focus-out={enableKeyDownEvent}
- />
- </div>
- {/if}
- {#if isShared && album && isShowActivity && user}
- <div
- transition:fly={{ duration: 150 }}
- id="activity-panel"
- class="z-[1002] row-start-1 row-span-5 w-[460px] overflow-y-auto bg-immich-bg transition-all dark:border-l dark:border-l-immich-dark-gray dark:bg-immich-dark-bg pl-4"
- translate="yes"
- >
- <ActivityViewer
- {user}
- assetType={asset.type}
- albumOwnerId={album.ownerId}
- albumId={album.id}
- assetId={asset.id}
- bind:reactions
- on:addComment={() => numberOfComments++}
- on:deleteComment={() => numberOfComments--}
- on:deleteLike={() => (isLiked = null)}
- on:close={() => (isShowActivity = false)}
- />
- </div>
- {/if}
- {#if isShowAlbumPicker}
- <AlbumSelectionModal
- shared={addToSharedAlbum}
- on:newAlbum={handleAddToNewAlbum}
- on:newSharedAlbum={handleAddToNewAlbum}
- on:album={handleAddToAlbum}
- on:close={() => (isShowAlbumPicker = false)}
- />
- {/if}
- {#if isShowDeleteConfirmation}
- <ConfirmDialogue
- title="Delete {getAssetType(asset.type)}"
- confirmText="Delete"
- on:confirm={deleteAsset}
- on:cancel={() => (isShowDeleteConfirmation = false)}
- >
- <svelte:fragment slot="prompt">
- <p>
- Are you sure you want to delete this {getAssetType(asset.type).toLowerCase()}? This will also remove it from
- its album(s).
- </p>
- <p><b>You cannot undo this action!</b></p>
- </svelte:fragment>
- </ConfirmDialogue>
- {/if}
- {#if isShowProfileImageCrop}
- <ProfileImageCropper
- {asset}
- on:close={() => (isShowProfileImageCrop = false)}
- on:close-viewer={handleCloseViewer}
- />
- {/if}
- </section>
- <style>
- #immich-asset-viewer {
- contain: layout;
- }
- .horizontal-scrollbar::-webkit-scrollbar {
- width: 8px;
- height: 10px;
- }
- /* Track */
- .horizontal-scrollbar::-webkit-scrollbar-track {
- background: #000000;
- border-radius: 16px;
- }
- /* Handle */
- .horizontal-scrollbar::-webkit-scrollbar-thumb {
- background: rgba(159, 159, 159, 0.408);
- border-radius: 16px;
- }
- /* Handle on hover */
- .horizontal-scrollbar::-webkit-scrollbar-thumb:hover {
- background: #adcbfa;
- border-radius: 16px;
- }
- </style>
|