1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-07 20:36:48 +01:00

fix(web): improve focus and shortcuts (#7983)

* fix(web): improve focus and shortcuts

* fix shiftKeyIsDown
This commit is contained in:
Michel Heusschen 2024-03-15 17:01:35 +01:00 committed by GitHub
parent a46366d336
commit 029dd99ae0
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
7 changed files with 31 additions and 34 deletions

View file

@ -35,8 +35,6 @@
<tr <tr
class="flex h-[50px] w-full place-items-center border-[3px] border-transparent p-2 text-center odd:bg-immich-gray even:bg-immich-bg hover:cursor-pointer hover:border-immich-primary/75 odd:dark:bg-immich-dark-gray/75 even:dark:bg-immich-dark-gray/50 dark:hover:border-immich-dark-primary/75 md:p-5" class="flex h-[50px] w-full place-items-center border-[3px] border-transparent p-2 text-center odd:bg-immich-gray even:bg-immich-bg hover:cursor-pointer hover:border-immich-primary/75 odd:dark:bg-immich-dark-gray/75 even:dark:bg-immich-dark-gray/50 dark:hover:border-immich-dark-primary/75 md:p-5"
on:click={() => goto(`${AppRoute.ALBUMS}/${album.id}`)} on:click={() => goto(`${AppRoute.ALBUMS}/${album.id}`)}
on:keydown={(event) => event.key === 'Enter' && goto(`${AppRoute.ALBUMS}/${album.id}`)}
tabindex="0"
> >
<a data-sveltekit-preload-data="hover" class="flex w-full" href="{AppRoute.ALBUMS}/{album.id}"> <a data-sveltekit-preload-data="hover" class="flex w-full" href="{AppRoute.ALBUMS}/{album.id}">
<td class="text-md text-ellipsis text-left w-8/12 sm:w-4/12 md:w-4/12 xl:w-[30%] 2xl:w-[40%]" <td class="text-md text-ellipsis text-left w-8/12 sm:w-4/12 md:w-4/12 xl:w-[30%] 2xl:w-[40%]"

View file

@ -45,7 +45,7 @@
<!-- Image grid --> <!-- Image grid -->
<div class="flex flex-wrap gap-[2px]"> <div class="flex flex-wrap gap-[2px]">
{#each album.assets as asset (asset.id)} {#each album.assets as asset (asset.id)}
<Thumbnail {asset} on:click={() => (selectedThumbnail = asset)} selected={isSelected(asset.id)} /> <Thumbnail {asset} onClick={() => (selectedThumbnail = asset)} selected={isSelected(asset.id)} />
{/each} {/each}
</div> </div>
</section> </section>

View file

@ -646,7 +646,7 @@
? 'bg-transparent border-2 border-white' ? 'bg-transparent border-2 border-white'
: 'bg-gray-700/40'} inline-block hover:bg-transparent" : 'bg-gray-700/40'} inline-block hover:bg-transparent"
asset={stackedAsset} asset={stackedAsset}
on:click={() => { onClick={() => {
asset = stackedAsset; asset = stackedAsset;
preloadAssets = index + 1 >= $stackAssetsStore.length ? [] : [$stackAssetsStore[index + 1]]; preloadAssets = index + 1 >= $stackAssetsStore.length ? [] : [$stackAssetsStore[index + 1]];
}} }}

View file

@ -21,9 +21,9 @@
import { fade } from 'svelte/transition'; import { fade } from 'svelte/transition';
import ImageThumbnail from './image-thumbnail.svelte'; import ImageThumbnail from './image-thumbnail.svelte';
import VideoThumbnail from './video-thumbnail.svelte'; import VideoThumbnail from './video-thumbnail.svelte';
import { shortcut } from '$lib/utils/shortcut';
const dispatch = createEventDispatcher<{ const dispatch = createEventDispatcher<{
click: { asset: AssetResponseDto };
select: { asset: AssetResponseDto }; select: { asset: AssetResponseDto };
'mouse-event': { isMouseOver: boolean; selectedGroupIndex: number }; 'mouse-event': { isMouseOver: boolean; selectedGroupIndex: number };
}>(); }>();
@ -40,12 +40,13 @@
export let readonly = false; export let readonly = false;
export let showArchiveIcon = false; export let showArchiveIcon = false;
export let showStackedIcon = true; export let showStackedIcon = true;
export let intersecting = false; export let onClick: ((asset: AssetResponseDto) => void) | undefined = undefined;
let className = ''; let className = '';
export { className as class }; export { className as class };
let mouseOver = false; let mouseOver = false;
$: clickable = !disabled && onClick;
$: dispatch('mouse-event', { isMouseOver: mouseOver, selectedGroupIndex: groupIndex }); $: dispatch('mouse-event', { isMouseOver: mouseOver, selectedGroupIndex: groupIndex });
@ -62,14 +63,8 @@
})(); })();
const thumbnailClickedHandler = () => { const thumbnailClickedHandler = () => {
if (!disabled) { if (clickable) {
dispatch('click', { asset }); onClick?.(asset);
}
};
const thumbnailKeyDownHandler = (e: KeyboardEvent) => {
if (e.key === 'Enter') {
thumbnailClickedHandler();
} }
}; };
@ -89,20 +84,22 @@
}; };
</script> </script>
<IntersectionObserver once={false} on:intersected bind:intersecting> <IntersectionObserver once={false} on:intersected let:intersecting>
<!-- svelte-ignore a11y-no-static-element-interactions --> <!-- svelte-ignore a11y-no-noninteractive-tabindex -->
<div <div
style:width="{width}px" style:width="{width}px"
style:height="{height}px" style:height="{height}px"
class="group relative overflow-hidden {disabled class="group focus-visible:outline-none relative overflow-hidden {disabled
? 'bg-gray-300' ? 'bg-gray-300'
: 'bg-immich-primary/20 dark:bg-immich-dark-primary/20'}" : 'bg-immich-primary/20 dark:bg-immich-dark-primary/20'}"
class:cursor-not-allowed={disabled} class:cursor-not-allowed={disabled}
class:hover:cursor-pointer={!disabled} class:hover:cursor-pointer={clickable}
on:mouseenter={onMouseEnter} on:mouseenter={onMouseEnter}
on:mouseleave={onMouseLeave} on:mouseleave={onMouseLeave}
role={clickable ? 'button' : undefined}
tabindex={clickable ? 0 : undefined}
on:click={thumbnailClickedHandler} on:click={thumbnailClickedHandler}
on:keydown={thumbnailKeyDownHandler} use:shortcut={{ shortcut: { key: 'Enter' }, onShortcut: thumbnailClickedHandler }}
> >
{#if intersecting} {#if intersecting}
<div class="absolute z-20 h-full w-full {className}"> <div class="absolute z-20 h-full w-full {className}">
@ -140,6 +137,11 @@
class:rounded-xl={selected} class:rounded-xl={selected}
/> />
<!-- Outline on focus -->
<div
class="absolute size-full group-focus-visible:outline outline-4 -outline-offset-4 outline-immich-primary"
/>
<!-- Favorite asset star --> <!-- Favorite asset star -->
{#if !isSharedLink() && asset.isFavorite} {#if !isSharedLink() && asset.isFavorite}
<div class="absolute bottom-2 left-2 z-10"> <div class="absolute bottom-2 left-2 z-10">

View file

@ -178,7 +178,7 @@
{showArchiveIcon} {showArchiveIcon}
{asset} {asset}
{groupIndex} {groupIndex}
on:click={() => assetClickHandler(asset, groupAssets, groupTitle)} onClick={() => assetClickHandler(asset, groupAssets, groupTitle)}
on:select={() => assetSelectHandler(asset, groupAssets, groupTitle)} on:select={() => assetSelectHandler(asset, groupAssets, groupTitle)}
on:mouse-event={() => assetMouseEventHandler(groupTitle, asset)} on:mouse-event={() => assetMouseEventHandler(groupTitle, asset)}
selected={$selectedAssets.has(asset) || $assetStore.albumAssets.has(asset.id)} selected={$selectedAssets.has(asset) || $assetStore.albumAssets.has(asset.id)}

View file

@ -8,7 +8,7 @@
import { isSearchEnabled } from '$lib/stores/search.store'; import { isSearchEnabled } from '$lib/stores/search.store';
import { featureFlags } from '$lib/stores/server-config.store'; import { featureFlags } from '$lib/stores/server-config.store';
import { deleteAssets } from '$lib/utils/actions'; 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 { formatGroupTitle, splitBucketIntoDateGroups } from '$lib/utils/timeline-util';
import type { AlbumResponseDto, AssetResponseDto } from '@immich/sdk'; import type { AlbumResponseDto, AssetResponseDto } from '@immich/sdk';
import { DateTime } from 'luxon'; import { DateTime } from 'luxon';
@ -202,24 +202,24 @@
let shiftKeyIsDown = false; let shiftKeyIsDown = false;
const onKeyDown = (e: KeyboardEvent) => { const onKeyDown = (event: KeyboardEvent) => {
if ($isSearchEnabled) { if ($isSearchEnabled) {
return; return;
} }
if (e.key == 'Shift') { if (matchesShortcut(event, { key: 'Shift' })) {
e.preventDefault(); event.preventDefault();
shiftKeyIsDown = true; shiftKeyIsDown = true;
} }
}; };
const onKeyUp = (e: KeyboardEvent) => { const onKeyUp = (event: KeyboardEvent) => {
if ($isSearchEnabled) { if ($isSearchEnabled) {
return; return;
} }
if (e.key == 'Shift') { if (matchesShortcut(event, { key: 'Shift' })) {
e.preventDefault(); event.preventDefault();
shiftKeyIsDown = false; shiftKeyIsDown = false;
} }
}; };

View file

@ -26,17 +26,14 @@
let currentViewAssetIndex = 0; let currentViewAssetIndex = 0;
$: isMultiSelectionMode = selectedAssets.size > 0; $: isMultiSelectionMode = selectedAssets.size > 0;
const viewAssetHandler = (event: CustomEvent) => { const viewAssetHandler = (asset: AssetResponseDto) => {
const { asset }: { asset: AssetResponseDto } = event.detail;
currentViewAssetIndex = assets.findIndex((a) => a.id == asset.id); currentViewAssetIndex = assets.findIndex((a) => a.id == asset.id);
selectedAsset = assets[currentViewAssetIndex]; selectedAsset = assets[currentViewAssetIndex];
$showAssetViewer = true; $showAssetViewer = true;
updateAssetState(selectedAsset.id, false); updateAssetState(selectedAsset.id, false);
}; };
const selectAssetHandler = (event: CustomEvent) => { const selectAssetHandler = (asset: AssetResponseDto) => {
const { asset }: { asset: AssetResponseDto } = event.detail;
let temporary = new Set(selectedAssets); let temporary = new Set(selectedAssets);
if (selectedAssets.has(asset)) { if (selectedAssets.has(asset)) {
@ -123,8 +120,8 @@
<Thumbnail <Thumbnail
{asset} {asset}
readonly={disableAssetSelect} readonly={disableAssetSelect}
on:click={(e) => (isMultiSelectionMode ? selectAssetHandler(e) : viewAssetHandler(e))} onClick={(e) => (isMultiSelectionMode ? selectAssetHandler(e) : viewAssetHandler(e))}
on:select={selectAssetHandler} on:select={(e) => selectAssetHandler(e.detail.asset)}
on:intersected={(event) => on:intersected={(event) =>
i === Math.max(1, assets.length - 7) ? dispatch('intersected', event.detail) : undefined} i === Math.max(1, assets.length - 7) ? dispatch('intersected', event.detail) : undefined}
selected={selectedAssets.has(asset)} selected={selectedAssets.has(asset)}