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

feat(web): add cover images to individual shares (#9988)

* feat(web): add cover images to individual shares

* Update wording in share modal

* Use translation function

* Add and use new translations

* Fix formatting

* Update with suggestions

* Update test language

* Update test and language file per suggestions

* Fix formatting

* Remove unused translation
This commit is contained in:
Snowknight26 2024-06-14 18:16:48 -05:00 committed by GitHub
parent 78f600ebce
commit aea1c46bea
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
13 changed files with 214 additions and 30 deletions

View file

@ -0,0 +1,42 @@
import AlbumCover from '$lib/components/album-page/album-cover.svelte';
import { getAssetThumbnailUrl } from '$lib/utils';
import { albumFactory } from '@test-data';
import { render } from '@testing-library/svelte';
vi.mock('$lib/utils');
describe('AlbumCover component', () => {
it('renders an image when the album has a thumbnail', () => {
vi.mocked(getAssetThumbnailUrl).mockReturnValue('/asdf');
const component = render(AlbumCover, {
album: albumFactory.build({
albumName: 'someName',
albumThumbnailAssetId: '123',
}),
preload: false,
class: 'text',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('someName');
expect(img.getAttribute('loading')).toBe('lazy');
expect(img.className).toBe('z-0 rounded-xl object-cover text');
expect(img.getAttribute('src')).toBe('/asdf');
expect(getAssetThumbnailUrl).toHaveBeenCalledWith({ id: '123' });
});
it('renders an image when the album has no thumbnail', () => {
const component = render(AlbumCover, {
album: albumFactory.build({
albumName: '',
albumThumbnailAssetId: null,
}),
preload: true,
class: 'asdf',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('unnamed_album');
expect(img.getAttribute('loading')).toBe('eager');
expect(img.className).toBe('z-0 rounded-xl object-cover asdf');
expect(img.getAttribute('src')).toStrictEqual(expect.any(String));
});
});

View file

@ -46,7 +46,7 @@
</div> </div>
{/if} {/if}
<AlbumCover {album} {preload} css="h-full w-full transition-all duration-300 hover:shadow-lg" /> <AlbumCover {album} {preload} class="h-full w-full transition-all duration-300 hover:shadow-lg" />
<div class="mt-4"> <div class="mt-4">
<p <p

View file

@ -1,35 +1,23 @@
<script lang="ts"> <script lang="ts">
import { getAssetThumbnailUrl } from '$lib/utils'; import { getAssetThumbnailUrl } from '$lib/utils';
import { type AlbumResponseDto } from '@immich/sdk'; import { type AlbumResponseDto } from '@immich/sdk';
import NoCover from '$lib/components/sharedlinks-page/covers/no-cover.svelte';
import AssetCover from '$lib/components/sharedlinks-page/covers/asset-cover.svelte';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
export let album: AlbumResponseDto | undefined; export let album: AlbumResponseDto;
export let preload = false; export let preload = false;
export let css = ''; let className = '';
export { className as class };
$: thumbnailUrl = $: alt = album.albumName || $t('unnamed_album');
album && album.albumThumbnailAssetId ? getAssetThumbnailUrl({ id: album.albumThumbnailAssetId }) : null; $: thumbnailUrl = album.albumThumbnailAssetId ? getAssetThumbnailUrl({ id: album.albumThumbnailAssetId }) : null;
</script> </script>
<div class="relative aspect-square"> <div class="relative aspect-square">
{#if thumbnailUrl} {#if thumbnailUrl}
<img <AssetCover {alt} class={className} src={thumbnailUrl} {preload} />
loading={preload ? 'eager' : 'lazy'}
src={thumbnailUrl}
alt={album?.albumName ?? $t('unknown_album')}
class="z-0 rounded-xl object-cover {css}"
data-testid="album-image"
draggable="false"
/>
{:else} {:else}
<enhanced:img <NoCover {alt} class={className} {preload} />
loading={preload ? 'eager' : 'lazy'}
src="$lib/assets/no-thumbnail.png"
sizes="min(271px,186px)"
alt={album?.albumName ?? $t('empty_album')}
class="z-0 rounded-xl object-cover {css}"
data-testid="album-image"
draggable="false"
/>
{/if} {/if}
</div> </div>

View file

@ -41,7 +41,7 @@
<form on:submit|preventDefault={handleUpdateAlbumInfo} autocomplete="off" id="edit-album-form"> <form on:submit|preventDefault={handleUpdateAlbumInfo} autocomplete="off" id="edit-album-form">
<div class="flex items-center"> <div class="flex items-center">
<div class="hidden sm:flex"> <div class="hidden sm:flex">
<AlbumCover {album} css="h-[200px] w-[200px] m-4 shadow-lg" /> <AlbumCover {album} class="h-[200px] w-[200px] m-4 shadow-lg" />
</div> </div>
<div class="flex-grow"> <div class="flex-grow">

View file

@ -193,9 +193,8 @@
<div>Let anyone with the link see the selected photo(s)</div> <div>Let anyone with the link see the selected photo(s)</div>
{:else} {:else}
<div class="text-sm"> <div class="text-sm">
Individual shared | <span class="text-immich-primary dark:text-immich-dark-primary" {$t('individual_share')} |
>{editingLink.description || ''}</span <span class="text-immich-primary dark:text-immich-dark-primary">{editingLink.description || ''}</span>
>
</div> </div>
{/if} {/if}
{/if} {/if}

View file

@ -0,0 +1,18 @@
import AssetCover from '$lib/components/sharedlinks-page/covers/asset-cover.svelte';
import { render } from '@testing-library/svelte';
describe('AssetCover component', () => {
it('renders correctly', () => {
const component = render(AssetCover, {
alt: '123',
preload: true,
src: 'wee',
class: 'asdf',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('123');
expect(img.getAttribute('src')).toBe('wee');
expect(img.getAttribute('loading')).toBe('eager');
expect(img.className).toBe('z-0 rounded-xl object-cover asdf');
});
});

View file

@ -0,0 +1,17 @@
import NoCover from '$lib/components/sharedlinks-page/covers/no-cover.svelte';
import { render } from '@testing-library/svelte';
describe('NoCover component', () => {
it('renders correctly', () => {
const component = render(NoCover, {
alt: '123',
preload: true,
class: 'asdf',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('123');
expect(img.className).toBe('z-0 rounded-xl object-cover asdf');
expect(img.getAttribute('loading')).toBe('eager');
expect(img.src).toStrictEqual(expect.any(String));
});
});

View file

@ -0,0 +1,60 @@
import ShareCover from '$lib/components/sharedlinks-page/covers/share-cover.svelte';
import { getAssetThumbnailUrl } from '$lib/utils';
import type { SharedLinkResponseDto } from '@immich/sdk';
import { albumFactory } from '@test-data';
import { render } from '@testing-library/svelte';
vi.mock('$lib/utils');
describe('ShareCover component', () => {
it('renders an image when the shared link is an album', () => {
const component = render(ShareCover, {
link: {
album: albumFactory.build({
albumName: '123',
}),
} as SharedLinkResponseDto,
preload: false,
class: 'text',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('123');
expect(img.getAttribute('loading')).toBe('lazy');
expect(img.className).toBe('z-0 rounded-xl object-cover text');
});
it('renders an image when the shared link is an individual share', () => {
vi.mocked(getAssetThumbnailUrl).mockReturnValue('/asdf');
const component = render(ShareCover, {
link: {
assets: [
{
id: 'someId',
},
],
} as SharedLinkResponseDto,
preload: false,
class: 'text',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('individual_share');
expect(img.getAttribute('loading')).toBe('lazy');
expect(img.className).toBe('z-0 rounded-xl object-cover text');
expect(img.getAttribute('src')).toBe('/asdf');
expect(getAssetThumbnailUrl).toHaveBeenCalledWith('someId');
});
it('renders an image when the shared link has no album or assets', () => {
const component = render(ShareCover, {
link: {
assets: [],
} as unknown as SharedLinkResponseDto,
preload: false,
class: 'text',
});
const img = component.getByTestId('album-image') as HTMLImageElement;
expect(img.alt).toBe('unnamed_share');
expect(img.getAttribute('loading')).toBe('lazy');
expect(img.className).toBe('z-0 rounded-xl object-cover text');
});
});

View file

@ -0,0 +1,16 @@
<script lang="ts">
export let alt;
export let preload = false;
export let src: string;
let className = '';
export { className as class };
</script>
<img
{alt}
class="z-0 rounded-xl object-cover {className}"
data-testid="album-image"
draggable="false"
loading={preload ? 'eager' : 'lazy'}
{src}
/>

View file

@ -0,0 +1,16 @@
<script lang="ts">
export let alt = '';
export let preload = false;
let className = '';
export { className as class };
</script>
<enhanced:img
{alt}
class="z-0 rounded-xl object-cover {className}"
data-testid="album-image"
draggable="false"
loading={preload ? 'eager' : 'lazy'}
sizes="min(271px,186px)"
src="$lib/assets/no-thumbnail.png"
/>

View file

@ -0,0 +1,28 @@
<script lang="ts">
import type { SharedLinkResponseDto } from '@immich/sdk';
import AlbumCover from '$lib/components/album-page/album-cover.svelte';
import NoCover from '$lib/components/sharedlinks-page/covers/no-cover.svelte';
import AssetCover from '$lib/components/sharedlinks-page/covers/asset-cover.svelte';
import { getAssetThumbnailUrl } from '$lib/utils';
import { t } from 'svelte-i18n';
export let link: SharedLinkResponseDto;
export let preload = false;
let className = '';
export { className as class };
</script>
<div class="relative aspect-square">
{#if link?.album}
<AlbumCover album={link.album} class={className} {preload} />
{:else if link.assets[0]}
<AssetCover
alt={$t('individual_share')}
class={className}
{preload}
src={getAssetThumbnailUrl(link.assets[0].id)}
/>
{:else}
<NoCover alt={$t('unnamed_share')} class={className} {preload} />
{/if}
</div>

View file

@ -7,7 +7,7 @@
import { createEventDispatcher } from 'svelte'; import { createEventDispatcher } from 'svelte';
import CircleIconButton from '../elements/buttons/circle-icon-button.svelte'; import CircleIconButton from '../elements/buttons/circle-icon-button.svelte';
import { locale } from '$lib/stores/preferences.store'; import { locale } from '$lib/stores/preferences.store';
import AlbumCover from '$lib/components/album-page/album-cover.svelte'; import ShareCover from '$lib/components/sharedlinks-page/covers/share-cover.svelte';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
export let link: SharedLinkResponseDto; export let link: SharedLinkResponseDto;
@ -52,7 +52,7 @@
class="flex w-full gap-4 border-b border-gray-200 py-4 transition-all hover:border-immich-primary dark:border-gray-600 dark:text-immich-gray dark:hover:border-immich-dark-primary" class="flex w-full gap-4 border-b border-gray-200 py-4 transition-all hover:border-immich-primary dark:border-gray-600 dark:text-immich-gray dark:hover:border-immich-dark-primary"
> >
<div> <div>
<AlbumCover album={link?.album} css="h-[100px] w-[100px] transition-all duration-300 hover:shadow-lg" /> <ShareCover class="h-[100px] w-[100px] transition-all duration-300 hover:shadow-lg" {link} />
</div> </div>
<div class="flex flex-col justify-between"> <div class="flex flex-col justify-between">

View file

@ -445,7 +445,6 @@
"edited": "Edited", "edited": "Edited",
"editor": "Editor", "editor": "Editor",
"email": "Email", "email": "Email",
"empty_album": "Empty Album",
"empty_trash": "Empty trash", "empty_trash": "Empty trash",
"end_date": "End date", "end_date": "End date",
"error": "Error", "error": "Error",
@ -868,11 +867,12 @@
"unfavorite": "Unfavorite", "unfavorite": "Unfavorite",
"unhide_person": "Unhide person", "unhide_person": "Unhide person",
"unknown": "Unknown", "unknown": "Unknown",
"unknown_album": "Unknown Album",
"unknown_year": "Unknown Year", "unknown_year": "Unknown Year",
"unlimited": "Unlimited", "unlimited": "Unlimited",
"unlink_oauth": "Unlink Oauth", "unlink_oauth": "Unlink Oauth",
"unlinked_oauth_account": "Unlinked OAuth account", "unlinked_oauth_account": "Unlinked OAuth account",
"unnamed_album": "Unnamed Album",
"unnamed_share": "Unnamed Share",
"unselect_all": "Unselect all", "unselect_all": "Unselect all",
"unstack": "Un-stack", "unstack": "Un-stack",
"untracked_files": "Untracked files", "untracked_files": "Untracked files",