refactor(web): set birthdate (#25139)

This commit is contained in:
Jason Rasmussen 2026-01-08 15:41:20 -05:00 committed by GitHub
parent a2ba36c16d
commit 6997ed83c4
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
6 changed files with 101 additions and 104 deletions

View file

@ -1,13 +1,14 @@
<script lang="ts"> <script lang="ts">
import { focusOutside } from '$lib/actions/focus-outside'; import { focusOutside } from '$lib/actions/focus-outside';
import ActionMenuItem from '$lib/components/ActionMenuItem.svelte';
import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte'; import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte';
import { AppRoute, QueryParameter } from '$lib/constants'; import { AppRoute, QueryParameter } from '$lib/constants';
import { getPersonActions } from '$lib/services/person.service';
import { getPeopleThumbnailUrl } from '$lib/utils'; import { getPeopleThumbnailUrl } from '$lib/utils';
import { type PersonResponseDto } from '@immich/sdk'; import { type PersonResponseDto } from '@immich/sdk';
import { Icon } from '@immich/ui'; import { Icon } from '@immich/ui';
import { import {
mdiAccountMultipleCheckOutline, mdiAccountMultipleCheckOutline,
mdiCalendarEditOutline,
mdiDotsVertical, mdiDotsVertical,
mdiEyeOffOutline, mdiEyeOffOutline,
mdiHeart, mdiHeart,
@ -18,17 +19,18 @@
import ImageThumbnail from '../assets/thumbnail/image-thumbnail.svelte'; import ImageThumbnail from '../assets/thumbnail/image-thumbnail.svelte';
import MenuOption from '../shared-components/context-menu/menu-option.svelte'; import MenuOption from '../shared-components/context-menu/menu-option.svelte';
interface Props { type Props = {
person: PersonResponseDto; person: PersonResponseDto;
onSetBirthDate: () => void;
onMergePeople: () => void; onMergePeople: () => void;
onHidePerson: () => void; onHidePerson: () => void;
onToggleFavorite: () => void; onToggleFavorite: () => void;
} };
let { person, onSetBirthDate, onMergePeople, onHidePerson, onToggleFavorite }: Props = $props(); let { person, onMergePeople, onHidePerson, onToggleFavorite }: Props = $props();
let showVerticalDots = $state(false); let showVerticalDots = $state(false);
const { SetDateOfBirth } = $derived(getPersonActions($t, person));
</script> </script>
<div <div
@ -73,7 +75,7 @@
title={$t('show_person_options')} title={$t('show_person_options')}
> >
<MenuOption onClick={onHidePerson} icon={mdiEyeOffOutline} text={$t('hide_person')} /> <MenuOption onClick={onHidePerson} icon={mdiEyeOffOutline} text={$t('hide_person')} />
<MenuOption onClick={onSetBirthDate} icon={mdiCalendarEditOutline} text={$t('set_date_of_birth')} /> <ActionMenuItem action={SetDateOfBirth} />
<MenuOption onClick={onMergePeople} icon={mdiAccountMultipleCheckOutline} text={$t('merge_people')} /> <MenuOption onClick={onMergePeople} icon={mdiAccountMultipleCheckOutline} text={$t('merge_people')} />
<MenuOption <MenuOption
onClick={onToggleFavorite} onClick={onToggleFavorite}

View file

@ -6,6 +6,7 @@ import type {
AssetResponseDto, AssetResponseDto,
LibraryResponseDto, LibraryResponseDto,
LoginResponseDto, LoginResponseDto,
PersonResponseDto,
QueueResponseDto, QueueResponseDto,
SharedLinkResponseDto, SharedLinkResponseDto,
SystemConfigDto, SystemConfigDto,
@ -33,6 +34,8 @@ export type Events = {
AlbumUpdate: [AlbumResponseDto]; AlbumUpdate: [AlbumResponseDto];
AlbumDelete: [AlbumResponseDto]; AlbumDelete: [AlbumResponseDto];
PersonUpdate: [PersonResponseDto];
QueueUpdate: [QueueResponseDto]; QueueUpdate: [QueueResponseDto];
SharedLinkCreate: [SharedLinkResponseDto]; SharedLinkCreate: [SharedLinkResponseDto];

View file

@ -1,73 +1,46 @@
<script lang="ts"> <script lang="ts">
import DateInput from '$lib/elements/DateInput.svelte'; import DateInput from '$lib/elements/DateInput.svelte';
import { handleError } from '$lib/utils/handle-error'; import { handleUpdatePersonBirthDate } from '$lib/services/person.service';
import { updatePerson, type PersonResponseDto } from '@immich/sdk'; import { type PersonResponseDto } from '@immich/sdk';
import { Button, HStack, Modal, ModalBody, ModalFooter, toastManager } from '@immich/ui'; import { Button, FormModal, Text } from '@immich/ui';
import { mdiCake } from '@mdi/js'; import { mdiCake } from '@mdi/js';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
interface Props { type Props = {
person: PersonResponseDto; person: PersonResponseDto;
onClose: (updatedPerson?: PersonResponseDto) => void; onClose: () => void;
} };
let { person, onClose }: Props = $props(); let { person, onClose }: Props = $props();
let birthDate = $state(person.birthDate ?? ''); let birthDate = $derived(person.birthDate ?? '');
const todayFormatted = new Date().toISOString().split('T')[0]; const onSubmit = async () => {
const success = await handleUpdatePersonBirthDate(person, birthDate);
const handleUpdateBirthDate = async () => { if (success) {
try { onClose();
const updatedPerson = await updatePerson({
id: person.id,
personUpdateDto: { birthDate },
});
toastManager.success($t('date_of_birth_saved'));
onClose(updatedPerson);
} catch (error) {
handleError(error, $t('errors.unable_to_save_date_of_birth'));
} }
}; };
const todayFormatted = new Date().toISOString().split('T')[0];
</script> </script>
<Modal title={$t('set_date_of_birth')} icon={mdiCake} {onClose} size="small"> <FormModal title={$t('set_date_of_birth')} size="small" icon={mdiCake} {onClose} {onSubmit}>
<ModalBody> <Text size="small">{$t('birthdate_set_description')}</Text>
<div class="text-primary"> <div class="my-4 flex flex-col gap-2">
<p class="text-sm dark:text-immich-dark-fg"> <DateInput
{$t('birthdate_set_description')} class="immich-form-input"
</p> id="birthDate"
</div> name="birthDate"
type="date"
<form onsubmit={() => handleUpdateBirthDate()} autocomplete="off" id="set-birth-date-form"> bind:value={birthDate}
<div class="my-4 flex flex-col gap-2"> max={todayFormatted}
<DateInput />
class="immich-form-input" {#if person.birthDate}
id="birthDate" <div class="flex justify-end">
name="birthDate" <Button shape="round" color="secondary" size="small" onclick={() => (birthDate = '')}>
type="date" {$t('clear')}
bind:value={birthDate} </Button>
max={todayFormatted}
/>
{#if person.birthDate}
<div class="flex justify-end">
<Button shape="round" color="secondary" size="small" onclick={() => (birthDate = '')}>
{$t('clear')}
</Button>
</div>
{/if}
</div> </div>
</form> {/if}
</ModalBody> </div>
</FormModal>
<ModalFooter>
<HStack fullWidth>
<Button shape="round" color="secondary" fullWidth onclick={() => onClose()}>
{$t('cancel')}
</Button>
<Button type="submit" shape="round" color="primary" fullWidth form="set-birth-date-form">
{$t('save')}
</Button>
</HStack>
</ModalFooter>
</Modal>

View file

@ -0,0 +1,31 @@
import { eventManager } from '$lib/managers/event-manager.svelte';
import PersonEditBirthDateModal from '$lib/modals/PersonEditBirthDateModal.svelte';
import { handleError } from '$lib/utils/handle-error';
import { getFormatter } from '$lib/utils/i18n';
import { updatePerson, type PersonResponseDto } from '@immich/sdk';
import { modalManager, toastManager, type ActionItem } from '@immich/ui';
import { mdiCalendarEditOutline } from '@mdi/js';
import type { MessageFormatter } from 'svelte-i18n';
export const getPersonActions = ($t: MessageFormatter, person: PersonResponseDto) => {
const SetDateOfBirth: ActionItem = {
title: $t('set_date_of_birth'),
icon: mdiCalendarEditOutline,
onAction: () => modalManager.show(PersonEditBirthDateModal, { person }),
};
return { SetDateOfBirth };
};
export const handleUpdatePersonBirthDate = async (person: PersonResponseDto, birthDate: string) => {
const $t = await getFormatter();
try {
const response = await updatePerson({ id: person.id, personUpdateDto: { birthDate } });
toastManager.success($t('date_of_birth_saved'));
eventManager.emit('PersonUpdate', response);
return true;
} catch (error) {
handleError(error, $t('errors.unable_to_save_date_of_birth'));
}
};

View file

@ -9,8 +9,8 @@
import PeopleInfiniteScroll from '$lib/components/faces-page/people-infinite-scroll.svelte'; import PeopleInfiniteScroll from '$lib/components/faces-page/people-infinite-scroll.svelte';
import SearchPeople from '$lib/components/faces-page/people-search.svelte'; import SearchPeople from '$lib/components/faces-page/people-search.svelte';
import UserPageLayout from '$lib/components/layouts/user-page-layout.svelte'; import UserPageLayout from '$lib/components/layouts/user-page-layout.svelte';
import OnEvents from '$lib/components/OnEvents.svelte';
import { ActionQueryParameterValue, AppRoute, QueryParameter, SessionStorageKey } from '$lib/constants'; import { ActionQueryParameterValue, AppRoute, QueryParameter, SessionStorageKey } from '$lib/constants';
import PersonEditBirthDateModal from '$lib/modals/PersonEditBirthDateModal.svelte';
import PersonMergeSuggestionModal from '$lib/modals/PersonMergeSuggestionModal.svelte'; import PersonMergeSuggestionModal from '$lib/modals/PersonMergeSuggestionModal.svelte';
import { locale } from '$lib/stores/preferences.store'; import { locale } from '$lib/stores/preferences.store';
import { websocketEvents } from '$lib/stores/websocket'; import { websocketEvents } from '$lib/stores/websocket';
@ -210,21 +210,6 @@
); );
}; };
const handleChangeBirthDate = async (person: PersonResponseDto) => {
const updatedPerson = await modalManager.show(PersonEditBirthDateModal, { person });
if (!updatedPerson) {
return;
}
people = people.map((person: PersonResponseDto) => {
if (person.id === updatedPerson.id) {
return updatedPerson;
}
return person;
});
};
const onResetSearchBar = async () => { const onResetSearchBar = async () => {
await clearQueryParam(QueryParameter.SEARCHED_PEOPLE, $page.url); await clearQueryParam(QueryParameter.SEARCHED_PEOPLE, $page.url);
}; };
@ -293,10 +278,21 @@
(person) => person.name.toLowerCase() === name.toLowerCase() && person.id !== personId && person.name, (person) => person.name.toLowerCase() === name.toLowerCase() && person.id !== personId && person.name,
); );
}; };
const onPersonUpdate = (response: PersonResponseDto) => {
people = people.map((person: PersonResponseDto) => {
if (person.id === response.id) {
return response;
}
return person;
});
};
</script> </script>
<svelte:window bind:innerHeight /> <svelte:window bind:innerHeight />
<OnEvents {onPersonUpdate} />
<UserPageLayout <UserPageLayout
title={$t('people')} title={$t('people')}
description={countVisiblePeople === 0 && !searchName ? undefined : `(${countVisiblePeople.toLocaleString($locale)})`} description={countVisiblePeople === 0 && !searchName ? undefined : `(${countVisiblePeople.toLocaleString($locale)})`}
@ -353,7 +349,6 @@
> >
<PeopleCard <PeopleCard
{person} {person}
onSetBirthDate={() => handleChangeBirthDate(person)}
onMergePeople={() => handleMergePeople(person)} onMergePeople={() => handleMergePeople(person)}
onHidePerson={() => handleHidePerson(person)} onHidePerson={() => handleHidePerson(person)}
onToggleFavorite={() => handleToggleFavorite(person)} onToggleFavorite={() => handleToggleFavorite(person)}

View file

@ -4,10 +4,12 @@
import { clickOutside } from '$lib/actions/click-outside'; import { clickOutside } from '$lib/actions/click-outside';
import { listNavigation } from '$lib/actions/list-navigation'; import { listNavigation } from '$lib/actions/list-navigation';
import { scrollMemoryClearer } from '$lib/actions/scroll-memory'; import { scrollMemoryClearer } from '$lib/actions/scroll-memory';
import ActionMenuItem from '$lib/components/ActionMenuItem.svelte';
import ImageThumbnail from '$lib/components/assets/thumbnail/image-thumbnail.svelte'; import ImageThumbnail from '$lib/components/assets/thumbnail/image-thumbnail.svelte';
import EditNameInput from '$lib/components/faces-page/edit-name-input.svelte'; import EditNameInput from '$lib/components/faces-page/edit-name-input.svelte';
import MergeFaceSelector from '$lib/components/faces-page/merge-face-selector.svelte'; import MergeFaceSelector from '$lib/components/faces-page/merge-face-selector.svelte';
import UnMergeFaceSelector from '$lib/components/faces-page/unmerge-face-selector.svelte'; import UnMergeFaceSelector from '$lib/components/faces-page/unmerge-face-selector.svelte';
import OnEvents from '$lib/components/OnEvents.svelte';
import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte'; import ButtonContextMenu from '$lib/components/shared-components/context-menu/button-context-menu.svelte';
import MenuOption from '$lib/components/shared-components/context-menu/menu-option.svelte'; import MenuOption from '$lib/components/shared-components/context-menu/menu-option.svelte';
import ControlAppBar from '$lib/components/shared-components/control-app-bar.svelte'; import ControlAppBar from '$lib/components/shared-components/control-app-bar.svelte';
@ -28,8 +30,8 @@
import { AppRoute, PersonPageViewMode, QueryParameter, SessionStorageKey } from '$lib/constants'; import { AppRoute, PersonPageViewMode, QueryParameter, SessionStorageKey } from '$lib/constants';
import { TimelineManager } from '$lib/managers/timeline-manager/timeline-manager.svelte'; import { TimelineManager } from '$lib/managers/timeline-manager/timeline-manager.svelte';
import type { TimelineAsset } from '$lib/managers/timeline-manager/types'; import type { TimelineAsset } from '$lib/managers/timeline-manager/types';
import PersonEditBirthDateModal from '$lib/modals/PersonEditBirthDateModal.svelte';
import PersonMergeSuggestionModal from '$lib/modals/PersonMergeSuggestionModal.svelte'; import PersonMergeSuggestionModal from '$lib/modals/PersonMergeSuggestionModal.svelte';
import { getPersonActions } from '$lib/services/person.service';
import { AssetInteraction } from '$lib/stores/asset-interaction.svelte'; import { AssetInteraction } from '$lib/stores/asset-interaction.svelte';
import { assetViewingStore } from '$lib/stores/asset-viewing.store'; import { assetViewingStore } from '$lib/stores/asset-viewing.store';
import { locale } from '$lib/stores/preferences.store'; import { locale } from '$lib/stores/preferences.store';
@ -50,7 +52,6 @@
mdiAccountBoxOutline, mdiAccountBoxOutline,
mdiAccountMultipleCheckOutline, mdiAccountMultipleCheckOutline,
mdiArrowLeft, mdiArrowLeft,
mdiCalendarEditOutline,
mdiDotsVertical, mdiDotsVertical,
mdiEyeOffOutline, mdiEyeOffOutline,
mdiEyeOutline, mdiEyeOutline,
@ -79,7 +80,6 @@
let viewMode: PersonPageViewMode = $state(PersonPageViewMode.VIEW_ASSETS); let viewMode: PersonPageViewMode = $state(PersonPageViewMode.VIEW_ASSETS);
let isEditingName = $state(false); let isEditingName = $state(false);
let previousRoute: string = $state(AppRoute.EXPLORE); let previousRoute: string = $state(AppRoute.EXPLORE);
let people: PersonResponseDto[] = [];
let personMerge1: PersonResponseDto | undefined = $state(); let personMerge1: PersonResponseDto | undefined = $state();
let personMerge2: PersonResponseDto | undefined = $state(); let personMerge2: PersonResponseDto | undefined = $state();
let potentialMergePeople: PersonResponseDto[] = $state([]); let potentialMergePeople: PersonResponseDto[] = $state([]);
@ -223,9 +223,8 @@
return { merged: false }; return { merged: false };
} }
const [personToMerge, personToBeMergedInto] = result; const [, personToBeMergedInto] = result;
people = people.filter((person: PersonResponseDto) => person.id !== personToMerge.id);
if (personToBeMergedInto.name != personName && person.id === personToBeMergedInto.id) { if (personToBeMergedInto.name != personName && person.id === personToBeMergedInto.id) {
await updateAssetCount(); await updateAssetCount();
return { merged: true }; return { merged: true };
@ -309,22 +308,6 @@
await changeName(); await changeName();
}; };
const handleSetBirthDate = async () => {
const updatedPerson = await modalManager.show(PersonEditBirthDateModal, { person });
if (!updatedPerson) {
return;
}
person = updatedPerson;
people = people.map((person: PersonResponseDto) => {
if (person.id === updatedPerson.id) {
return updatedPerson;
}
return person;
});
};
const handleGoBack = async () => { const handleGoBack = async () => {
viewMode = PersonPageViewMode.VIEW_ASSETS; viewMode = PersonPageViewMode.VIEW_ASSETS;
if ($page.url.searchParams.has(QueryParameter.ACTION)) { if ($page.url.searchParams.has(QueryParameter.ACTION)) {
@ -351,8 +334,18 @@
timelineManager.removeAssets(assetIds); timelineManager.removeAssets(assetIds);
assetInteraction.clearMultiselect(); assetInteraction.clearMultiselect();
}; };
const onPersonUpdate = (response: PersonResponseDto) => {
if (person.id === response.id) {
return (person = response);
}
};
const { SetDateOfBirth } = $derived(getPersonActions($t, person));
</script> </script>
<OnEvents {onPersonUpdate} />
<main <main
class="relative z-0 h-dvh overflow-hidden px-2 md:px-6 md:pt-(--navbar-height-md) pt-(--navbar-height)" class="relative z-0 h-dvh overflow-hidden px-2 md:px-6 md:pt-(--navbar-height-md) pt-(--navbar-height)"
use:scrollMemoryClearer={{ use:scrollMemoryClearer={{
@ -535,7 +528,7 @@
icon={person.isHidden ? mdiEyeOutline : mdiEyeOffOutline} icon={person.isHidden ? mdiEyeOutline : mdiEyeOffOutline}
onClick={() => toggleHidePerson()} onClick={() => toggleHidePerson()}
/> />
<MenuOption text={$t('set_date_of_birth')} icon={mdiCalendarEditOutline} onClick={handleSetBirthDate} /> <ActionMenuItem action={SetDateOfBirth} />
<MenuOption <MenuOption
text={$t('merge_people')} text={$t('merge_people')}
icon={mdiAccountMultipleCheckOutline} icon={mdiAccountMultipleCheckOutline}