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

feat(web): suggest people when typing a name (#4126)

* feat(web): suggest people when entering a name

* fix: border size from 2 to 1 pixel

* pr feedback

* fix: web unit test

* pr feedback

---------

Co-authored-by: Alex <alex.tran1502@gmail.com>
This commit is contained in:
martin 2023-09-23 06:58:51 +02:00 committed by GitHub
parent 9a7e48eaa6
commit fc64be6603
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 117 additions and 48 deletions

View file

@ -3,10 +3,10 @@
import { createEventDispatcher } from 'svelte'; import { createEventDispatcher } from 'svelte';
import ImageThumbnail from '../assets/thumbnail/image-thumbnail.svelte'; import ImageThumbnail from '../assets/thumbnail/image-thumbnail.svelte';
import Button from '../elements/buttons/button.svelte'; import Button from '../elements/buttons/button.svelte';
import { clickOutside } from '$lib/utils/click-outside';
export let person: PersonResponseDto; export let person: PersonResponseDto;
let name = person.name; export let name: string;
export let suggestedPeople = false;
const dispatch = createEventDispatcher<{ const dispatch = createEventDispatcher<{
change: string; change: string;
@ -15,9 +15,9 @@
</script> </script>
<div <div
class="flex max-w-lg place-items-center rounded-lg border bg-gray-100 p-2 dark:border-transparent dark:bg-gray-700" class="flex w-full place-items-center {suggestedPeople
use:clickOutside ? 'rounded-t-lg border-b dark:border-immich-dark-gray'
on:outclick={() => dispatch('cancel')} : 'rounded-lg'} bg-gray-100 p-2 dark:bg-gray-700"
> >
<ImageThumbnail <ImageThumbnail
circle circle

View file

@ -17,16 +17,7 @@
export let personMerge1: PersonResponseDto; export let personMerge1: PersonResponseDto;
export let personMerge2: PersonResponseDto; export let personMerge2: PersonResponseDto;
export let people: PersonResponseDto[]; export let potentialMergePeople: PersonResponseDto[];
let potentialMergePeople: PersonResponseDto[] = people
.filter(
(person: PersonResponseDto) =>
personMerge2.name.toLowerCase() === person.name.toLowerCase() &&
person.id !== personMerge2.id &&
person.id !== personMerge1.id &&
!person.isHidden,
)
.slice(0, 3);
let choosePersonToMerge = false; let choosePersonToMerge = false;
@ -48,7 +39,9 @@
<h1 class="truncate px-4 py-4 font-medium text-immich-primary dark:text-immich-dark-primary"> <h1 class="truncate px-4 py-4 font-medium text-immich-primary dark:text-immich-dark-primary">
Merge faces - {title} Merge faces - {title}
</h1> </h1>
<CircleIconButton logo={Close} on:click={() => dispatch('close')} /> <div class="p-2">
<CircleIconButton logo={Close} on:click={() => dispatch('close')} />
</div>
</div> </div>
<div class="flex items-center justify-center px-2 py-4 md:h-36 md:px-4 md:py-4"> <div class="flex items-center justify-center px-2 py-4 md:h-36 md:px-4 md:py-4">

View file

@ -42,6 +42,7 @@
let personName = ''; let personName = '';
let personMerge1: PersonResponseDto; let personMerge1: PersonResponseDto;
let personMerge2: PersonResponseDto; let personMerge2: PersonResponseDto;
let potentialMergePeople: PersonResponseDto[] = [];
let edittingPerson: PersonResponseDto | null = null; let edittingPerson: PersonResponseDto | null = null;
people.forEach((person: PersonResponseDto) => { people.forEach((person: PersonResponseDto) => {
@ -248,6 +249,7 @@
}; };
const submitNameChange = async () => { const submitNameChange = async () => {
potentialMergePeople = [];
showChangeNameModal = false; showChangeNameModal = false;
if (!edittingPerson || personName === edittingPerson.name) { if (!edittingPerson || personName === edittingPerson.name) {
return; return;
@ -264,6 +266,15 @@
if (existingPerson) { if (existingPerson) {
personMerge2 = existingPerson; personMerge2 = existingPerson;
showMergeModal = true; showMergeModal = true;
potentialMergePeople = people
.filter(
(person: PersonResponseDto) =>
personMerge2.name.toLowerCase() === person.name.toLowerCase() &&
person.id !== personMerge2.id &&
person.id !== personMerge1.id &&
!person.isHidden,
)
.slice(0, 3);
return; return;
} }
changeName(); changeName();
@ -332,7 +343,7 @@
<MergeSuggestionModal <MergeSuggestionModal
{personMerge1} {personMerge1}
{personMerge2} {personMerge2}
{people} {potentialMergePeople}
on:close={() => (showMergeModal = false)} on:close={() => (showMergeModal = false)}
on:reject={() => changeName()} on:reject={() => changeName()}
on:confirm={(event) => handleMergeSameFace(event.detail)} on:confirm={(event) => handleMergeSameFace(event.detail)}

View file

@ -32,6 +32,7 @@
import DotsVertical from 'svelte-material-icons/DotsVertical.svelte'; import DotsVertical from 'svelte-material-icons/DotsVertical.svelte';
import Plus from 'svelte-material-icons/Plus.svelte'; import Plus from 'svelte-material-icons/Plus.svelte';
import type { PageData } from './$types'; import type { PageData } from './$types';
import { clickOutside } from '$lib/utils/click-outside';
export let data: PageData; export let data: PageData;
@ -58,12 +59,27 @@
let people = data.people.people; let people = data.people.people;
let personMerge1: PersonResponseDto; let personMerge1: PersonResponseDto;
let personMerge2: PersonResponseDto; let personMerge2: PersonResponseDto;
let potentialMergePeople: PersonResponseDto[] = [];
let personName = ''; let personName = '';
let name: string = data.person.name;
let suggestedPeople: PersonResponseDto[] = [];
$: isAllArchive = Array.from($selectedAssets).every((asset) => asset.isArchived); $: isAllArchive = Array.from($selectedAssets).every((asset) => asset.isArchived);
$: isAllFavorite = Array.from($selectedAssets).every((asset) => asset.isFavorite); $: isAllFavorite = Array.from($selectedAssets).every((asset) => asset.isFavorite);
$: {
suggestedPeople = !name
? []
: people
.filter(
(person: PersonResponseDto) =>
person.name.toLowerCase().startsWith(name.toLowerCase()) && person.id !== data.person.id,
)
.slice(0, 5);
}
onMount(() => { onMount(() => {
const action = $page.url.searchParams.get('action'); const action = $page.url.searchParams.get('action');
if (action == 'merge') { if (action == 'merge') {
@ -147,6 +163,14 @@
} }
}; };
const handleSuggestPeople = (person: PersonResponseDto) => {
isEditingName = false;
potentialMergePeople = [];
personMerge1 = data.person;
personMerge2 = person;
viewMode = ViewMode.SUGGEST_MERGE;
};
const changeName = async () => { const changeName = async () => {
viewMode = ViewMode.VIEW_ASSETS; viewMode = ViewMode.VIEW_ASSETS;
data.person.name = personName; data.person.name = personName;
@ -183,6 +207,7 @@
}; };
const handleNameChange = async (name: string) => { const handleNameChange = async (name: string) => {
potentialMergePeople = [];
personName = name; personName = name;
if (data.person.name === personName) { if (data.person.name === personName) {
@ -196,6 +221,15 @@
if (existingPerson) { if (existingPerson) {
personMerge2 = existingPerson; personMerge2 = existingPerson;
personMerge1 = data.person; personMerge1 = data.person;
potentialMergePeople = people
.filter(
(person: PersonResponseDto) =>
personMerge2.name.toLowerCase() === person.name.toLowerCase() &&
person.id !== personMerge2.id &&
person.id !== personMerge1.id &&
!person.isHidden,
)
.slice(0, 3);
viewMode = ViewMode.SUGGEST_MERGE; viewMode = ViewMode.SUGGEST_MERGE;
return; return;
} }
@ -238,7 +272,7 @@
<MergeSuggestionModal <MergeSuggestionModal
{personMerge1} {personMerge1}
{personMerge2} {personMerge2}
{people} {potentialMergePeople}
on:close={() => (viewMode = ViewMode.VIEW_ASSETS)} on:close={() => (viewMode = ViewMode.VIEW_ASSETS)}
on:reject={() => changeName()} on:reject={() => changeName()}
on:confirm={(event) => handleMergeSameFace(event.detail)} on:confirm={(event) => handleMergeSameFace(event.detail)}
@ -306,39 +340,70 @@
> >
{#if viewMode === ViewMode.VIEW_ASSETS || viewMode === ViewMode.SUGGEST_MERGE || viewMode === ViewMode.BIRTH_DATE} {#if viewMode === ViewMode.VIEW_ASSETS || viewMode === ViewMode.SUGGEST_MERGE || viewMode === ViewMode.BIRTH_DATE}
<!-- Face information block --> <!-- Face information block -->
<section class="flex place-items-center p-4 sm:px-6"> <div
{#if isEditingName} role="button"
<EditNameInput class="relative w-fit p-4 sm:px-6"
person={data.person} use:clickOutside
on:change={(event) => handleNameChange(event.detail)} on:outclick={() => handleCancelEditName()}
on:cancel={() => handleCancelEditName()} >
/> <section class="flex w-96 place-items-center border-black">
{:else} {#if isEditingName}
<button on:click={() => (viewMode = ViewMode.VIEW_ASSETS)}> <EditNameInput
<ImageThumbnail person={data.person}
circle suggestedPeople={suggestedPeople.length > 0}
shadow bind:name
url={api.getPeopleThumbnailUrl(data.person.id)} on:change={(event) => handleNameChange(event.detail)}
altText={data.person.name}
widthStyle="3.375rem"
heightStyle="3.375rem"
/> />
</button> {:else}
<button on:click={() => (viewMode = ViewMode.VIEW_ASSETS)}>
<ImageThumbnail
circle
shadow
url={api.getPeopleThumbnailUrl(data.person.id)}
altText={data.person.name}
widthStyle="3.375rem"
heightStyle="3.375rem"
/>
</button>
<button <button
title="Edit name" title="Edit name"
class="px-4 text-immich-primary dark:text-immich-dark-primary" class="px-4 text-immich-primary dark:text-immich-dark-primary"
on:click={() => (isEditingName = true)} on:click={() => (isEditingName = true)}
> >
{#if data.person.name} {#if data.person.name}
<p class="py-2 font-medium">{data.person.name}</p> <p class="py-2 font-medium">{data.person.name}</p>
{:else} {:else}
<p class="w-fit font-medium">Add a name</p> <p class="w-fit font-medium">Add a name</p>
<p class="text-sm text-gray-500 dark:text-immich-gray">Find them fast by name with search</p> <p class="text-sm text-gray-500 dark:text-immich-gray">Find them fast by name with search</p>
{/if} {/if}
</button> </button>
{/if}
</section>
{#if isEditingName}
<div class="absolute z-[999] w-96">
{#each suggestedPeople as person, index (person.id)}
<div
class="flex {index === suggestedPeople.length - 1
? 'rounded-b-lg'
: 'border-b dark:border-immich-dark-gray'} place-items-center bg-gray-100 p-2 dark:bg-gray-700"
>
<button class="flex w-full place-items-center" on:click={() => handleSuggestPeople(person)}>
<ImageThumbnail
circle
shadow
url={api.getPeopleThumbnailUrl(person.id)}
altText={person.name}
widthStyle="2rem"
heightStyle="2rem"
/>
<p class="ml-4 text-gray-700 dark:text-gray-100">{person.name}</p>
</button>
</div>
{/each}
</div>
{/if} {/if}
</section> </div>
{/if} {/if}
</AssetGrid> </AssetGrid>
{/key} {/key}