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

fix(web): single row of items (#11729)

* fix(web): single row of items

* remove filterBoxWidth

* slight size adjustment

* rewrite action as component
This commit is contained in:
Michel Heusschen 2024-08-13 14:20:08 +02:00 committed by GitHub
parent e384692025
commit 5acdc958b6
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
4 changed files with 68 additions and 49 deletions

View file

@ -77,8 +77,6 @@
: MediaType.All, : MediaType.All,
}; };
let filterBoxWidth = 0;
const resetForm = () => { const resetForm = () => {
filter = { filter = {
personIds: new Set(), personIds: new Set(),
@ -120,7 +118,6 @@
</script> </script>
<div <div
bind:clientWidth={filterBoxWidth}
transition:fly={{ y: 25, duration: 250 }} transition:fly={{ y: 25, duration: 250 }}
class="absolute w-full rounded-b-3xl border-2 border-t-0 border-gray-200 bg-white shadow-2xl dark:border-gray-700 dark:bg-immich-dark-gray dark:text-gray-300" class="absolute w-full rounded-b-3xl border-2 border-t-0 border-gray-200 bg-white shadow-2xl dark:border-gray-700 dark:bg-immich-dark-gray dark:text-gray-300"
> >
@ -132,7 +129,7 @@
> >
<div class="px-4 sm:px-6 py-4 space-y-10 max-h-[calc(100dvh-12rem)] overflow-y-auto immich-scrollbar" tabindex="-1"> <div class="px-4 sm:px-6 py-4 space-y-10 max-h-[calc(100dvh-12rem)] overflow-y-auto immich-scrollbar" tabindex="-1">
<!-- PEOPLE --> <!-- PEOPLE -->
<SearchPeopleSection width={filterBoxWidth} bind:selectedPeople={filter.personIds} /> <SearchPeopleSection bind:selectedPeople={filter.personIds} />
<!-- TEXT --> <!-- TEXT -->
<SearchTextSection bind:filename={filter.filename} bind:context={filter.context} /> <SearchTextSection bind:filename={filter.filename} bind:context={filter.context} />

View file

@ -8,14 +8,14 @@
import { mdiClose, mdiArrowRight } from '@mdi/js'; import { mdiClose, mdiArrowRight } from '@mdi/js';
import { handleError } from '$lib/utils/handle-error'; import { handleError } from '$lib/utils/handle-error';
import { t } from 'svelte-i18n'; import { t } from 'svelte-i18n';
import SingleGridRow from '$lib/components/shared-components/single-grid-row.svelte';
export let width: number;
export let selectedPeople: Set<string>; export let selectedPeople: Set<string>;
let peoplePromise = getPeople(); let peoplePromise = getPeople();
let showAllPeople = false; let showAllPeople = false;
let name = ''; let name = '';
$: numberOfPeople = (width - 80) / 85; let numberOfPeople = 1;
function orderBySelectedPeopleFirst(people: PersonResponseDto[]) { function orderBySelectedPeopleFirst(people: PersonResponseDto[]) {
return [ return [
@ -60,11 +60,14 @@
<SearchBar bind:name placeholder={$t('filter_people')} showLoadingSpinner={false} /> <SearchBar bind:name placeholder={$t('filter_people')} showLoadingSpinner={false} />
</div> </div>
<div class="flex -mx-1 max-h-64 gap-1 mt-2 flex-wrap overflow-y-auto immich-scrollbar"> <SingleGridRow
class="grid grid-cols-[repeat(auto-fill,minmax(5rem,1fr))] -mx-1 gap-1 mt-2 overflow-y-auto immich-scrollbar"
bind:itemCount={numberOfPeople}
>
{#each peopleList as person (person.id)} {#each peopleList as person (person.id)}
<button <button
type="button" type="button"
class="flex flex-col items-center w-20 rounded-3xl border-2 hover:bg-immich-gray dark:hover:bg-immich-dark-primary/20 p-2 transition-all {selectedPeople.has( class="flex flex-col items-center rounded-3xl border-2 hover:bg-immich-gray dark:hover:bg-immich-dark-primary/20 p-2 transition-all {selectedPeople.has(
person.id, person.id,
) )
? 'dark:border-slate-500 border-slate-400 bg-slate-200 dark:bg-slate-800 dark:text-white' ? 'dark:border-slate-500 border-slate-400 bg-slate-200 dark:bg-slate-800 dark:text-white'
@ -75,7 +78,7 @@
<p class="mt-2 line-clamp-2 text-sm font-medium dark:text-white">{person.name}</p> <p class="mt-2 line-clamp-2 text-sm font-medium dark:text-white">{person.name}</p>
</button> </button>
{/each} {/each}
</div> </SingleGridRow>
{#if showAllPeople || people.length > peopleList.length} {#if showAllPeople || people.length > peopleList.length}
<div class="flex justify-center mt-2"> <div class="flex justify-center mt-2">

View file

@ -0,0 +1,38 @@
<script lang="ts">
let className = '';
export { className as class };
export let itemCount = 1;
let container: HTMLElement | undefined;
let contentRect: DOMRectReadOnly | undefined;
const getGridGap = (element: Element) => {
const style = getComputedStyle(element);
return {
columnGap: parsePixels(style.columnGap),
};
};
const parsePixels = (style: string) => Number.parseInt(style, 10) || 0;
const getItemCount = (container: HTMLElement, containerWidth: number) => {
if (!container.firstElementChild) {
return 1;
}
const childContentRect = container.firstElementChild.getBoundingClientRect();
const childWidth = Math.floor(childContentRect.width || Infinity);
const { columnGap } = getGridGap(container);
return Math.floor((containerWidth + columnGap) / (childWidth + columnGap)) || 1;
};
$: if (container && contentRect) {
itemCount = getItemCount(container, contentRect.width);
}
</script>
<div class={className} bind:this={container} bind:contentRect>
<slot {itemCount} />
</div>

View file

@ -10,6 +10,7 @@
import EmptyPlaceholder from '$lib/components/shared-components/empty-placeholder.svelte'; import EmptyPlaceholder from '$lib/components/shared-components/empty-placeholder.svelte';
import { onMount } from 'svelte'; import { onMount } from 'svelte';
import { websocketEvents } from '$lib/stores/websocket'; import { websocketEvents } from '$lib/stores/websocket';
import SingleGridRow from '$lib/components/shared-components/single-grid-row.svelte';
export let data: PageData; export let data: PageData;
@ -19,25 +20,14 @@
OBJECTS = 'smartInfo.objects', OBJECTS = 'smartInfo.objects',
} }
let MAX_PEOPLE_ITEMS: number;
let MAX_PLACE_ITEMS: number;
let innerWidth: number;
let screenSize: number;
const getFieldItems = (items: SearchExploreResponseDto[], field: Field) => { const getFieldItems = (items: SearchExploreResponseDto[], field: Field) => {
const targetField = items.find((item) => item.fieldName === field); const targetField = items.find((item) => item.fieldName === field);
return targetField?.items || []; return targetField?.items || [];
}; };
$: places = getFieldItems(data.items, Field.CITY).slice(0, MAX_PLACE_ITEMS); $: places = getFieldItems(data.items, Field.CITY);
$: people = data.response.people.slice(0, MAX_PEOPLE_ITEMS); $: people = data.response.people;
$: hasPeople = data.response.total > 0; $: hasPeople = data.response.total > 0;
$: {
if (innerWidth && screenSize) {
// Set the number of faces according to the screen size and the div size
MAX_PEOPLE_ITEMS = screenSize < 768 ? Math.floor(innerWidth / 96) : Math.floor(innerWidth / 120);
MAX_PLACE_ITEMS = screenSize < 768 ? Math.floor(innerWidth / 150) : Math.floor(innerWidth / 172);
}
}
onMount(() => { onMount(() => {
return websocketEvents.on('on_person_thumbnail', (personId: string) => { return websocketEvents.on('on_person_thumbnail', (personId: string) => {
@ -52,8 +42,6 @@
}); });
</script> </script>
<svelte:window bind:innerWidth={screenSize} />
<UserPageLayout title={data.meta.title}> <UserPageLayout title={data.meta.title}>
{#if hasPeople} {#if hasPeople}
<div class="mb-6 mt-2"> <div class="mb-6 mt-2">
@ -65,25 +53,17 @@
draggable="false">{$t('view_all')}</a draggable="false">{$t('view_all')}</a
> >
</div> </div>
<div <SingleGridRow
class="flex flex-row {MAX_PEOPLE_ITEMS < 5 ? 'justify-center' : ''} flex-wrap gap-4" class="grid md:grid-cols-[repeat(auto-fill,minmax(7rem,1fr))] grid-cols-[repeat(auto-fill,minmax(5rem,1fr))] gap-x-4"
bind:offsetWidth={innerWidth} let:itemCount
> >
{#if MAX_PEOPLE_ITEMS} {#each people.slice(0, itemCount) as person (person.id)}
{#each people as person (person.id)} <a href="{AppRoute.PEOPLE}/{person.id}" class="text-center">
<a href="{AppRoute.PEOPLE}/{person.id}" class="w-20 md:w-24 text-center"> <ImageThumbnail circle shadow url={getPeopleThumbnailUrl(person)} altText={person.name} widthStyle="100%" />
<ImageThumbnail <p class="mt-2 text-ellipsis text-sm font-medium dark:text-white">{person.name}</p>
circle </a>
shadow {/each}
url={getPeopleThumbnailUrl(person)} </SingleGridRow>
altText={person.name}
widthStyle="100%"
/>
<p class="mt-2 text-ellipsis text-sm font-medium dark:text-white">{person.name}</p>
</a>
{/each}
{/if}
</div>
</div> </div>
{/if} {/if}
@ -97,16 +77,17 @@
draggable="false">{$t('view_all')}</a draggable="false">{$t('view_all')}</a
> >
</div> </div>
<div class="flex flex-row flex-wrap gap-4"> <SingleGridRow
{#each places as item (item.data.id)} class="grid md:grid-cols-[repeat(auto-fill,minmax(9rem,1fr))] grid-cols-[repeat(auto-fill,minmax(7rem,1fr))] gap-x-4"
let:itemCount
>
{#each places.slice(0, itemCount) as item (item.data.id)}
<a class="relative" href="{AppRoute.SEARCH}?{getMetadataSearchQuery({ city: item.value })}" draggable="false"> <a class="relative" href="{AppRoute.SEARCH}?{getMetadataSearchQuery({ city: item.value })}" draggable="false">
<div <div class="flex justify-center overflow-hidden rounded-xl brightness-75 filter">
class="flex w-[calc((100vw-(72px+5rem))/2)] max-w-[156px] justify-center overflow-hidden rounded-xl brightness-75 filter"
>
<img <img
src={getAssetThumbnailUrl({ id: item.data.id, size: AssetMediaSize.Thumbnail })} src={getAssetThumbnailUrl({ id: item.data.id, size: AssetMediaSize.Thumbnail })}
alt={item.value} alt={item.value}
class="object-cover w-[156px] h-[156px]" class="object-cover aspect-square w-full"
/> />
</div> </div>
<span <span
@ -116,7 +97,7 @@
</span> </span>
</a> </a>
{/each} {/each}
</div> </SingleGridRow>
</div> </div>
{/if} {/if}