From 029dd99ae03334b32b59cf6640bebc240b03fb1f Mon Sep 17 00:00:00 2001
From: Michel Heusschen <59014050+michelheusschen@users.noreply.github.com>
Date: Fri, 15 Mar 2024 17:01:35 +0100
Subject: [PATCH] fix(web): improve focus and shortcuts (#7983)
* fix(web): improve focus and shortcuts
* fix shiftKeyIsDown
---
.../components/album-page/albums-table.svelte | 2 --
.../album-page/thumbnail-selection.svelte | 2 +-
.../asset-viewer/asset-viewer.svelte | 2 +-
.../assets/thumbnail/thumbnail.svelte | 32 ++++++++++---------
.../photos-page/asset-date-group.svelte | 2 +-
.../components/photos-page/asset-grid.svelte | 14 ++++----
.../gallery-viewer/gallery-viewer.svelte | 11 +++----
7 files changed, 31 insertions(+), 34 deletions(-)
diff --git a/web/src/lib/components/album-page/albums-table.svelte b/web/src/lib/components/album-page/albums-table.svelte
index 467d8d2c56..e77dc93d88 100644
--- a/web/src/lib/components/album-page/albums-table.svelte
+++ b/web/src/lib/components/album-page/albums-table.svelte
@@ -35,8 +35,6 @@
goto(`${AppRoute.ALBUMS}/${album.id}`)}
- on:keydown={(event) => event.key === 'Enter' && goto(`${AppRoute.ALBUMS}/${album.id}`)}
- tabindex="0"
>
{#each album.assets as asset (asset.id)}
- (selectedThumbnail = asset)} selected={isSelected(asset.id)} />
+ (selectedThumbnail = asset)} selected={isSelected(asset.id)} />
{/each}
diff --git a/web/src/lib/components/asset-viewer/asset-viewer.svelte b/web/src/lib/components/asset-viewer/asset-viewer.svelte
index 2ddd7ada4f..f9d2f787bb 100644
--- a/web/src/lib/components/asset-viewer/asset-viewer.svelte
+++ b/web/src/lib/components/asset-viewer/asset-viewer.svelte
@@ -646,7 +646,7 @@
? 'bg-transparent border-2 border-white'
: 'bg-gray-700/40'} inline-block hover:bg-transparent"
asset={stackedAsset}
- on:click={() => {
+ onClick={() => {
asset = stackedAsset;
preloadAssets = index + 1 >= $stackAssetsStore.length ? [] : [$stackAssetsStore[index + 1]];
}}
diff --git a/web/src/lib/components/assets/thumbnail/thumbnail.svelte b/web/src/lib/components/assets/thumbnail/thumbnail.svelte
index 796be13be7..efbe384244 100644
--- a/web/src/lib/components/assets/thumbnail/thumbnail.svelte
+++ b/web/src/lib/components/assets/thumbnail/thumbnail.svelte
@@ -21,9 +21,9 @@
import { fade } from 'svelte/transition';
import ImageThumbnail from './image-thumbnail.svelte';
import VideoThumbnail from './video-thumbnail.svelte';
+ import { shortcut } from '$lib/utils/shortcut';
const dispatch = createEventDispatcher<{
- click: { asset: AssetResponseDto };
select: { asset: AssetResponseDto };
'mouse-event': { isMouseOver: boolean; selectedGroupIndex: number };
}>();
@@ -40,12 +40,13 @@
export let readonly = false;
export let showArchiveIcon = false;
export let showStackedIcon = true;
- export let intersecting = false;
+ export let onClick: ((asset: AssetResponseDto) => void) | undefined = undefined;
let className = '';
export { className as class };
let mouseOver = false;
+ $: clickable = !disabled && onClick;
$: dispatch('mouse-event', { isMouseOver: mouseOver, selectedGroupIndex: groupIndex });
@@ -62,14 +63,8 @@
})();
const thumbnailClickedHandler = () => {
- if (!disabled) {
- dispatch('click', { asset });
- }
- };
-
- const thumbnailKeyDownHandler = (e: KeyboardEvent) => {
- if (e.key === 'Enter') {
- thumbnailClickedHandler();
+ if (clickable) {
+ onClick?.(asset);
}
};
@@ -89,20 +84,22 @@
};
-
-
+
+
{#if intersecting}
@@ -140,6 +137,11 @@
class:rounded-xl={selected}
/>
+
+
+
{#if !isSharedLink() && asset.isFavorite}
diff --git a/web/src/lib/components/photos-page/asset-date-group.svelte b/web/src/lib/components/photos-page/asset-date-group.svelte
index 6f7af44efa..e3af259843 100644
--- a/web/src/lib/components/photos-page/asset-date-group.svelte
+++ b/web/src/lib/components/photos-page/asset-date-group.svelte
@@ -178,7 +178,7 @@
{showArchiveIcon}
{asset}
{groupIndex}
- on:click={() => assetClickHandler(asset, groupAssets, groupTitle)}
+ onClick={() => assetClickHandler(asset, groupAssets, groupTitle)}
on:select={() => assetSelectHandler(asset, groupAssets, groupTitle)}
on:mouse-event={() => assetMouseEventHandler(groupTitle, asset)}
selected={$selectedAssets.has(asset) || $assetStore.albumAssets.has(asset.id)}
diff --git a/web/src/lib/components/photos-page/asset-grid.svelte b/web/src/lib/components/photos-page/asset-grid.svelte
index cf2619026e..82224babea 100644
--- a/web/src/lib/components/photos-page/asset-grid.svelte
+++ b/web/src/lib/components/photos-page/asset-grid.svelte
@@ -8,7 +8,7 @@
import { isSearchEnabled } from '$lib/stores/search.store';
import { featureFlags } from '$lib/stores/server-config.store';
import { deleteAssets } from '$lib/utils/actions';
- import { shortcuts, type ShortcutOptions } from '$lib/utils/shortcut';
+ import { shortcuts, type ShortcutOptions, matchesShortcut } from '$lib/utils/shortcut';
import { formatGroupTitle, splitBucketIntoDateGroups } from '$lib/utils/timeline-util';
import type { AlbumResponseDto, AssetResponseDto } from '@immich/sdk';
import { DateTime } from 'luxon';
@@ -202,24 +202,24 @@
let shiftKeyIsDown = false;
- const onKeyDown = (e: KeyboardEvent) => {
+ const onKeyDown = (event: KeyboardEvent) => {
if ($isSearchEnabled) {
return;
}
- if (e.key == 'Shift') {
- e.preventDefault();
+ if (matchesShortcut(event, { key: 'Shift' })) {
+ event.preventDefault();
shiftKeyIsDown = true;
}
};
- const onKeyUp = (e: KeyboardEvent) => {
+ const onKeyUp = (event: KeyboardEvent) => {
if ($isSearchEnabled) {
return;
}
- if (e.key == 'Shift') {
- e.preventDefault();
+ if (matchesShortcut(event, { key: 'Shift' })) {
+ event.preventDefault();
shiftKeyIsDown = false;
}
};
diff --git a/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte b/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte
index 04345c4c55..9219227d0e 100644
--- a/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte
+++ b/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte
@@ -26,17 +26,14 @@
let currentViewAssetIndex = 0;
$: isMultiSelectionMode = selectedAssets.size > 0;
- const viewAssetHandler = (event: CustomEvent) => {
- const { asset }: { asset: AssetResponseDto } = event.detail;
-
+ const viewAssetHandler = (asset: AssetResponseDto) => {
currentViewAssetIndex = assets.findIndex((a) => a.id == asset.id);
selectedAsset = assets[currentViewAssetIndex];
$showAssetViewer = true;
updateAssetState(selectedAsset.id, false);
};
- const selectAssetHandler = (event: CustomEvent) => {
- const { asset }: { asset: AssetResponseDto } = event.detail;
+ const selectAssetHandler = (asset: AssetResponseDto) => {
let temporary = new Set(selectedAssets);
if (selectedAssets.has(asset)) {
@@ -123,8 +120,8 @@
(isMultiSelectionMode ? selectAssetHandler(e) : viewAssetHandler(e))}
- on:select={selectAssetHandler}
+ onClick={(e) => (isMultiSelectionMode ? selectAssetHandler(e) : viewAssetHandler(e))}
+ on:select={(e) => selectAssetHandler(e.detail.asset)}
on:intersected={(event) =>
i === Math.max(1, assets.length - 7) ? dispatch('intersected', event.detail) : undefined}
selected={selectedAssets.has(asset)}
|