1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-04 02:46:47 +01:00

feat(web): Duplicate-Page shortcut changes (#11183)

* duplicate page assign other shortcut keys, add 'open image' shortcut

* add shortcut info page to duplicates with own list of keys

* edit translations, add translationkeys

* format fix

* remove typo

---------

Co-authored-by: Zack Pollard <zackpollard@ymail.com>
Co-authored-by: Alex <alex.tran1502@gmail.com>
This commit is contained in:
Jan 2024-07-26 23:47:51 +02:00 committed by GitHub
parent a78eeb9b9c
commit e1ac73718c
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 85 additions and 40 deletions

View file

@ -16,7 +16,7 @@
info?: string; info?: string;
} }
const shortcuts: Shortcuts = { export let shortcuts: Shortcuts = {
general: [ general: [
{ key: ['←', '→'], action: $t('previous_or_next_photo') }, { key: ['←', '→'], action: $t('previous_or_next_photo') },
{ key: ['Esc'], action: $t('back_close_deselect') }, { key: ['Esc'], action: $t('back_close_deselect') },
@ -40,45 +40,48 @@
<FullScreenModal title={$t('keyboard_shortcuts')} width="auto" onClose={() => dispatch('close')}> <FullScreenModal title={$t('keyboard_shortcuts')} width="auto" onClose={() => dispatch('close')}>
<div class="grid grid-cols-1 gap-4 px-4 pb-4 md:grid-cols-2"> <div class="grid grid-cols-1 gap-4 px-4 pb-4 md:grid-cols-2">
<div class="p-4"> {#if shortcuts.general.length > 0}
<h2>{$t('general')}</h2> <div class="p-4">
<div class="text-sm"> <h2>{$t('general')}</h2>
{#each shortcuts.general as shortcut} <div class="text-sm">
<div class="grid grid-cols-[30%_70%] items-center gap-4 pt-4 text-sm"> {#each shortcuts.general as shortcut}
<div class="flex justify-self-end"> <div class="grid grid-cols-[30%_70%] items-center gap-4 pt-4 text-sm">
{#each shortcut.key as key} <div class="flex justify-self-end">
<p class="mr-1 flex items-center justify-center justify-self-end rounded-lg bg-immich-primary/25 p-2"> {#each shortcut.key as key}
{key} <p class="mr-1 flex items-center justify-center justify-self-end rounded-lg bg-immich-primary/25 p-2">
</p> {key}
{/each} </p>
</div> {/each}
<p class="mb-1 mt-1 flex">{shortcut.action}</p> </div>
</div>
{/each}
</div>
</div>
<div class="p-4">
<h2>{$t('actions')}</h2>
<div class="text-sm">
{#each shortcuts.actions as shortcut}
<div class="grid grid-cols-[30%_70%] items-center gap-4 pt-4 text-sm">
<div class="flex justify-self-end">
{#each shortcut.key as key}
<p class="mr-1 flex items-center justify-center justify-self-end rounded-lg bg-immich-primary/25 p-2">
{key}
</p>
{/each}
</div>
<div class="flex items-center gap-2">
<p class="mb-1 mt-1 flex">{shortcut.action}</p> <p class="mb-1 mt-1 flex">{shortcut.action}</p>
{#if shortcut.info}
<Icon path={mdiInformationOutline} title={shortcut.info} />
{/if}
</div> </div>
</div> {/each}
{/each} </div>
</div> </div>
</div> {/if}
{#if shortcuts.actions.length > 0}
<div class="p-4">
<h2>{$t('actions')}</h2>
<div class="text-sm">
{#each shortcuts.actions as shortcut}
<div class="grid grid-cols-[30%_70%] items-center gap-4 pt-4 text-sm">
<div class="flex justify-self-end">
{#each shortcut.key as key}
<p class="mr-1 flex items-center justify-center justify-self-end rounded-lg bg-immich-primary/25 p-2">
{key}
</p>
{/each}
</div>
<div class="flex items-center gap-2">
<p class="mb-1 mt-1 flex">{shortcut.action}</p>
{#if shortcut.info}
<Icon path={mdiInformationOutline} title={shortcut.info} />
{/if}
</div>
</div>
{/each}
</div>
</div>
{/if}
</div> </div>
</FullScreenModal> </FullScreenModal>

View file

@ -64,8 +64,14 @@
<svelte:window <svelte:window
use:shortcuts={[ use:shortcuts={[
{ shortcut: { key: 'k', shift: true }, onShortcut: onSelectAll }, { shortcut: { key: 'a' }, onShortcut: onSelectAll },
{ shortcut: { key: 't', shift: true }, onShortcut: onSelectNone }, {
shortcut: { key: 's' },
onShortcut: () => {
setAsset(assets[0]);
},
},
{ shortcut: { key: 'd' }, onShortcut: onSelectNone },
{ shortcut: { key: 'c', shift: true }, onShortcut: handleResolve }, { shortcut: { key: 'c', shift: true }, onShortcut: handleResolve },
]} ]}
/> />

View file

@ -997,6 +997,7 @@
"reset_password": "Reset password", "reset_password": "Reset password",
"reset_people_visibility": "Reset people visibility", "reset_people_visibility": "Reset people visibility",
"reset_to_default": "Reset to default", "reset_to_default": "Reset to default",
"resolve_duplicates": "Resolve duplicates",
"resolved_all_duplicates": "Resolved all duplicates", "resolved_all_duplicates": "Resolved all duplicates",
"restore": "Restore", "restore": "Restore",
"restore_all": "Restore all", "restore_all": "Restore all",
@ -1041,6 +1042,7 @@
"see_all_people": "See all people", "see_all_people": "See all people",
"select_album_cover": "Select album cover", "select_album_cover": "Select album cover",
"select_all": "Select all", "select_all": "Select all",
"select_all_duplicates": "Select all duplicates",
"select_avatar_color": "Select avatar color", "select_avatar_color": "Select avatar color",
"select_face": "Select face", "select_face": "Select face",
"select_featured_photo": "Select featured photo", "select_featured_photo": "Select featured photo",
@ -1166,6 +1168,7 @@
"unnamed_share": "Unnamed Share", "unnamed_share": "Unnamed Share",
"unsaved_change": "Unsaved change", "unsaved_change": "Unsaved change",
"unselect_all": "Unselect all", "unselect_all": "Unselect all",
"unselect_all_duplicates": "Unselect all duplicates",
"unstack": "Un-stack", "unstack": "Un-stack",
"unstacked_assets_count": "Un-stacked {count, plural, one {# asset} other {# assets}}", "unstacked_assets_count": "Un-stacked {count, plural, one {# asset} other {# assets}}",
"untracked_files": "Untracked files", "untracked_files": "Untracked files",

View file

@ -13,10 +13,34 @@
import type { PageData } from './$types'; import type { PageData } from './$types';
import { suggestDuplicateByFileSize } from '$lib/utils'; import { suggestDuplicateByFileSize } from '$lib/utils';
import LinkButton from '$lib/components/elements/buttons/link-button.svelte'; import LinkButton from '$lib/components/elements/buttons/link-button.svelte';
import ShowShortcuts from '$lib/components/shared-components/show-shortcuts.svelte';
import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte';
import { mdiKeyboard } from '@mdi/js';
import { mdiCheckOutline, mdiTrashCanOutline } from '@mdi/js'; import { mdiCheckOutline, mdiTrashCanOutline } from '@mdi/js';
import Icon from '$lib/components/elements/icon.svelte'; import Icon from '$lib/components/elements/icon.svelte';
export let data: PageData; export let data: PageData;
export let isShowKeyboardShortcut = false;
interface Shortcuts {
general: ExplainedShortcut[];
actions: ExplainedShortcut[];
}
interface ExplainedShortcut {
key: string[];
action: string;
info?: string;
}
const duplicateShortcuts: Shortcuts = {
general: [],
actions: [
{ key: ['a'], action: $t('select_all_duplicates') },
{ key: ['s'], action: $t('view') },
{ key: ['d'], action: $t('unselect_all_duplicates') },
{ key: ['⇧', 'c'], action: $t('resolve_duplicates') },
],
};
$: hasDuplicates = data.duplicates.length > 0; $: hasDuplicates = data.duplicates.length > 0;
@ -132,6 +156,11 @@
{$t('keep_all')} {$t('keep_all')}
</div> </div>
</LinkButton> </LinkButton>
<CircleIconButton
icon={mdiKeyboard}
title={$t('show_keyboard_shortcuts')}
on:click={() => (isShowKeyboardShortcut = !isShowKeyboardShortcut)}
/>
</div> </div>
<div class="mt-4"> <div class="mt-4">
@ -153,3 +182,7 @@
{/if} {/if}
</div> </div>
</UserPageLayout> </UserPageLayout>
{#if isShowKeyboardShortcut}
<ShowShortcuts shortcuts={duplicateShortcuts} on:close={() => (isShowKeyboardShortcut = false)} />
{/if}