mirror of
https://github.com/immich-app/immich.git
synced 2025-01-19 18:26:46 +01:00
fix(web): shared link expiration date accessibility (#12060)
- use native select - shows focus, automatically has keyboard navigation, accessible for screen readers - remove DropdownButton component - fix dropdown styling in Safari
This commit is contained in:
parent
6b6d2a6621
commit
9894b9513b
3 changed files with 46 additions and 113 deletions
|
@ -8,7 +8,6 @@
|
||||||
import { SharedLinkType, createSharedLink, updateSharedLink, type SharedLinkResponseDto } from '@immich/sdk';
|
import { SharedLinkType, createSharedLink, updateSharedLink, type SharedLinkResponseDto } from '@immich/sdk';
|
||||||
import { mdiContentCopy, mdiLink } from '@mdi/js';
|
import { mdiContentCopy, mdiLink } from '@mdi/js';
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
import DropdownButton, { type DropDownOption } from '../dropdown-button.svelte';
|
|
||||||
import { NotificationType, notificationController } from '../notification/notification';
|
import { NotificationType, notificationController } from '../notification/notification';
|
||||||
import SettingInputField, { SettingInputFieldType } from '../settings/setting-input-field.svelte';
|
import SettingInputField, { SettingInputFieldType } from '../settings/setting-input-field.svelte';
|
||||||
import SettingSwitch from '../settings/setting-switch.svelte';
|
import SettingSwitch from '../settings/setting-switch.svelte';
|
||||||
|
@ -16,6 +15,7 @@
|
||||||
import { t } from 'svelte-i18n';
|
import { t } from 'svelte-i18n';
|
||||||
import { locale } from '$lib/stores/preferences.store';
|
import { locale } from '$lib/stores/preferences.store';
|
||||||
import { DateTime, Duration } from 'luxon';
|
import { DateTime, Duration } from 'luxon';
|
||||||
|
import SettingSelect from '$lib/components/shared-components/settings/setting-select.svelte';
|
||||||
|
|
||||||
export let onClose: () => void;
|
export let onClose: () => void;
|
||||||
export let albumId: string | undefined = undefined;
|
export let albumId: string | undefined = undefined;
|
||||||
|
@ -27,7 +27,7 @@
|
||||||
let allowDownload = true;
|
let allowDownload = true;
|
||||||
let allowUpload = false;
|
let allowUpload = false;
|
||||||
let showMetadata = true;
|
let showMetadata = true;
|
||||||
let expirationOption: DropDownOption<number> | undefined;
|
let expirationOption: number = 0;
|
||||||
let password = '';
|
let password = '';
|
||||||
let shouldChangeExpirationTime = false;
|
let shouldChangeExpirationTime = false;
|
||||||
let enablePassword = false;
|
let enablePassword = false;
|
||||||
|
@ -48,14 +48,12 @@
|
||||||
];
|
];
|
||||||
|
|
||||||
$: relativeTime = new Intl.RelativeTimeFormat($locale);
|
$: relativeTime = new Intl.RelativeTimeFormat($locale);
|
||||||
$: expiredDateOption = [
|
$: expiredDateOptions = [
|
||||||
{ label: $t('never'), value: 0 },
|
{ text: $t('never'), value: 0 },
|
||||||
...expirationOptions.map(
|
...expirationOptions.map(([value, unit]) => ({
|
||||||
([value, unit]): DropDownOption<number> => ({
|
text: relativeTime.format(value, unit),
|
||||||
label: relativeTime.format(value, unit),
|
|
||||||
value: Duration.fromObject({ [unit]: value }).toMillis(),
|
value: Duration.fromObject({ [unit]: value }).toMillis(),
|
||||||
}),
|
})),
|
||||||
),
|
|
||||||
];
|
];
|
||||||
|
|
||||||
$: shareType = albumId ? SharedLinkType.Album : SharedLinkType.Individual;
|
$: shareType = albumId ? SharedLinkType.Album : SharedLinkType.Individual;
|
||||||
|
@ -82,8 +80,7 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
const handleCreateSharedLink = async () => {
|
const handleCreateSharedLink = async () => {
|
||||||
const expirationDate =
|
const expirationDate = expirationOption > 0 ? DateTime.now().plus(expirationOption).toISO() : undefined;
|
||||||
expirationOption && expirationOption.value > 0 ? DateTime.now().plus(expirationOption.value).toISO() : undefined;
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const data = await createSharedLink({
|
const data = await createSharedLink({
|
||||||
|
@ -112,8 +109,7 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
const expirationDate =
|
const expirationDate = expirationOption > 0 ? DateTime.now().plus(expirationOption).toISO() : null;
|
||||||
expirationOption && expirationOption.value > 0 ? DateTime.now().plus(expirationOption.value).toISO() : null;
|
|
||||||
|
|
||||||
await updateSharedLink({
|
await updateSharedLink({
|
||||||
id: editingLink.id,
|
id: editingLink.id,
|
||||||
|
@ -212,19 +208,18 @@
|
||||||
<SettingSwitch bind:checked={allowUpload} title={$t('allow_public_user_to_upload')} />
|
<SettingSwitch bind:checked={allowUpload} title={$t('allow_public_user_to_upload')} />
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="text-sm">
|
|
||||||
{#if editingLink}
|
{#if editingLink}
|
||||||
<p class="immich-form-label my-2">
|
<div class="my-3">
|
||||||
<SettingSwitch bind:checked={shouldChangeExpirationTime} title={$t('change_expiration_time')} />
|
<SettingSwitch bind:checked={shouldChangeExpirationTime} title={$t('change_expiration_time')} />
|
||||||
</p>
|
</div>
|
||||||
{:else}
|
|
||||||
<p class="immich-form-label my-2">{$t('expire_after')}</p>
|
|
||||||
{/if}
|
{/if}
|
||||||
|
<div class="mt-3">
|
||||||
<DropdownButton
|
<SettingSelect
|
||||||
options={expiredDateOption}
|
bind:value={expirationOption}
|
||||||
bind:selected={expirationOption}
|
options={expiredDateOptions}
|
||||||
|
label={$t('expire_after')}
|
||||||
disabled={editingLink && !shouldChangeExpirationTime}
|
disabled={editingLink && !shouldChangeExpirationTime}
|
||||||
|
number={true}
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -1,74 +0,0 @@
|
||||||
<script lang="ts" context="module">
|
|
||||||
export type DropDownOption<T = unknown> = {
|
|
||||||
label: string;
|
|
||||||
value: T;
|
|
||||||
};
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<script lang="ts">
|
|
||||||
export let options: DropDownOption[];
|
|
||||||
export let selected = options.at(0);
|
|
||||||
export let disabled = false;
|
|
||||||
|
|
||||||
export let isOpen = false;
|
|
||||||
const toggle = () => (isOpen = !isOpen);
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<div id="immich-dropdown" class="relative">
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
{disabled}
|
|
||||||
on:click={toggle}
|
|
||||||
aria-expanded={isOpen}
|
|
||||||
class="flex w-full place-items-center justify-between rounded-lg bg-gray-200 p-2 disabled:cursor-not-allowed disabled:bg-gray-600 dark:bg-gray-600 dark:disabled:bg-gray-300"
|
|
||||||
>
|
|
||||||
{#if selected}
|
|
||||||
<div>
|
|
||||||
{selected.label}
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<div>
|
|
||||||
<svg
|
|
||||||
style="tran"
|
|
||||||
width="20"
|
|
||||||
height="20"
|
|
||||||
fill="none"
|
|
||||||
stroke-linecap="round"
|
|
||||||
stroke-linejoin="round"
|
|
||||||
stroke-width="2"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
stroke="currentColor"
|
|
||||||
>
|
|
||||||
<path d="M19 9l-7 7-7-7" />
|
|
||||||
</svg>
|
|
||||||
</div>
|
|
||||||
</button>
|
|
||||||
|
|
||||||
{#if isOpen}
|
|
||||||
<div class="absolute mt-2 flex w-full flex-col">
|
|
||||||
{#each options as option}
|
|
||||||
<button
|
|
||||||
type="button"
|
|
||||||
on:click={() => {
|
|
||||||
selected = option;
|
|
||||||
isOpen = false;
|
|
||||||
}}
|
|
||||||
class="flex w-full bg-gray-200 p-2 transition-all hover:bg-gray-300 dark:bg-gray-500 dark:hover:bg-gray-700"
|
|
||||||
>
|
|
||||||
{option.label}
|
|
||||||
</button>
|
|
||||||
{/each}
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
svg {
|
|
||||||
transition: transform 0.2s ease-in;
|
|
||||||
}
|
|
||||||
|
|
||||||
[aria-expanded='true'] svg {
|
|
||||||
transform: rotate(0.5turn);
|
|
||||||
}
|
|
||||||
</style>
|
|
|
@ -3,6 +3,8 @@
|
||||||
import { fly } from 'svelte/transition';
|
import { fly } from 'svelte/transition';
|
||||||
import { createEventDispatcher } from 'svelte';
|
import { createEventDispatcher } from 'svelte';
|
||||||
import { t } from 'svelte-i18n';
|
import { t } from 'svelte-i18n';
|
||||||
|
import Icon from '$lib/components/elements/icon.svelte';
|
||||||
|
import { mdiChevronDown } from '@mdi/js';
|
||||||
|
|
||||||
export let value: string | number;
|
export let value: string | number;
|
||||||
export let options: { value: string | number; text: string }[];
|
export let options: { value: string | number; text: string }[];
|
||||||
|
@ -46,8 +48,17 @@
|
||||||
</p>
|
</p>
|
||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
|
<div class="grid">
|
||||||
|
<Icon
|
||||||
|
path={mdiChevronDown}
|
||||||
|
size={'1.2em'}
|
||||||
|
ariaHidden={true}
|
||||||
|
class="pointer-events-none right-1 relative col-start-1 row-start-1 self-center justify-self-end {disabled
|
||||||
|
? 'text-immich-bg'
|
||||||
|
: 'text-immich-fg dark:text-immich-bg'}"
|
||||||
|
/>
|
||||||
<select
|
<select
|
||||||
class="immich-form-input w-full pb-2"
|
class="immich-form-input w-full appearance-none row-start-1 col-start-1 !pr-6"
|
||||||
{disabled}
|
{disabled}
|
||||||
aria-describedby={desc ? `${name}-desc` : undefined}
|
aria-describedby={desc ? `${name}-desc` : undefined}
|
||||||
{name}
|
{name}
|
||||||
|
@ -59,4 +70,5 @@
|
||||||
<option value={option.value}>{option.text}</option>
|
<option value={option.value}>{option.text}</option>
|
||||||
{/each}
|
{/each}
|
||||||
</select>
|
</select>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
Loading…
Reference in a new issue