mirror of
https://github.com/samsonjs/immich.git
synced 2026-04-27 15:07:45 +00:00
refactor(web): set birthdate (#25139)
This commit is contained in:
parent
a2ba36c16d
commit
6997ed83c4
6 changed files with 101 additions and 104 deletions
|
|
@ -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}
|
||||||
|
|
|
||||||
|
|
@ -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];
|
||||||
|
|
|
||||||
|
|
@ -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>
|
|
||||||
|
|
|
||||||
31
web/src/lib/services/person.service.ts
Normal file
31
web/src/lib/services/person.service.ts
Normal 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'));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
@ -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)}
|
||||||
|
|
|
||||||
|
|
@ -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}
|
||||||
|
|
|
||||||
Loading…
Reference in a new issue