mirror of
https://github.com/immich-app/immich
synced 2025-11-14 17:36:12 +00:00
refactor(web): search people (#9082)
* refactor: search people * fix: test * fix: timeout * fix: callbacks * fix: simplify * remove unused var * refactor: rename file * fix: focus when deleting last character --------- Co-authored-by: Alex Tran <alex.tran1502@gmail.com> Co-authored-by: Jason Rasmussen <jrasm91@gmail.com>
This commit is contained in:
parent
72ce81f0c2
commit
5722c830ff
12 changed files with 207 additions and 267 deletions
100
web/src/lib/components/faces-page/people-search.svelte
Normal file
100
web/src/lib/components/faces-page/people-search.svelte
Normal file
|
|
@ -0,0 +1,100 @@
|
|||
<script lang="ts">
|
||||
import SearchBar from '$lib/components/elements/search-bar.svelte';
|
||||
import { maximumLengthSearchPeople, timeBeforeShowLoadingSpinner } from '$lib/constants';
|
||||
import { handleError } from '$lib/utils/handle-error';
|
||||
import { searchNameLocal } from '$lib/utils/person';
|
||||
import { searchPerson, type PersonResponseDto } from '@immich/sdk';
|
||||
|
||||
export let searchName: string;
|
||||
export let searchedPeopleLocal: PersonResponseDto[];
|
||||
export let type: 'searchBar' | 'input';
|
||||
export let numberPeopleToSearch: number = maximumLengthSearchPeople;
|
||||
export let inputClass: string = 'w-full gap-2 bg-immich-bg dark:bg-immich-dark-bg';
|
||||
export let showLoadingSpinner: boolean = false;
|
||||
export let placeholder: string = 'Name or nickname';
|
||||
export let onReset = () => {};
|
||||
export let onSearch = () => {};
|
||||
|
||||
let searchedPeople: PersonResponseDto[] = [];
|
||||
let searchWord: string;
|
||||
let abortController: AbortController | null = null;
|
||||
let timeout: NodeJS.Timeout | null = null;
|
||||
|
||||
const search = () => {
|
||||
searchedPeopleLocal = searchNameLocal(searchName, searchedPeople, numberPeopleToSearch);
|
||||
};
|
||||
|
||||
const reset = () => {
|
||||
searchedPeopleLocal = [];
|
||||
cancelPreviousRequest();
|
||||
onReset();
|
||||
};
|
||||
|
||||
const cancelPreviousRequest = () => {
|
||||
if (abortController) {
|
||||
abortController.abort();
|
||||
abortController = null;
|
||||
}
|
||||
if (timeout) {
|
||||
clearTimeout(timeout);
|
||||
timeout = null;
|
||||
}
|
||||
};
|
||||
|
||||
export let handleSearch = async (force?: boolean, name?: string) => {
|
||||
searchName = name ?? searchName;
|
||||
onSearch();
|
||||
if (searchName === '') {
|
||||
reset();
|
||||
return;
|
||||
}
|
||||
if (!force && searchedPeople.length < maximumLengthSearchPeople && searchName.startsWith(searchWord)) {
|
||||
search();
|
||||
return;
|
||||
}
|
||||
cancelPreviousRequest();
|
||||
abortController = new AbortController();
|
||||
timeout = setTimeout(() => (showLoadingSpinner = true), timeBeforeShowLoadingSpinner);
|
||||
try {
|
||||
const data = await searchPerson({ name: searchName }, { signal: abortController?.signal });
|
||||
searchedPeople = data;
|
||||
searchWord = searchName;
|
||||
} catch (error) {
|
||||
handleError(error, "Can't search people");
|
||||
} finally {
|
||||
clearTimeout(timeout);
|
||||
timeout = null;
|
||||
abortController = null;
|
||||
showLoadingSpinner = false;
|
||||
search();
|
||||
}
|
||||
};
|
||||
|
||||
const initInput = (element: HTMLInputElement) => {
|
||||
element.focus();
|
||||
};
|
||||
|
||||
const handleReset = () => {
|
||||
reset();
|
||||
onReset();
|
||||
};
|
||||
</script>
|
||||
|
||||
{#if type === 'searchBar'}
|
||||
<SearchBar
|
||||
bind:name={searchName}
|
||||
{showLoadingSpinner}
|
||||
{placeholder}
|
||||
on:reset={handleReset}
|
||||
on:search={({ detail }) => handleSearch(detail.force ?? false)}
|
||||
/>
|
||||
{:else}
|
||||
<input
|
||||
class={inputClass}
|
||||
type="text"
|
||||
{placeholder}
|
||||
bind:value={searchName}
|
||||
on:input={() => handleSearch(false)}
|
||||
use:initInput
|
||||
/>
|
||||
{/if}
|
||||
Loading…
Add table
Add a link
Reference in a new issue