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

fix(web): previous/next asset navigation (#3107)

* fix(web): previous/next asset navigation

* Apply suggestions from code review

Co-authored-by: Thomas <9749173+uhthomas@users.noreply.github.com>

* Call setViewingAsset once

* Make code more readable

* Avoid recursive call

* Simplify return statement

* Set position of the bucket to Unknown

---------

Co-authored-by: Thomas <9749173+uhthomas@users.noreply.github.com>
Co-authored-by: Alex Tran <alex.tran1502@gmail.com>
This commit is contained in:
Sergey Kondrikov 2023-07-05 23:24:23 +03:00 committed by GitHub
parent 814030be77
commit 37edef834e
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
3 changed files with 71 additions and 20 deletions

View file

@ -46,6 +46,11 @@ export class AssetGridState {
*/ */
assets: AssetResponseDto[] = []; assets: AssetResponseDto[] = [];
/**
* Total assets that have been loaded along with additional data
*/
loadedAssets: Record<string, number> = {};
/** /**
* User that owns assets * User that owns assets
*/ */

View file

@ -66,30 +66,62 @@ function createAssetInteractionStore() {
isViewingAssetStoreState.set(isViewing); isViewingAssetStoreState.set(isViewing);
}; };
const getNextAsset = async (currentBucketIndex: number, assetId: string): Promise<AssetResponseDto | null> => {
const currentBucket = _assetGridState.buckets[currentBucketIndex];
const assetIndex = currentBucket.assets.findIndex(({ id }) => id == assetId);
if (assetIndex === -1) {
return null;
}
if (assetIndex + 1 < currentBucket.assets.length) {
return currentBucket.assets[assetIndex + 1];
}
const nextBucketIndex = currentBucketIndex + 1;
if (nextBucketIndex >= _assetGridState.buckets.length) {
return null;
}
const nextBucket = _assetGridState.buckets[nextBucketIndex];
await assetStore.getAssetsByBucket(nextBucket.bucketDate, BucketPosition.Unknown);
return nextBucket.assets[0] ?? null;
};
const getPrevAsset = async (currentBucketIndex: number, assetId: string): Promise<AssetResponseDto | null> => {
const currentBucket = _assetGridState.buckets[currentBucketIndex];
const assetIndex = currentBucket.assets.findIndex(({ id }) => id == assetId);
if (assetIndex === -1) {
return null;
}
if (assetIndex > 0) {
return currentBucket.assets[assetIndex - 1];
}
const prevBucketIndex = currentBucketIndex - 1;
if (prevBucketIndex < 0) {
return null;
}
const prevBucket = _assetGridState.buckets[prevBucketIndex];
await assetStore.getAssetsByBucket(prevBucket.bucketDate, BucketPosition.Unknown);
return prevBucket.assets[prevBucket.assets.length - 1] ?? null;
};
const navigateAsset = async (direction: 'next' | 'previous') => { const navigateAsset = async (direction: 'next' | 'previous') => {
let index = _assetGridState.assets.findIndex(({ id }) => id === _viewingAssetStoreState.id); const currentAssetId = _viewingAssetStoreState.id;
const currentBucketIndex = _assetGridState.loadedAssets[currentAssetId];
index = direction === 'next' ? index + 1 : index - 1; if (currentBucketIndex < 0 || currentBucketIndex >= _assetGridState.buckets.length) {
const needMoreAbove = index < 0;
const needMoreBelow = index >= _assetGridState.assets.length;
// Try to load more assets if we're at the end.
if (needMoreAbove || needMoreBelow) {
for (const bucket of _assetGridState.buckets) {
if (bucket.assets.length === 0) {
await assetStore.getAssetsByBucket(
bucket.bucketDate,
needMoreAbove ? BucketPosition.Above : BucketPosition.Below,
);
navigateAsset(direction);
break;
}
}
return; return;
} }
const asset = _assetGridState.assets[index]; const asset =
direction === 'next'
? await getNextAsset(currentBucketIndex, currentAssetId)
: await getPrevAsset(currentBucketIndex, currentAssetId);
if (asset) { if (asset) {
setViewingAsset(asset); setViewingAsset(asset);
} }

View file

@ -30,6 +30,15 @@ function createAssetStore() {
return height; return height;
}; };
const refreshLoadedAssets = (state: AssetGridState): void => {
state.loadedAssets = {};
state.buckets.forEach((bucket, bucketIndex) =>
bucket.assets.map((asset) => {
state.loadedAssets[asset.id] = bucketIndex;
}),
);
};
/** /**
* Set initial state * Set initial state
* @param viewportHeight * @param viewportHeight
@ -54,6 +63,7 @@ function createAssetStore() {
position: BucketPosition.Unknown, position: BucketPosition.Unknown,
})), })),
assets: [], assets: [],
loadedAssets: {},
userId, userId,
}); });
@ -101,6 +111,7 @@ function createAssetStore() {
state.buckets[bucketIndex].assets = assets; state.buckets[bucketIndex].assets = assets;
state.buckets[bucketIndex].position = position; state.buckets[bucketIndex].position = position;
state.assets = state.buckets.flatMap((b) => b.assets); state.assets = state.buckets.flatMap((b) => b.assets);
refreshLoadedAssets(state);
return state; return state;
}); });
// eslint-disable-next-line @typescript-eslint/no-explicit-any // eslint-disable-next-line @typescript-eslint/no-explicit-any
@ -123,6 +134,7 @@ function createAssetStore() {
_removeBucket(state.buckets[bucketIndex].bucketDate); _removeBucket(state.buckets[bucketIndex].bucketDate);
} }
state.assets = state.buckets.flatMap((b) => b.assets); state.assets = state.buckets.flatMap((b) => b.assets);
refreshLoadedAssets(state);
return state; return state;
}); });
}; };
@ -132,6 +144,7 @@ function createAssetStore() {
const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucketDate); const bucketIndex = state.buckets.findIndex((b) => b.bucketDate === bucketDate);
state.buckets.splice(bucketIndex, 1); state.buckets.splice(bucketIndex, 1);
state.assets = state.buckets.flatMap((b) => b.assets); state.assets = state.buckets.flatMap((b) => b.assets);
refreshLoadedAssets(state);
return state; return state;
}); });
}; };
@ -180,6 +193,7 @@ function createAssetStore() {
state.buckets[bucketIndex].assets[assetIndex].isFavorite = isFavorite; state.buckets[bucketIndex].assets[assetIndex].isFavorite = isFavorite;
state.assets = state.buckets.flatMap((b) => b.assets); state.assets = state.buckets.flatMap((b) => b.assets);
refreshLoadedAssets(state);
return state; return state;
}); });
}; };