1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-01 08:31:59 +00:00

feat(web): improve alt text (#7596)

* alt text

* memory lane alt text

* revert sql generator change

* use getAltText

* oops

* handle large number of people in asset

* nit

* add aria-label to search button

* update api

* fixed tests

* fixed typing

* fixed spacing

* fix displaying null
This commit is contained in:
Mert 2024-03-03 16:42:17 -05:00 committed by GitHub
parent 07c926bb12
commit 2fa10a254c
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
24 changed files with 143 additions and 51 deletions

Binary file not shown.

View file

@ -8291,6 +8291,14 @@
}, },
"MapMarkerResponseDto": { "MapMarkerResponseDto": {
"properties": { "properties": {
"city": {
"nullable": true,
"type": "string"
},
"country": {
"nullable": true,
"type": "string"
},
"id": { "id": {
"type": "string" "type": "string"
}, },
@ -8301,12 +8309,19 @@
"lon": { "lon": {
"format": "double", "format": "double",
"type": "number" "type": "number"
},
"state": {
"nullable": true,
"type": "string"
} }
}, },
"required": [ "required": [
"city",
"country",
"id", "id",
"lat", "lat",
"lon" "lon",
"state"
], ],
"type": "object" "type": "object"
}, },

View file

@ -260,9 +260,12 @@ export type AssetJobsDto = {
name: AssetJobName; name: AssetJobName;
}; };
export type MapMarkerResponseDto = { export type MapMarkerResponseDto = {
city: string | null;
country: string | null;
id: string; id: string;
lat: number; lat: number;
lon: number; lon: number;
state: string | null;
}; };
export type MemoryLaneResponseDto = { export type MemoryLaneResponseDto = {
assets: AssetResponseDto[]; assets: AssetResponseDto[];

View file

@ -286,27 +286,22 @@ describe(AssetService.name, () => {
describe('getMapMarkers', () => { describe('getMapMarkers', () => {
it('should get geo information of assets', async () => { it('should get geo information of assets', async () => {
const asset = assetStub.withLocation;
const marker = {
id: asset.id,
lat: asset.exifInfo!.latitude!,
lon: asset.exifInfo!.longitude!,
city: asset.exifInfo!.city,
state: asset.exifInfo!.state,
country: asset.exifInfo!.country,
};
partnerMock.getAll.mockResolvedValue([]); partnerMock.getAll.mockResolvedValue([]);
assetMock.getMapMarkers.mockResolvedValue( assetMock.getMapMarkers.mockResolvedValue([marker]);
[assetStub.withLocation].map((asset) => ({
id: asset.id,
/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */
lat: asset.exifInfo!.latitude!,
/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */
lon: asset.exifInfo!.longitude!,
})),
);
const markers = await sut.getMapMarkers(authStub.user1, {}); const markers = await sut.getMapMarkers(authStub.user1, {});
expect(markers).toHaveLength(1); expect(markers).toHaveLength(1);
expect(markers[0]).toEqual({ expect(markers[0]).toEqual(marker);
id: assetStub.withLocation.id,
lat: 100,
lon: 100,
});
}); });
}); });

View file

@ -9,4 +9,13 @@ export class MapMarkerResponseDto {
@ApiProperty({ format: 'double' }) @ApiProperty({ format: 'double' })
lon!: number; lon!: number;
@ApiProperty()
city!: string | null;
@ApiProperty()
state!: string | null;
@ApiProperty()
country!: string | null;
} }

View file

@ -1,4 +1,9 @@
import { AssetSearchOneToOneRelationOptions, AssetSearchOptions, SearchExploreItem } from '@app/domain'; import {
AssetSearchOneToOneRelationOptions,
AssetSearchOptions,
ReverseGeocodeResult,
SearchExploreItem,
} from '@app/domain';
import { AssetEntity, AssetJobStatusEntity, AssetType, ExifEntity } from '@app/infra/entities'; import { AssetEntity, AssetJobStatusEntity, AssetType, ExifEntity } from '@app/infra/entities';
import { FindOptionsRelations, FindOptionsSelect } from 'typeorm'; import { FindOptionsRelations, FindOptionsSelect } from 'typeorm';
import { Paginated, PaginationOptions } from '../domain.util'; import { Paginated, PaginationOptions } from '../domain.util';
@ -25,7 +30,7 @@ export interface MapMarkerSearchOptions {
fileCreatedAfter?: Date; fileCreatedAfter?: Date;
} }
export interface MapMarker { export interface MapMarker extends ReverseGeocodeResult {
id: string; id: string;
lat: number; lat: number;
lon: number; lon: number;

View file

@ -507,6 +507,9 @@ export class AssetRepository implements IAssetRepository {
select: { select: {
id: true, id: true,
exifInfo: { exifInfo: {
city: true,
state: true,
country: true,
latitude: true, latitude: true,
longitude: true, longitude: true,
}, },
@ -532,12 +535,11 @@ export class AssetRepository implements IAssetRepository {
return assets.map((asset) => ({ return assets.map((asset) => ({
id: asset.id, id: asset.id,
/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */
lat: asset.exifInfo!.latitude!, lat: asset.exifInfo!.latitude!,
/* eslint-disable-next-line @typescript-eslint/no-non-null-assertion */
lon: asset.exifInfo!.longitude!, lon: asset.exifInfo!.longitude!,
city: asset.exifInfo!.city,
state: asset.exifInfo!.state,
country: asset.exifInfo!.country,
})); }));
} }

View file

@ -482,6 +482,9 @@ export const assetStub = {
latitude: 100, latitude: 100,
longitude: 100, longitude: 100,
fileSizeInByte: 23_456, fileSizeInByte: 23_456,
city: 'test-city',
state: 'test-state',
country: 'test-country',
} as ExifEntity, } as ExifEntity,
deletedAt: null, deletedAt: null,
}), }),

View file

@ -47,11 +47,11 @@ describe('AlbumCard component', () => {
const detailsText = `${count} items` + (shared ? ' . Shared' : ''); const detailsText = `${count} items` + (shared ? ' . Shared' : '');
expect(albumImgElement).toHaveAttribute('src'); expect(albumImgElement).toHaveAttribute('src');
expect(albumImgElement).toHaveAttribute('alt', album.id); expect(albumImgElement).toHaveAttribute('alt', album.albumName);
await waitFor(() => expect(albumImgElement).toHaveAttribute('src')); await waitFor(() => expect(albumImgElement).toHaveAttribute('src'));
expect(albumImgElement).toHaveAttribute('alt', album.id); expect(albumImgElement).toHaveAttribute('alt', album.albumName);
expect(sdkMock.getAssetThumbnail).not.toHaveBeenCalled(); expect(sdkMock.getAssetThumbnail).not.toHaveBeenCalled();
expect(albumNameElement).toHaveTextContent(album.albumName); expect(albumNameElement).toHaveTextContent(album.albumName);
@ -74,11 +74,11 @@ describe('AlbumCard component', () => {
const albumImgElement = sut.getByTestId('album-image'); const albumImgElement = sut.getByTestId('album-image');
const albumNameElement = sut.getByTestId('album-name'); const albumNameElement = sut.getByTestId('album-name');
const albumDetailsElement = sut.getByTestId('album-details'); const albumDetailsElement = sut.getByTestId('album-details');
expect(albumImgElement).toHaveAttribute('alt', album.id); expect(albumImgElement).toHaveAttribute('alt', album.albumName);
await waitFor(() => expect(albumImgElement).toHaveAttribute('src', thumbnailUrl)); await waitFor(() => expect(albumImgElement).toHaveAttribute('src', thumbnailUrl));
expect(albumImgElement).toHaveAttribute('alt', album.id); expect(albumImgElement).toHaveAttribute('alt', album.albumName);
expect(sdkMock.getAssetThumbnail).toHaveBeenCalledTimes(1); expect(sdkMock.getAssetThumbnail).toHaveBeenCalledTimes(1);
expect(sdkMock.getAssetThumbnail).toHaveBeenCalledWith({ expect(sdkMock.getAssetThumbnail).toHaveBeenCalledWith({
id: 'thumbnailIdOne', id: 'thumbnailIdOne',

View file

@ -72,7 +72,7 @@
<img <img
loading={preload ? 'eager' : 'lazy'} loading={preload ? 'eager' : 'lazy'}
src={imageData} src={imageData}
alt={album.id} alt={album.albumName}
class="z-0 h-full w-full rounded-xl object-cover transition-all duration-300 hover:shadow-lg" class="z-0 h-full w-full rounded-xl object-cover transition-all duration-300 hover:shadow-lg"
data-testid="album-image" data-testid="album-image"
draggable="false" draggable="false"
@ -82,7 +82,7 @@
loading={preload ? 'eager' : 'lazy'} loading={preload ? 'eager' : 'lazy'}
src="$lib/assets/no-thumbnail.png" src="$lib/assets/no-thumbnail.png"
sizes="min(271px,186px)" sizes="min(271px,186px)"
alt={album.id} alt={album.albumName}
class="z-0 h-full w-full rounded-xl object-cover transition-all duration-300 hover:shadow-lg" class="z-0 h-full w-full rounded-xl object-cover transition-all duration-300 hover:shadow-lg"
data-testid="album-image" data-testid="album-image"
draggable="false" draggable="false"

View file

@ -195,7 +195,7 @@
<img <img
class="rounded-lg w-[75px] h-[75px] object-cover" class="rounded-lg w-[75px] h-[75px] object-cover"
src={getAssetThumbnailUrl(reaction.assetId, ThumbnailFormat.Webp)} src={getAssetThumbnailUrl(reaction.assetId, ThumbnailFormat.Webp)}
alt="comment-thumbnail" alt="Profile picture of {reaction.user.name}, who commented on this asset"
/> />
</div> </div>
{/if} {/if}
@ -241,7 +241,7 @@
<img <img
class="rounded-lg w-[75px] h-[75px] object-cover" class="rounded-lg w-[75px] h-[75px] object-cover"
src={getAssetThumbnailUrl(reaction.assetId, ThumbnailFormat.Webp)} src={getAssetThumbnailUrl(reaction.assetId, ThumbnailFormat.Webp)}
alt="like-thumbnail" alt="Profile picture of {reaction.user.name}, who liked this asset"
/> />
</div> </div>
{/if} {/if}

View file

@ -616,7 +616,16 @@
{:then component} {:then component}
<svelte:component <svelte:component
this={component.default} this={component.default}
mapMarkers={[{ lat: latlng.lat, lon: latlng.lng, id: asset.id }]} mapMarkers={[
{
lat: latlng.lat,
lon: latlng.lng,
id: asset.id,
city: asset.exifInfo?.city ?? null,
state: asset.exifInfo?.state ?? null,
country: asset.exifInfo?.country ?? null,
},
]}
center={latlng} center={latlng}
zoom={15} zoom={15}
simplified simplified

View file

@ -13,6 +13,7 @@
import { fade } from 'svelte/transition'; import { fade } from 'svelte/transition';
import LoadingSpinner from '../shared-components/loading-spinner.svelte'; import LoadingSpinner from '../shared-components/loading-spinner.svelte';
import { NotificationType, notificationController } from '../shared-components/notification/notification'; import { NotificationType, notificationController } from '../shared-components/notification/notification';
import { getAltText } from '$lib/utils/thumbnail-util';
export let asset: AssetResponseDto; export let asset: AssetResponseDto;
export let element: HTMLDivElement | undefined = undefined; export let element: HTMLDivElement | undefined = undefined;
@ -133,7 +134,7 @@
bind:this={$photoViewer} bind:this={$photoViewer}
transition:fade={{ duration: haveFadeTransition ? 150 : 0 }} transition:fade={{ duration: haveFadeTransition ? 150 : 0 }}
src={assetData} src={assetData}
alt={asset.id} alt={getAltText(asset)}
class="h-full w-full object-contain" class="h-full w-full object-contain"
draggable="false" draggable="false"
/> />

View file

@ -8,7 +8,7 @@
import Icon from '$lib/components/elements/icon.svelte'; import Icon from '$lib/components/elements/icon.svelte';
export let url: string; export let url: string;
export let altText: string; export let altText: string | undefined;
export let title: string | null = null; export let title: string | null = null;
export let heightStyle: string | undefined = undefined; export let heightStyle: string | undefined = undefined;
export let widthStyle: string; export let widthStyle: string;

View file

@ -3,6 +3,7 @@
import Icon from '$lib/components/elements/icon.svelte'; import Icon from '$lib/components/elements/icon.svelte';
import { ProjectionType } from '$lib/constants'; import { ProjectionType } from '$lib/constants';
import { getAssetFileUrl, getAssetThumbnailUrl, isSharedLink } from '$lib/utils'; import { getAssetFileUrl, getAssetThumbnailUrl, isSharedLink } from '$lib/utils';
import { getAltText } from '$lib/utils/thumbnail-util';
import { timeToSeconds } from '$lib/utils/date-time'; import { timeToSeconds } from '$lib/utils/date-time';
import { AssetTypeEnum, ThumbnailFormat, type AssetResponseDto } from '@immich/sdk'; import { AssetTypeEnum, ThumbnailFormat, type AssetResponseDto } from '@immich/sdk';
import { import {
@ -177,7 +178,7 @@
{#if asset.resized} {#if asset.resized}
<ImageThumbnail <ImageThumbnail
url={getAssetThumbnailUrl(asset.id, format)} url={getAssetThumbnailUrl(asset.id, format)}
altText={asset.originalFileName} altText={getAltText(asset)}
widthStyle="{width}px" widthStyle="{width}px"
heightStyle="{height}px" heightStyle="{height}px"
thumbhash={asset.thumbhash} thumbhash={asset.thumbhash}

View file

@ -223,14 +223,14 @@
url={selectedPersonToCreate[index] || url={selectedPersonToCreate[index] ||
getPeopleThumbnailUrl(selectedPersonToReassign[index]?.id || face.person.id)} getPeopleThumbnailUrl(selectedPersonToReassign[index]?.id || face.person.id)}
altText={selectedPersonToReassign[index] altText={selectedPersonToReassign[index]
? selectedPersonToReassign[index]?.name || '' ? selectedPersonToReassign[index]?.name
: selectedPersonToCreate[index] : selectedPersonToCreate[index]
? 'new person' ? 'New person'
: getPersonNameWithHiddenValue(face.person?.name, face.person?.isHidden)} : getPersonNameWithHiddenValue(face.person?.name, face.person?.isHidden)}
title={selectedPersonToReassign[index] title={selectedPersonToReassign[index]
? selectedPersonToReassign[index]?.name || '' ? selectedPersonToReassign[index]?.name
: selectedPersonToCreate[index] : selectedPersonToCreate[index]
? 'new person' ? 'New person'
: getPersonNameWithHiddenValue(face.person?.name, face.person?.isHidden)} : getPersonNameWithHiddenValue(face.person?.name, face.person?.isHidden)}
widthStyle="90px" widthStyle="90px"
heightStyle="90px" heightStyle="90px"

View file

@ -171,7 +171,7 @@
<img <img
class="h-full w-full rounded-2xl object-cover" class="h-full w-full rounded-2xl object-cover"
src={getAssetThumbnailUrl(previousMemory.assets[0].id, ThumbnailFormat.Jpeg)} src={getAssetThumbnailUrl(previousMemory.assets[0].id, ThumbnailFormat.Jpeg)}
alt="" alt="Previous memory"
draggable="false" draggable="false"
/> />
{:else} {:else}
@ -179,7 +179,7 @@
class="h-full w-full rounded-2xl object-cover" class="h-full w-full rounded-2xl object-cover"
src="$lib/assets/no-thumbnail.png" src="$lib/assets/no-thumbnail.png"
sizes="min(271px,186px)" sizes="min(271px,186px)"
alt="" alt="Previous memory"
draggable="false" draggable="false"
/> />
{/if} {/if}
@ -203,7 +203,7 @@
transition:fade transition:fade
class="h-full w-full rounded-2xl object-contain transition-all" class="h-full w-full rounded-2xl object-contain transition-all"
src={getAssetThumbnailUrl(currentAsset.id, ThumbnailFormat.Jpeg)} src={getAssetThumbnailUrl(currentAsset.id, ThumbnailFormat.Jpeg)}
alt="" alt={currentAsset.exifInfo?.description}
draggable="false" draggable="false"
/> />
{/key} {/key}
@ -244,7 +244,7 @@
<img <img
class="h-full w-full rounded-2xl object-cover" class="h-full w-full rounded-2xl object-cover"
src={getAssetThumbnailUrl(nextMemory.assets[0].id, ThumbnailFormat.Jpeg)} src={getAssetThumbnailUrl(nextMemory.assets[0].id, ThumbnailFormat.Jpeg)}
alt="" alt="Next memory"
draggable="false" draggable="false"
/> />
{:else} {:else}
@ -252,7 +252,7 @@
class="h-full w-full rounded-2xl object-cover" class="h-full w-full rounded-2xl object-cover"
src="$lib/assets/no-thumbnail.png" src="$lib/assets/no-thumbnail.png"
sizes="min(271px,186px)" sizes="min(271px,186px)"
alt="" alt="Next memory"
draggable="false" draggable="false"
/> />
{/if} {/if}

View file

@ -4,6 +4,7 @@
import { AppRoute, QueryParameter } from '$lib/constants'; import { AppRoute, QueryParameter } from '$lib/constants';
import { memoryStore } from '$lib/stores/memory.store'; import { memoryStore } from '$lib/stores/memory.store';
import { getAssetThumbnailUrl } from '$lib/utils'; import { getAssetThumbnailUrl } from '$lib/utils';
import { getAltText } from '$lib/utils/thumbnail-util';
import { ThumbnailFormat, getMemoryLane } from '@immich/sdk'; import { ThumbnailFormat, getMemoryLane } from '@immich/sdk';
import { mdiChevronLeft, mdiChevronRight } from '@mdi/js'; import { mdiChevronLeft, mdiChevronRight } from '@mdi/js';
import { onMount } from 'svelte'; import { onMount } from 'svelte';
@ -64,7 +65,6 @@
{/if} {/if}
</div> </div>
{/if} {/if}
<div class="inline-block" bind:offsetWidth={innerWidth}> <div class="inline-block" bind:offsetWidth={innerWidth}>
{#each $memoryStore as memory, index (memory.title)} {#each $memoryStore as memory, index (memory.title)}
<button <button
@ -74,7 +74,7 @@
<img <img
class="h-full w-full rounded-xl object-cover" class="h-full w-full rounded-xl object-cover"
src={getAssetThumbnailUrl(memory.assets[0].id, ThumbnailFormat.Jpeg)} src={getAssetThumbnailUrl(memory.assets[0].id, ThumbnailFormat.Jpeg)}
alt={memory.title} alt={`Memory Lane ${getAltText(memory.assets[0])}`}
draggable="false" draggable="false"
/> />
<p class="absolute bottom-2 left-4 z-10 text-lg text-white">{memory.title}</p> <p class="absolute bottom-2 left-4 z-10 text-lg text-white">{memory.title}</p>

View file

@ -192,7 +192,18 @@
{:then component} {:then component}
<svelte:component <svelte:component
this={component.default} this={component.default}
mapMarkers={lat && lng && asset ? [{ id: asset.id, lat, lon: lng }] : []} mapMarkers={lat && lng && asset
? [
{
id: asset.id,
lat,
lon: lng,
city: asset.exifInfo?.city ?? null,
state: asset.exifInfo?.state ?? null,
country: asset.exifInfo?.country ?? null,
},
]
: []}
{zoom} {zoom}
bind:addClipMapMarker bind:addClipMapMarker
center={lat && lng ? { lat, lng } : undefined} center={lat && lng ? { lat, lng } : undefined}

View file

@ -88,7 +88,7 @@
} }
} }
type FeaturePoint = Feature<Point, { id: string }>; type FeaturePoint = Feature<Point, { id: string; city: string | null; state: string | null; country: string | null }>;
const asFeature = (marker: MapMarkerResponseDto): FeaturePoint => { const asFeature = (marker: MapMarkerResponseDto): FeaturePoint => {
return { return {
@ -96,6 +96,9 @@
geometry: { type: 'Point', coordinates: [marker.lon, marker.lat] }, geometry: { type: 'Point', coordinates: [marker.lon, marker.lat] },
properties: { properties: {
id: marker.id, id: marker.id,
city: marker.city,
state: marker.state,
country: marker.country,
}, },
}; };
}; };
@ -107,6 +110,9 @@
lat: coords.lat, lat: coords.lat,
lon: coords.lng, lon: coords.lng,
id: featurePoint.properties.id, id: featurePoint.properties.id,
city: featurePoint.properties.city,
state: featurePoint.properties.state,
country: featurePoint.properties.country,
}; };
}; };
</script> </script>
@ -178,7 +184,9 @@
<img <img
src={getAssetThumbnailUrl(feature.properties?.id, undefined)} src={getAssetThumbnailUrl(feature.properties?.id, undefined)}
class="rounded-full w-[60px] h-[60px] border-2 border-immich-primary shadow-lg hover:border-immich-dark-primary transition-all duration-200 hover:scale-150 object-cover bg-immich-primary" class="rounded-full w-[60px] h-[60px] border-2 border-immich-primary shadow-lg hover:border-immich-dark-primary transition-all duration-200 hover:scale-150 object-cover bg-immich-primary"
alt={`Image with id ${feature.properties?.id}`} alt={feature.properties?.city && feature.properties.country
? `Map marker for images taken in ${feature.properties.city}, ${feature.properties.country}`
: 'Map marker with image'}
/> />
{/if} {/if}
{#if $$slots.popup} {#if $$slots.popup}

View file

@ -97,7 +97,7 @@
<div class="absolute inset-y-0 left-0 flex items-center pl-6"> <div class="absolute inset-y-0 left-0 flex items-center pl-6">
<div class="dark:text-immich-dark-fg/75"> <div class="dark:text-immich-dark-fg/75">
<button class="flex items-center"> <button class="flex items-center">
<Icon path={mdiMagnify} size="1.5em" /> <Icon ariaLabel="search" path={mdiMagnify} size="1.5em" />
</button> </button>
</div> </div>
</div> </div>
@ -134,7 +134,7 @@
type="reset" type="reset"
class="rounded-full p-2 hover:bg-immich-primary/5 active:bg-immich-primary/10 dark:text-immich-dark-fg/75 dark:hover:bg-immich-dark-primary/25 dark:active:bg-immich-dark-primary/[.35]" class="rounded-full p-2 hover:bg-immich-primary/5 active:bg-immich-primary/10 dark:text-immich-dark-fg/75 dark:hover:bg-immich-dark-primary/25 dark:active:bg-immich-dark-primary/[.35]"
> >
<Icon path={mdiClose} size="1.5em" /> <Icon ariaLabel="clear" path={mdiClose} size="1.5em" />
</button> </button>
</div> </div>
{/if} {/if}

View file

@ -1,3 +1,6 @@
import type { AssetResponseDto } from '@immich/sdk';
import { fromLocalDateTime } from './timeline-util';
/** /**
* Calculate thumbnail size based on number of assets and viewport width * Calculate thumbnail size based on number of assets and viewport width
* @param assetCount Number of assets in the view * @param assetCount Number of assets in the view
@ -31,3 +34,30 @@ export function getThumbnailSize(assetCount: number, viewWidth: number): number
return 300; return 300;
} }
export function getAltText(asset: AssetResponseDto) {
if (asset.exifInfo?.description) {
return asset.exifInfo.description;
}
let altText = 'Image taken';
if (asset.exifInfo?.city && asset.exifInfo.country) {
altText += ` in ${asset.exifInfo.city}, ${asset.exifInfo.country}`;
}
const names = asset.people?.filter((p) => p.name).map((p) => p.name) ?? [];
if (names.length == 1) {
altText += ` with ${names[0]}`;
}
if (names.length > 1 && names.length <= 3) {
altText += ` with ${names.slice(0, -1).join(', ')} and ${names.at(-1)}`;
}
if (names.length > 3) {
altText += ` with ${names.slice(0, 2).join(', ')}, and ${names.length - 2} others`;
}
const date = fromLocalDateTime(asset.localDateTime).toLocaleString({ dateStyle: 'long' });
altText += ` on ${date}`;
return altText;
}