1
0
Fork 0
mirror of https://github.com/immich-app/immich.git synced 2025-01-19 18:26:46 +01:00

feat(web, server): Implement justified layout for AssetGrid (#2666)

* Implement justified layout for timeline

* Add withoutThumbs field to GetTimelineLayotDto

* Back to rough estimation of initial buckets height

* Remove getTimelineLayout endpoint

* Estimate rough viewport height better

* Fix shift/jump issues while scrolling up

---------

Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
This commit is contained in:
Sergey Kondrikov 2023-06-08 18:22:45 +03:00 committed by GitHub
parent 8ebac41318
commit 5764bf16f3
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
11 changed files with 172 additions and 45 deletions

View file

@ -104,6 +104,7 @@ export class AssetRepository implements IAssetRepository {
// Get asset entity from a list of time buckets // Get asset entity from a list of time buckets
let builder = this.assetRepository let builder = this.assetRepository
.createQueryBuilder('asset') .createQueryBuilder('asset')
.leftJoinAndSelect('asset.exifInfo', 'exifInfo')
.where('asset.ownerId = :userId', { userId: userId }) .where('asset.ownerId = :userId', { userId: userId })
.andWhere(`date_trunc('month', "fileCreatedAt") IN (:...buckets)`, { .andWhere(`date_trunc('month', "fileCreatedAt") IN (:...buckets)`, {
buckets: [...dto.timeBucket], buckets: [...dto.timeBucket],

11
web/package-lock.json generated
View file

@ -12,6 +12,7 @@
"axios": "^0.27.2", "axios": "^0.27.2",
"copy-image-clipboard": "^2.1.2", "copy-image-clipboard": "^2.1.2",
"handlebars": "^4.7.7", "handlebars": "^4.7.7",
"justified-layout": "^4.1.0",
"leaflet": "^1.9.3", "leaflet": "^1.9.3",
"leaflet.markercluster": "^1.5.3", "leaflet.markercluster": "^1.5.3",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",
@ -9076,6 +9077,11 @@
"node": ">=6" "node": ">=6"
} }
}, },
"node_modules/justified-layout": {
"version": "4.1.0",
"resolved": "https://registry.npmjs.org/justified-layout/-/justified-layout-4.1.0.tgz",
"integrity": "sha512-M5FimNMXgiOYerVRGsXZ2YK9YNCaTtwtYp7Hb2308U1Q9TXXHx5G0p08mcVR5O53qf8bWY4NJcPBxE6zuayXSg=="
},
"node_modules/kind-of": { "node_modules/kind-of": {
"version": "6.0.3", "version": "6.0.3",
"resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",
@ -18186,6 +18192,11 @@
"integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==", "integrity": "sha512-XmOWe7eyHYH14cLdVPoyg+GOH3rYX++KpzrylJwSW98t3Nk+U8XOl8FWKOgwtzdb8lXGf6zYwDUzeHMWfxasyg==",
"dev": true "dev": true
}, },
"justified-layout": {
"version": "4.1.0",
"resolved": "https://registry.npmjs.org/justified-layout/-/justified-layout-4.1.0.tgz",
"integrity": "sha512-M5FimNMXgiOYerVRGsXZ2YK9YNCaTtwtYp7Hb2308U1Q9TXXHx5G0p08mcVR5O53qf8bWY4NJcPBxE6zuayXSg=="
},
"kind-of": { "kind-of": {
"version": "6.0.3", "version": "6.0.3",
"resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz", "resolved": "https://registry.npmjs.org/kind-of/-/kind-of-6.0.3.tgz",

View file

@ -62,6 +62,7 @@
"axios": "^0.27.2", "axios": "^0.27.2",
"copy-image-clipboard": "^2.1.2", "copy-image-clipboard": "^2.1.2",
"handlebars": "^4.7.7", "handlebars": "^4.7.7",
"justified-layout": "^4.1.0",
"leaflet": "^1.9.3", "leaflet": "^1.9.3",
"leaflet.markercluster": "^1.5.3", "leaflet.markercluster": "^1.5.3",
"lodash-es": "^4.17.21", "lodash-es": "^4.17.21",

View file

@ -1,4 +1,5 @@
<script lang="ts"> <script lang="ts">
import { BucketPosition } from '$lib/models/asset-grid-state';
import { onMount } from 'svelte'; import { onMount } from 'svelte';
import { createEventDispatcher } from 'svelte'; import { createEventDispatcher } from 'svelte';
@ -28,7 +29,17 @@
} }
if (intersecting) { if (intersecting) {
dispatch('intersected', container); let position: BucketPosition = BucketPosition.Visible;
if (entries[0].boundingClientRect.top + 50 > entries[0].intersectionRect.bottom) {
position = BucketPosition.Below;
} else if (entries[0].boundingClientRect.bottom < 0) {
position = BucketPosition.Above;
}
dispatch('intersected', {
container,
position
});
} }
}, },
{ {

View file

@ -9,17 +9,20 @@
import { assetStore } from '$lib/stores/assets.store'; import { assetStore } from '$lib/stores/assets.store';
import { locale } from '$lib/stores/preferences.store'; import { locale } from '$lib/stores/preferences.store';
import type { AssetResponseDto } from '@api'; import type { AssetResponseDto } from '@api';
import justifiedLayout from 'justified-layout';
import lodash from 'lodash-es'; import lodash from 'lodash-es';
import CheckCircle from 'svelte-material-icons/CheckCircle.svelte'; import CheckCircle from 'svelte-material-icons/CheckCircle.svelte';
import CircleOutline from 'svelte-material-icons/CircleOutline.svelte'; import CircleOutline from 'svelte-material-icons/CircleOutline.svelte';
import { flip } from 'svelte/animate';
import { fly } from 'svelte/transition'; import { fly } from 'svelte/transition';
import { getAssetRatio } from '$lib/utils/asset-utils';
import Thumbnail from '../assets/thumbnail/thumbnail.svelte'; import Thumbnail from '../assets/thumbnail/thumbnail.svelte';
import { createEventDispatcher } from 'svelte';
export let assets: AssetResponseDto[]; export let assets: AssetResponseDto[];
export let bucketDate: string; export let bucketDate: string;
export let bucketHeight: number; export let bucketHeight: number;
export let isAlbumSelectionMode = false; export let isAlbumSelectionMode = false;
export let viewportWidth: number;
const groupDateFormat: Intl.DateTimeFormatOptions = { const groupDateFormat: Intl.DateTimeFormatOptions = {
weekday: 'short', weekday: 'short',
@ -28,21 +31,66 @@
year: 'numeric' year: 'numeric'
}; };
const dispatch = createEventDispatcher();
let isMouseOverGroup = false; let isMouseOverGroup = false;
let actualBucketHeight: number; let actualBucketHeight: number;
let hoveredDateGroup = ''; let hoveredDateGroup = '';
interface LayoutBox {
top: number;
left: number;
width: number;
}
$: assetsGroupByDate = lodash $: assetsGroupByDate = lodash
.chain(assets) .chain(assets)
.groupBy((a) => new Date(a.fileCreatedAt).toLocaleDateString($locale, groupDateFormat)) .groupBy((a) => new Date(a.fileCreatedAt).toLocaleDateString($locale, groupDateFormat))
.sortBy((group) => assets.indexOf(group[0])) .sortBy((group) => assets.indexOf(group[0]))
.value(); .value();
$: geometry = (() => {
const geometry = [];
for (let group of assetsGroupByDate) {
geometry.push(
justifiedLayout(group.map(getAssetRatio), {
boxSpacing: 2,
containerWidth: Math.floor(viewportWidth),
containerPadding: 0,
targetRowHeightTolerance: 0.15,
targetRowHeight: 235
})
);
}
return geometry;
})();
$: { $: {
if (actualBucketHeight && actualBucketHeight != 0 && actualBucketHeight != bucketHeight) { if (actualBucketHeight && actualBucketHeight != 0 && actualBucketHeight != bucketHeight) {
assetStore.updateBucketHeight(bucketDate, actualBucketHeight); const heightDelta = assetStore.updateBucketHeight(bucketDate, actualBucketHeight);
if (heightDelta !== 0) {
scrollTimeline(heightDelta);
}
} }
} }
function scrollTimeline(heightDelta: number) {
dispatch('shift', {
heightDelta
});
}
const calculateWidth = (boxes: LayoutBox[]): number => {
let width = 0;
for (const box of boxes) {
if (box.top < 100) {
width = box.left + box.width;
}
}
return width;
};
const assetClickHandler = ( const assetClickHandler = (
asset: AssetResponseDto, asset: AssetResponseDto,
assetsInDateGroup: AssetResponseDto[], assetsInDateGroup: AssetResponseDto[],
@ -112,8 +160,9 @@
<section <section
id="asset-group-by-date" id="asset-group-by-date"
class="flex flex-wrap gap-12 mt-5" class="flex flex-wrap gap-x-12"
bind:clientHeight={actualBucketHeight} bind:clientHeight={actualBucketHeight}
bind:clientWidth={viewportWidth}
> >
{#each assetsGroupByDate as assetsInDateGroup, groupIndex (assetsInDateGroup[0].id)} {#each assetsGroupByDate as assetsInDateGroup, groupIndex (assetsInDateGroup[0].id)}
{@const dateGroupTitle = new Date(assetsInDateGroup[0].fileCreatedAt).toLocaleDateString( {@const dateGroupTitle = new Date(assetsInDateGroup[0].fileCreatedAt).toLocaleDateString(
@ -123,8 +172,7 @@
<!-- Asset Group By Date --> <!-- Asset Group By Date -->
<div <div
animate:flip={{ duration: 300 }} class="flex flex-col mt-5"
class="flex flex-col"
on:mouseenter={() => { on:mouseenter={() => {
isMouseOverGroup = true; isMouseOverGroup = true;
assetMouseEventHandler(dateGroupTitle); assetMouseEventHandler(dateGroupTitle);
@ -156,9 +204,18 @@
</p> </p>
<!-- Image grid --> <!-- Image grid -->
<div class="flex flex-wrap gap-[2px]"> <div
{#each assetsInDateGroup as asset (asset.id)} class="relative"
<div animate:flip={{ duration: 300 }}> style="height: {geometry[groupIndex].containerHeight}px;width: {calculateWidth(
geometry[groupIndex].boxes
)}px"
>
{#each assetsInDateGroup as asset, index (asset.id)}
{@const box = geometry[groupIndex].boxes[index]}
<div
class="absolute"
style="width: {box.width}px; height: {box.height}px; top: {box.top}px; left: {box.left}px"
>
<Thumbnail <Thumbnail
{asset} {asset}
{groupIndex} {groupIndex}
@ -168,6 +225,8 @@
selected={$selectedAssets.has(asset) || selected={$selectedAssets.has(asset) ||
$assetsInAlbumStoreState.findIndex((a) => a.id == asset.id) != -1} $assetsInAlbumStoreState.findIndex((a) => a.id == asset.id) != -1}
disabled={$assetsInAlbumStoreState.findIndex((a) => a.id == asset.id) != -1} disabled={$assetsInAlbumStoreState.findIndex((a) => a.id == asset.id) != -1}
thumbnailWidth={box.width}
thumbnailHeight={box.height}
/> />
</div> </div>
{/each} {/each}

View file

@ -16,6 +16,7 @@
OnScrollbarDragDetail OnScrollbarDragDetail
} from '../shared-components/scrollbar/scrollbar.svelte'; } from '../shared-components/scrollbar/scrollbar.svelte';
import AssetDateGroup from './asset-date-group.svelte'; import AssetDateGroup from './asset-date-group.svelte';
import { BucketPosition } from '$lib/models/asset-grid-state';
export let user: UserResponseDto | undefined = undefined; export let user: UserResponseDto | undefined = undefined;
export let isAlbumSelectionMode = false; export let isAlbumSelectionMode = false;
@ -33,6 +34,7 @@
withoutThumbs: true withoutThumbs: true
} }
}); });
bucketInfo = assetCountByTimebucket; bucketInfo = assetCountByTimebucket;
assetStore.setInitialState(viewportHeight, viewportWidth, assetCountByTimebucket, user?.id); assetStore.setInitialState(viewportHeight, viewportWidth, assetCountByTimebucket, user?.id);
@ -51,7 +53,7 @@
}); });
bucketsToFetchInitially.forEach((bucketDate) => { bucketsToFetchInitially.forEach((bucketDate) => {
assetStore.getAssetsByBucket(bucketDate); assetStore.getAssetsByBucket(bucketDate, BucketPosition.Visible);
}); });
}); });
@ -60,15 +62,18 @@
}); });
function intersectedHandler(event: CustomEvent) { function intersectedHandler(event: CustomEvent) {
const el = event.detail as HTMLElement; const el = event.detail.container as HTMLElement;
const target = el.firstChild as HTMLElement; const target = el.firstChild as HTMLElement;
if (target) { if (target) {
const bucketDate = target.id.split('_')[1]; const bucketDate = target.id.split('_')[1];
assetStore.getAssetsByBucket(bucketDate); assetStore.getAssetsByBucket(bucketDate, event.detail.position);
} }
} }
function handleScrollTimeline(event: CustomEvent) {
assetGridElement.scrollBy(0, event.detail.heightDelta);
}
const navigateToPreviousAsset = () => { const navigateToPreviousAsset = () => {
assetInteractionStore.navigateAsset('previous'); assetInteractionStore.navigateAsset('previous');
}; };
@ -115,9 +120,10 @@
/> />
{/if} {/if}
<!-- Right margin MUST be equal to the width of immich-scrubbable-scrollbar -->
<section <section
id="asset-grid" id="asset-grid"
class="overflow-y-auto pl-4 scrollbar-hidden" class="overflow-y-auto ml-4 mb-4 mr-[60px] scrollbar-hidden"
bind:clientHeight={viewportHeight} bind:clientHeight={viewportHeight}
bind:clientWidth={viewportWidth} bind:clientWidth={viewportWidth}
bind:this={assetGridElement} bind:this={assetGridElement}
@ -143,9 +149,11 @@
{#if intersecting} {#if intersecting}
<AssetDateGroup <AssetDateGroup
{isAlbumSelectionMode} {isAlbumSelectionMode}
on:shift={handleScrollTimeline}
assets={bucket.assets} assets={bucket.assets}
bucketDate={bucket.bucketDate} bucketDate={bucket.bucketDate}
bucketHeight={bucket.bucketHeight} bucketHeight={bucket.bucketHeight}
{viewportWidth}
/> />
{/if} {/if}
</div> </div>

View file

@ -1,5 +1,12 @@
import type { AssetResponseDto } from '@api'; import type { AssetResponseDto } from '@api';
export enum BucketPosition {
Above = 'above',
Below = 'below',
Visible = 'visible',
Unknown = 'unknown'
}
export class AssetBucket { export class AssetBucket {
/** /**
* The DOM height of the bucket in pixel * The DOM height of the bucket in pixel
@ -9,6 +16,7 @@ export class AssetBucket {
bucketDate!: string; bucketDate!: string;
assets!: AssetResponseDto[]; assets!: AssetResponseDto[];
cancelToken!: AbortController; cancelToken!: AbortController;
position!: BucketPosition;
} }
export class AssetGridState { export class AssetGridState {

View file

@ -1,4 +1,4 @@
import { AssetGridState } from '$lib/models/asset-grid-state'; import { AssetGridState, BucketPosition } from '$lib/models/asset-grid-state';
import { api, AssetResponseDto } from '@api'; import { api, AssetResponseDto } from '@api';
import { derived, writable } from 'svelte/store'; import { derived, writable } from 'svelte/store';
import { assetGridState, assetStore } from './assets.store'; import { assetGridState, assetStore } from './assets.store';
@ -92,7 +92,7 @@ function createAssetInteractionStore() {
} }
if (nextBucket !== '') { if (nextBucket !== '') {
await assetStore.getAssetsByBucket(nextBucket); await assetStore.getAssetsByBucket(nextBucket, BucketPosition.Below);
navigateAsset(direction); navigateAsset(direction);
} }
return; return;

View file

@ -1,6 +1,5 @@
import { AssetGridState } from '$lib/models/asset-grid-state'; import { AssetGridState, BucketPosition } from '$lib/models/asset-grid-state';
import { calculateViewportHeightByNumberOfAsset } from '$lib/utils/viewport-utils'; import { AssetCountByTimeBucketResponseDto, api } from '@api';
import { api, AssetCountByTimeBucketResponseDto } from '@api';
import { sumBy, flatMap } from 'lodash-es'; import { sumBy, flatMap } from 'lodash-es';
import { writable } from 'svelte/store'; import { writable } from 'svelte/store';
@ -20,6 +19,18 @@ function createAssetStore() {
loadingBucketState.subscribe((state) => { loadingBucketState.subscribe((state) => {
_loadingBucketState = state; _loadingBucketState = state;
}); });
const estimateViewportHeight = (assetCount: number, viewportWidth: number): number => {
// Ideally we would use the average aspect ratio for the photoset, however assume
// a normal landscape aspect ratio of 3:2, then discount for the likelihood we
// will be scaling down and coalescing.
const thumbnailHeight = 235;
const unwrappedWidth = (3 / 2) * assetCount * thumbnailHeight * (7 / 10);
const rows = Math.ceil(unwrappedWidth / viewportWidth);
const height = rows * thumbnailHeight;
return height;
};
/** /**
* Set initial state * Set initial state
* @param viewportHeight * @param viewportHeight
@ -36,11 +47,12 @@ function createAssetStore() {
viewportHeight, viewportHeight,
viewportWidth, viewportWidth,
timelineHeight: 0, timelineHeight: 0,
buckets: data.buckets.map((d) => ({ buckets: data.buckets.map((bucket) => ({
bucketDate: d.timeBucket, bucketDate: bucket.timeBucket,
bucketHeight: calculateViewportHeightByNumberOfAsset(d.count, viewportWidth), bucketHeight: estimateViewportHeight(bucket.count, viewportWidth),
assets: [], assets: [],
cancelToken: new AbortController() cancelToken: new AbortController(),
position: BucketPosition.Unknown
})), })),
assets: [], assets: [],
userId userId
@ -53,10 +65,15 @@ function createAssetStore() {
}); });
}; };
const getAssetsByBucket = async (bucket: string) => { const getAssetsByBucket = async (bucket: string, position: BucketPosition) => {
try { try {
const currentBucketData = _assetGridState.buckets.find((b) => b.bucketDate === bucket); const currentBucketData = _assetGridState.buckets.find((b) => b.bucketDate === bucket);
if (currentBucketData?.assets && currentBucketData.assets.length > 0) { if (currentBucketData?.assets && currentBucketData.assets.length > 0) {
assetGridState.update((state) => {
const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucket);
state.buckets[bucketIndex].position = position;
return state;
});
return; return;
} }
@ -83,8 +100,8 @@ function createAssetStore() {
assetGridState.update((state) => { assetGridState.update((state) => {
const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucket); const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucket);
state.buckets[bucketIndex].assets = assets; state.buckets[bucketIndex].assets = assets;
state.buckets[bucketIndex].position = position;
state.assets = flatMap(state.buckets, (b) => b.assets); state.assets = flatMap(state.buckets, (b) => b.assets);
return state; return state;
}); });
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
@ -120,21 +137,31 @@ function createAssetStore() {
}); });
}; };
const updateBucketHeight = (bucket: string, actualBucketHeight: number) => { const updateBucketHeight = (bucket: string, actualBucketHeight: number): number => {
let scrollTimeline = false;
let heightDelta = 0;
assetGridState.update((state) => { assetGridState.update((state) => {
const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucket); const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucket);
// Update timeline height based on the new bucket height // Update timeline height based on the new bucket height
const estimateBucketHeight = state.buckets[bucketIndex].bucketHeight; const estimateBucketHeight = state.buckets[bucketIndex].bucketHeight;
if (actualBucketHeight >= estimateBucketHeight) { heightDelta = actualBucketHeight - estimateBucketHeight;
state.timelineHeight += actualBucketHeight - estimateBucketHeight; state.timelineHeight += heightDelta;
} else {
state.timelineHeight -= estimateBucketHeight - actualBucketHeight; scrollTimeline = state.buckets[bucketIndex].position == BucketPosition.Above;
}
state.buckets[bucketIndex].bucketHeight = actualBucketHeight; state.buckets[bucketIndex].bucketHeight = actualBucketHeight;
state.buckets[bucketIndex].position = BucketPosition.Unknown;
return state; return state;
}); });
if (scrollTimeline) {
return heightDelta;
}
return 0;
}; };
const cancelBucketRequest = async (token: AbortController, bucketDate: string) => { const cancelBucketRequest = async (token: AbortController, bucketDate: string) => {

View file

@ -150,3 +150,18 @@ export function getFileMimeType(file: File): string {
return ''; return '';
} }
} }
/**
* Returns aspect ratio for the asset
*/
export function getAssetRatio(asset: AssetResponseDto) {
let height = asset.exifInfo?.exifImageHeight || 235;
let width = asset.exifInfo?.exifImageWidth || 235;
const orientation = Number(asset.exifInfo?.orientation);
if (orientation) {
if (orientation == 6 || orientation == -90) {
[width, height] = [height, width];
}
}
return { width, height };
}

View file

@ -1,14 +0,0 @@
/**
* Glossary
* 1. Section: Group of assets in a month
*/
export function calculateViewportHeightByNumberOfAsset(assetCount: number, viewportWidth: number) {
const thumbnailHeight = 237;
// const unwrappedWidth = (3 / 2) * assetCount * thumbnailHeight * (7 / 10);
const unwrappedWidth = assetCount * thumbnailHeight;
const rows = Math.ceil(unwrappedWidth / viewportWidth);
const height = rows * thumbnailHeight;
return height;
}