diff --git a/e2e/src/web/specs/shared-link.e2e-spec.ts b/e2e/src/web/specs/shared-link.e2e-spec.ts
index e40c20388b..fe7da0b2c0 100644
--- a/e2e/src/web/specs/shared-link.e2e-spec.ts
+++ b/e2e/src/web/specs/shared-link.e2e-spec.ts
@@ -44,7 +44,7 @@ test.describe('Shared Links', () => {
   test('download from a shared link', async ({ page }) => {
     await page.goto(`/share/${sharedLink.key}`);
     await page.getByRole('heading', { name: 'Test Album' }).waitFor();
-    await page.locator('.group > div').first().hover();
+    await page.locator('.group').first().hover();
     await page.waitForSelector('#asset-group-by-date svg');
     await page.getByRole('checkbox').click();
     await page.getByRole('button', { name: 'Download' }).click();
diff --git a/web/src/lib/actions/autogrow.ts b/web/src/lib/actions/autogrow.ts
index b79671afc8..ff80454ef3 100644
--- a/web/src/lib/actions/autogrow.ts
+++ b/web/src/lib/actions/autogrow.ts
@@ -1,4 +1,7 @@
 export const autoGrowHeight = (textarea: HTMLTextAreaElement, height = 'auto') => {
+  if (!textarea) {
+    return;
+  }
   textarea.style.height = height;
   textarea.style.height = `${textarea.scrollHeight}px`;
 };
diff --git a/web/src/lib/actions/intersection-observer.ts b/web/src/lib/actions/intersection-observer.ts
new file mode 100644
index 0000000000..222f76be63
--- /dev/null
+++ b/web/src/lib/actions/intersection-observer.ts
@@ -0,0 +1,152 @@
+type Config = IntersectionObserverActionProperties & {
+  observer?: IntersectionObserver;
+};
+type TrackedProperties = {
+  root?: Element | Document | null;
+  threshold?: number | number[];
+  top?: string;
+  right?: string;
+  bottom?: string;
+  left?: string;
+};
+type OnIntersectCallback = (entryOrElement: IntersectionObserverEntry | HTMLElement) => unknown;
+type OnSeperateCallback = (element: HTMLElement) => unknown;
+type IntersectionObserverActionProperties = {
+  key?: string;
+  onSeparate?: OnSeperateCallback;
+  onIntersect?: OnIntersectCallback;
+
+  root?: Element | Document | null;
+  threshold?: number | number[];
+  top?: string;
+  right?: string;
+  bottom?: string;
+  left?: string;
+
+  disabled?: boolean;
+};
+type TaskKey = HTMLElement | string;
+
+function isEquivalent(a: TrackedProperties, b: TrackedProperties) {
+  return (
+    a?.bottom === b?.bottom &&
+    a?.top === b?.top &&
+    a?.left === b?.left &&
+    a?.right == b?.right &&
+    a?.threshold === b?.threshold &&
+    a?.root === b?.root
+  );
+}
+
+const elementToConfig = new Map<TaskKey, Config>();
+
+const observe = (key: HTMLElement | string, target: HTMLElement, properties: IntersectionObserverActionProperties) => {
+  if (!target.isConnected) {
+    elementToConfig.get(key)?.observer?.unobserve(target);
+    return;
+  }
+  const {
+    root,
+    threshold,
+    top = '0px',
+    right = '0px',
+    bottom = '0px',
+    left = '0px',
+    onSeparate,
+    onIntersect,
+  } = properties;
+  const rootMargin = `${top} ${right} ${bottom} ${left}`;
+  const observer = new IntersectionObserver(
+    (entries: IntersectionObserverEntry[]) => {
+      // This IntersectionObserver is limited to observing a single element, the one the
+      // action is attached to. If there are multiple entries, it means that this
+      // observer is being notified of multiple events that have occured quickly together,
+      // and the latest element is the one we are interested in.
+
+      entries.sort((a, b) => a.time - b.time);
+
+      const latestEntry = entries.pop();
+      if (latestEntry?.isIntersecting) {
+        onIntersect?.(latestEntry);
+      } else {
+        onSeparate?.(target);
+      }
+    },
+    {
+      rootMargin,
+      threshold,
+      root,
+    },
+  );
+  observer.observe(target);
+  elementToConfig.set(key, { ...properties, observer });
+};
+
+function configure(key: HTMLElement | string, element: HTMLElement, properties: IntersectionObserverActionProperties) {
+  elementToConfig.set(key, properties);
+  observe(key, element, properties);
+}
+
+function _intersectionObserver(
+  key: HTMLElement | string,
+  element: HTMLElement,
+  properties: IntersectionObserverActionProperties,
+) {
+  if (properties.disabled) {
+    properties.onIntersect?.(element);
+  } else {
+    configure(key, element, properties);
+  }
+  return {
+    update(properties: IntersectionObserverActionProperties) {
+      const config = elementToConfig.get(key);
+      if (!config) {
+        return;
+      }
+      if (isEquivalent(config, properties)) {
+        return;
+      }
+      configure(key, element, properties);
+    },
+    destroy: () => {
+      if (properties.disabled) {
+        properties.onSeparate?.(element);
+      } else {
+        const config = elementToConfig.get(key);
+        const { observer, onSeparate } = config || {};
+        observer?.unobserve(element);
+        elementToConfig.delete(key);
+        if (onSeparate) {
+          onSeparate?.(element);
+        }
+      }
+    },
+  };
+}
+
+export function intersectionObserver(
+  element: HTMLElement,
+  properties: IntersectionObserverActionProperties | IntersectionObserverActionProperties[],
+) {
+  // svelte doesn't allow multiple use:action directives of the same kind on the same element,
+  // so accept an array when multiple configurations are needed.
+  if (Array.isArray(properties)) {
+    if (!properties.every((p) => p.key)) {
+      throw new Error('Multiple configurations must specify key');
+    }
+    const observers = properties.map((p) => _intersectionObserver(p.key as string, element, p));
+    return {
+      update: (properties: IntersectionObserverActionProperties[]) => {
+        for (const [i, props] of properties.entries()) {
+          observers[i].update(props);
+        }
+      },
+      destroy: () => {
+        for (const observer of observers) {
+          observer.destroy();
+        }
+      },
+    };
+  }
+  return _intersectionObserver(element, element, properties);
+}
diff --git a/web/src/lib/actions/resize-observer.ts b/web/src/lib/actions/resize-observer.ts
new file mode 100644
index 0000000000..9f3adc44b0
--- /dev/null
+++ b/web/src/lib/actions/resize-observer.ts
@@ -0,0 +1,43 @@
+type OnResizeCallback = (resizeEvent: { target: HTMLElement; width: number; height: number }) => void;
+
+let observer: ResizeObserver;
+let callbacks: WeakMap<HTMLElement, OnResizeCallback>;
+
+/**
+ * Installs a resizeObserver on the given element - when the element changes
+ * size, invokes a callback function with the width/height. Intended as a
+ * replacement for bind:clientWidth and bind:clientHeight in svelte4 which use
+ * an iframe to measure the size of the element, which can be bad for
+ * performance and memory usage. In svelte5, they adapted bind:clientHeight and
+ * bind:clientWidth to use an internal resize observer.
+ *
+ * TODO: When svelte5 is ready, go back to bind:clientWidth and
+ * bind:clientHeight.
+ */
+export function resizeObserver(element: HTMLElement, onResize: OnResizeCallback) {
+  if (!observer) {
+    callbacks = new WeakMap();
+    observer = new ResizeObserver((entries) => {
+      for (const entry of entries) {
+        const onResize = callbacks.get(entry.target as HTMLElement);
+        if (onResize) {
+          onResize({
+            target: entry.target as HTMLElement,
+            width: entry.borderBoxSize[0].inlineSize,
+            height: entry.borderBoxSize[0].blockSize,
+          });
+        }
+      }
+    });
+  }
+
+  callbacks.set(element, onResize);
+  observer.observe(element);
+
+  return {
+    destroy: () => {
+      callbacks.delete(element);
+      observer.unobserve(element);
+    },
+  };
+}
diff --git a/web/src/lib/actions/thumbhash.ts b/web/src/lib/actions/thumbhash.ts
new file mode 100644
index 0000000000..ab9d28ffc9
--- /dev/null
+++ b/web/src/lib/actions/thumbhash.ts
@@ -0,0 +1,14 @@
+import { decodeBase64 } from '$lib/utils';
+import { thumbHashToRGBA } from 'thumbhash';
+
+export function thumbhash(canvas: HTMLCanvasElement, { base64ThumbHash }: { base64ThumbHash: string }) {
+  const ctx = canvas.getContext('2d');
+  if (ctx) {
+    const { w, h, rgba } = thumbHashToRGBA(decodeBase64(base64ThumbHash));
+    const pixels = ctx.createImageData(w, h);
+    canvas.width = w;
+    canvas.height = h;
+    pixels.data.set(rgba);
+    ctx.putImageData(pixels, 0, 0);
+  }
+}
diff --git a/web/src/lib/components/album-page/album-viewer.svelte b/web/src/lib/components/album-page/album-viewer.svelte
index 7a88aa740b..2256c79eb0 100644
--- a/web/src/lib/components/album-page/album-viewer.svelte
+++ b/web/src/lib/components/album-page/album-viewer.svelte
@@ -19,6 +19,7 @@
   import { handlePromiseError } from '$lib/utils';
   import AlbumSummary from './album-summary.svelte';
   import { t } from 'svelte-i18n';
+  import { onDestroy } from 'svelte';
 
   export let sharedLink: SharedLinkResponseDto;
   export let user: UserResponseDto | undefined = undefined;
@@ -38,6 +39,9 @@
       dragAndDropFilesStore.set({ isDragging: false, files: [] });
     }
   });
+  onDestroy(() => {
+    assetStore.destroy();
+  });
 </script>
 
 <svelte:window
@@ -94,7 +98,7 @@
 </header>
 
 <main class="relative h-screen overflow-hidden bg-immich-bg px-6 pt-[var(--navbar-height)] dark:bg-immich-dark-bg">
-  <AssetGrid {album} {assetStore} {assetInteractionStore}>
+  <AssetGrid enableRouting={true} {album} {assetStore} {assetInteractionStore}>
     <section class="pt-8 md:pt-24">
       <!-- ALBUM TITLE -->
       <h1
diff --git a/web/src/lib/components/asset-viewer/asset-viewer.svelte b/web/src/lib/components/asset-viewer/asset-viewer.svelte
index 7c541fbf7a..3ed955848b 100644
--- a/web/src/lib/components/asset-viewer/asset-viewer.svelte
+++ b/web/src/lib/components/asset-viewer/asset-viewer.svelte
@@ -15,7 +15,6 @@
   import { websocketEvents } from '$lib/stores/websocket';
   import { getAssetJobMessage, getSharedLink, handlePromiseError, isSharedLink } from '$lib/utils';
   import { handleError } from '$lib/utils/handle-error';
-  import { navigate } from '$lib/utils/navigation';
   import { SlideshowHistory } from '$lib/utils/slideshow-history';
   import {
     AssetJobName,
@@ -70,7 +69,8 @@
   } = slideshowStore;
 
   const dispatch = createEventDispatcher<{
-    close: void;
+    action: { type: AssetAction; asset: AssetResponseDto };
+    close: { asset: AssetResponseDto };
     next: void;
     previous: void;
   }>();
@@ -201,7 +201,6 @@
       websocketEvents.on('on_asset_update', onAssetUpdate),
     );
 
-    await navigate({ targetRoute: 'current', assetId: asset.id });
     slideshowStateUnsubscribe = slideshowState.subscribe((value) => {
       if (value === SlideshowState.PlaySlideshow) {
         slideshowHistory.reset();
@@ -268,9 +267,8 @@
     $isShowDetail = !$isShowDetail;
   };
 
-  const closeViewer = async () => {
-    dispatch('close');
-    await navigate({ targetRoute: 'current', assetId: null });
+  const closeViewer = () => {
+    dispatch('close', { asset });
   };
 
   const closeEditor = () => {
@@ -378,9 +376,7 @@
     }
   };
 
-  const handleStackedAssetMouseEvent = (e: CustomEvent<{ isMouseOver: boolean }>, asset: AssetResponseDto) => {
-    const { isMouseOver } = e.detail;
-
+  const handleStackedAssetMouseEvent = (isMouseOver: boolean, asset: AssetResponseDto) => {
     previewStackedAsset = isMouseOver ? asset : undefined;
   };
 
@@ -392,8 +388,7 @@
       }
 
       case AssetAction.UNSTACK: {
-        await closeViewer();
-        break;
+        closeViewer();
       }
     }
 
@@ -585,12 +580,11 @@
                 ? 'bg-transparent border-2 border-white'
                 : 'bg-gray-700/40'} inline-block hover:bg-transparent"
               asset={stackedAsset}
-              onClick={(stackedAsset, event) => {
-                event.preventDefault();
+              onClick={(stackedAsset) => {
                 asset = stackedAsset;
                 preloadAssets = index + 1 >= stackedAssets.length ? [] : [stackedAssets[index + 1]];
               }}
-              on:mouse-event={(e) => handleStackedAssetMouseEvent(e, stackedAsset)}
+              onMouseEvent={({ isMouseOver }) => handleStackedAssetMouseEvent(isMouseOver, stackedAsset)}
               readonly
               thumbnailSize={stackedAsset.id == asset.id ? 65 : 60}
               showStackedIcon={false}
diff --git a/web/src/lib/components/asset-viewer/detail-panel.svelte b/web/src/lib/components/asset-viewer/detail-panel.svelte
index 4ff2084b9a..88417f248f 100644
--- a/web/src/lib/components/asset-viewer/detail-panel.svelte
+++ b/web/src/lib/components/asset-viewer/detail-panel.svelte
@@ -212,7 +212,6 @@
                   title={person.name}
                   widthStyle="90px"
                   heightStyle="90px"
-                  thumbhash={null}
                   hidden={person.isHidden}
                 />
               </div>
diff --git a/web/src/lib/components/asset-viewer/intersection-observer.svelte b/web/src/lib/components/asset-viewer/intersection-observer.svelte
deleted file mode 100644
index df89a2ed7d..0000000000
--- a/web/src/lib/components/asset-viewer/intersection-observer.svelte
+++ /dev/null
@@ -1,82 +0,0 @@
-<script lang="ts">
-  import { BucketPosition } from '$lib/stores/assets.store';
-  import { createEventDispatcher, onMount } from 'svelte';
-
-  export let once = false;
-  export let top = 0;
-  export let bottom = 0;
-  export let left = 0;
-  export let right = 0;
-  export let root: HTMLElement | null = null;
-
-  export let intersecting = false;
-  let container: HTMLDivElement;
-  const dispatch = createEventDispatcher<{
-    hidden: HTMLDivElement;
-    intersected: {
-      container: HTMLDivElement;
-      position: BucketPosition;
-    };
-  }>();
-
-  onMount(() => {
-    if (typeof IntersectionObserver !== 'undefined') {
-      const rootMargin = `${top}px ${right}px ${bottom}px ${left}px`;
-      const observer = new IntersectionObserver(
-        (entries) => {
-          intersecting = entries.some((entry) => entry.isIntersecting);
-          if (!intersecting) {
-            dispatch('hidden', container);
-          }
-
-          if (intersecting && once) {
-            observer.unobserve(container);
-          }
-
-          if (intersecting) {
-            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,
-            });
-          }
-        },
-        {
-          rootMargin,
-          root,
-        },
-      );
-
-      observer.observe(container);
-      return () => observer.unobserve(container);
-    }
-
-    // The following is a fallback for older browsers
-    function handler() {
-      const bcr = container.getBoundingClientRect();
-
-      intersecting =
-        bcr.bottom + bottom > 0 &&
-        bcr.right + right > 0 &&
-        bcr.top - top < window.innerHeight &&
-        bcr.left - left < window.innerWidth;
-
-      if (intersecting && once) {
-        window.removeEventListener('scroll', handler);
-      }
-    }
-
-    window.addEventListener('scroll', handler);
-    return () => window.removeEventListener('scroll', handler);
-  });
-</script>
-
-<div bind:this={container}>
-  <slot {intersecting} />
-</div>
diff --git a/web/src/lib/components/asset-viewer/photo-viewer.svelte b/web/src/lib/components/asset-viewer/photo-viewer.svelte
index 966f382838..3919033e4a 100644
--- a/web/src/lib/components/asset-viewer/photo-viewer.svelte
+++ b/web/src/lib/components/asset-viewer/photo-viewer.svelte
@@ -12,7 +12,7 @@
   import { AssetTypeEnum, type AssetResponseDto, AssetMediaSize, type SharedLinkResponseDto } from '@immich/sdk';
   import { zoomImageAction, zoomed } from '$lib/actions/zoom-image';
   import { canCopyImagesToClipboard, copyImageToClipboard } from 'copy-image-clipboard';
-  import { onDestroy } from 'svelte';
+  import { onDestroy, onMount } from 'svelte';
 
   import { fade } from 'svelte/transition';
   import LoadingSpinner from '../shared-components/loading-spinner.svelte';
@@ -33,6 +33,7 @@
   let imageLoaded: boolean = false;
   let imageError: boolean = false;
   let forceUseOriginal: boolean = false;
+  let loader: HTMLImageElement;
 
   $: isWebCompatible = isWebCompatibleImage(asset);
   $: useOriginalByDefault = isWebCompatible && $alwaysLoadOriginalFile;
@@ -108,6 +109,25 @@
     event.preventDefault();
     handlePromiseError(copyImage());
   };
+
+  onMount(() => {
+    const onload = () => {
+      imageLoaded = true;
+      assetFileUrl = imageLoaderUrl;
+    };
+    const onerror = () => {
+      imageError = imageLoaded = true;
+    };
+    if (loader.complete) {
+      onload();
+    }
+    loader.addEventListener('load', onload);
+    loader.addEventListener('error', onerror);
+    return () => {
+      loader?.removeEventListener('load', onload);
+      loader?.removeEventListener('error', onerror);
+    };
+  });
 </script>
 
 <svelte:window
@@ -119,6 +139,8 @@
 {#if imageError}
   <div class="h-full flex items-center justify-center">{$t('error_loading_image')}</div>
 {/if}
+<!-- svelte-ignore a11y-missing-attribute -->
+<img bind:this={loader} style="display:none" src={imageLoaderUrl} aria-hidden="true" />
 <div bind:this={element} class="relative h-full select-none">
   <img
     style="display:none"
@@ -128,7 +150,7 @@
     on:error={() => (imageError = imageLoaded = true)}
   />
   {#if !imageLoaded}
-    <div class="flex h-full items-center justify-center">
+    <div id="spinner" class="flex h-full items-center justify-center">
       <LoadingSpinner />
     </div>
   {:else if !imageError}
@@ -159,3 +181,15 @@
     </div>
   {/if}
 </div>
+
+<style>
+  @keyframes delayedVisibility {
+    to {
+      visibility: visible;
+    }
+  }
+  #spinner {
+    visibility: hidden;
+    animation: 0s linear 0.4s forwards delayedVisibility;
+  }
+</style>
diff --git a/web/src/lib/components/assets/thumbnail/__test__/image-thumbnail.spec.ts b/web/src/lib/components/assets/thumbnail/__test__/image-thumbnail.spec.ts
index 91ea7d3ab1..2525b86160 100644
--- a/web/src/lib/components/assets/thumbnail/__test__/image-thumbnail.spec.ts
+++ b/web/src/lib/components/assets/thumbnail/__test__/image-thumbnail.spec.ts
@@ -3,8 +3,8 @@ import { render } from '@testing-library/svelte';
 
 describe('ImageThumbnail component', () => {
   beforeAll(() => {
-    Object.defineProperty(HTMLImageElement.prototype, 'decode', {
-      value: vi.fn(),
+    Object.defineProperty(HTMLImageElement.prototype, 'complete', {
+      value: true,
     });
   });
 
@@ -12,13 +12,11 @@ describe('ImageThumbnail component', () => {
     const sut = render(ImageThumbnail, {
       url: 'http://localhost/img.png',
       altText: 'test',
-      thumbhash: '1QcSHQRnh493V4dIh4eXh1h4kJUI',
+      base64ThumbHash: '1QcSHQRnh493V4dIh4eXh1h4kJUI',
       widthStyle: '250px',
     });
 
-    const [_, thumbhash] = sut.getAllByRole('img');
-    expect(thumbhash.getAttribute('src')).toContain(
-      '', // truncated
-    );
+    const thumbhash = sut.getByTestId('thumbhash');
+    expect(thumbhash).not.toBeFalsy();
   });
 });
diff --git a/web/src/lib/components/assets/thumbnail/image-thumbnail.svelte b/web/src/lib/components/assets/thumbnail/image-thumbnail.svelte
index 8e391ecb59..e03dd35653 100644
--- a/web/src/lib/components/assets/thumbnail/image-thumbnail.svelte
+++ b/web/src/lib/components/assets/thumbnail/image-thumbnail.svelte
@@ -1,17 +1,19 @@
 <script lang="ts">
-  import { onMount, tick } from 'svelte';
-  import { decodeBase64 } from '$lib/utils';
+  import { onMount } from 'svelte';
+
   import { fade } from 'svelte/transition';
-  import { thumbHashToDataURL } from 'thumbhash';
-  import { mdiEyeOffOutline } from '@mdi/js';
+
+  import { thumbhash } from '$lib/actions/thumbhash';
   import Icon from '$lib/components/elements/icon.svelte';
+  import { TUNABLES } from '$lib/utils/tunables';
+  import { mdiEyeOffOutline, mdiImageBrokenVariant } from '@mdi/js';
 
   export let url: string;
   export let altText: string | undefined;
   export let title: string | null = null;
   export let heightStyle: string | undefined = undefined;
   export let widthStyle: string;
-  export let thumbhash: string | null = null;
+  export let base64ThumbHash: string | null = null;
   export let curve = false;
   export let shadow = false;
   export let circle = false;
@@ -19,37 +21,58 @@
   export let border = false;
   export let preload = true;
   export let hiddenIconClass = 'text-white';
+  export let onComplete: (() => void) | undefined = undefined;
+
+  let {
+    IMAGE_THUMBNAIL: { THUMBHASH_FADE_DURATION },
+  } = TUNABLES;
+
+  let loaded = false;
+  let errored = false;
 
-  let complete = false;
   let img: HTMLImageElement;
 
-  onMount(async () => {
-    await img.decode();
-    await tick();
-    complete = true;
+  const setLoaded = () => {
+    loaded = true;
+    onComplete?.();
+  };
+  const setErrored = () => {
+    errored = true;
+    onComplete?.();
+  };
+  onMount(() => {
+    if (img.complete) {
+      setLoaded();
+    }
   });
 </script>
 
-<img
-  bind:this={img}
-  loading={preload ? 'eager' : 'lazy'}
-  style:width={widthStyle}
-  style:height={heightStyle}
-  style:filter={hidden ? 'grayscale(50%)' : 'none'}
-  style:opacity={hidden ? '0.5' : '1'}
-  src={url}
-  alt={altText}
-  {title}
-  class="object-cover transition duration-300 {border
-    ? 'border-[3px] border-immich-dark-primary/80 hover:border-immich-primary'
-    : ''}"
-  class:rounded-xl={curve}
-  class:shadow-lg={shadow}
-  class:rounded-full={circle}
-  class:aspect-square={circle || !heightStyle}
-  class:opacity-0={!thumbhash && !complete}
-  draggable="false"
-/>
+{#if errored}
+  <div class="absolute flex h-full w-full items-center justify-center p-4 z-10">
+    <Icon path={mdiImageBrokenVariant} size="48" />
+  </div>
+{:else}
+  <img
+    bind:this={img}
+    on:load={setLoaded}
+    on:error={setErrored}
+    loading={preload ? 'eager' : 'lazy'}
+    style:width={widthStyle}
+    style:height={heightStyle}
+    style:filter={hidden ? 'grayscale(50%)' : 'none'}
+    style:opacity={hidden ? '0.5' : '1'}
+    src={url}
+    alt={loaded || errored ? altText : ''}
+    {title}
+    class="object-cover {border ? 'border-[3px] border-immich-dark-primary/80 hover:border-immich-primary' : ''}"
+    class:rounded-xl={curve}
+    class:shadow-lg={shadow}
+    class:rounded-full={circle}
+    class:aspect-square={circle || !heightStyle}
+    class:opacity-0={!thumbhash && !loaded}
+    draggable="false"
+  />
+{/if}
 
 {#if hidden}
   <div class="absolute left-1/2 top-1/2 translate-x-[-50%] translate-y-[-50%] transform">
@@ -57,18 +80,18 @@
   </div>
 {/if}
 
-{#if thumbhash && !complete}
-  <img
+{#if base64ThumbHash && (!loaded || errored)}
+  <canvas
+    use:thumbhash={{ base64ThumbHash }}
+    data-testid="thumbhash"
     style:width={widthStyle}
     style:height={heightStyle}
-    src={thumbHashToDataURL(decodeBase64(thumbhash))}
-    alt={altText}
     {title}
     class="absolute top-0 object-cover"
     class:rounded-xl={curve}
     class:shadow-lg={shadow}
     class:rounded-full={circle}
     draggable="false"
-    out:fade={{ duration: 300 }}
+    out:fade={{ duration: THUMBHASH_FADE_DURATION }}
   />
 {/if}
diff --git a/web/src/lib/components/assets/thumbnail/thumbnail.svelte b/web/src/lib/components/assets/thumbnail/thumbnail.svelte
index 6b0bd2ee75..c9fbf133c8 100644
--- a/web/src/lib/components/assets/thumbnail/thumbnail.svelte
+++ b/web/src/lib/components/assets/thumbnail/thumbnail.svelte
@@ -1,5 +1,5 @@
 <script lang="ts">
-  import IntersectionObserver from '$lib/components/asset-viewer/intersection-observer.svelte';
+  import { intersectionObserver } from '$lib/actions/intersection-observer';
   import Icon from '$lib/components/elements/icon.svelte';
   import { ProjectionType } from '$lib/constants';
   import { getAssetThumbnailUrl, isSharedLink } from '$lib/utils';
@@ -18,18 +18,23 @@
     mdiMotionPlayOutline,
     mdiRotate360,
   } from '@mdi/js';
-  import { createEventDispatcher } from 'svelte';
+
   import { fade } from 'svelte/transition';
   import ImageThumbnail from './image-thumbnail.svelte';
   import VideoThumbnail from './video-thumbnail.svelte';
   import { currentUrlReplaceAssetId } from '$lib/utils/navigation';
+  import { AssetStore } from '$lib/stores/assets.store';
 
-  const dispatch = createEventDispatcher<{
-    select: { asset: AssetResponseDto };
-    'mouse-event': { isMouseOver: boolean; selectedGroupIndex: number };
-  }>();
+  import type { DateGroup } from '$lib/utils/timeline-util';
+
+  import { generateId } from '$lib/utils/generate-id';
+  import { onDestroy } from 'svelte';
+  import { TUNABLES } from '$lib/utils/tunables';
+  import { thumbhash } from '$lib/actions/thumbhash';
 
   export let asset: AssetResponseDto;
+  export let dateGroup: DateGroup | undefined = undefined;
+  export let assetStore: AssetStore | undefined = undefined;
   export let groupIndex = 0;
   export let thumbnailSize: number | undefined = undefined;
   export let thumbnailWidth: number | undefined = undefined;
@@ -40,72 +45,181 @@
   export let readonly = false;
   export let showArchiveIcon = false;
   export let showStackedIcon = true;
-  export let onClick: ((asset: AssetResponseDto, event: Event) => void) | undefined = undefined;
+  export let intersectionConfig: {
+    root?: HTMLElement;
+    bottom?: string;
+    top?: string;
+    left?: string;
+    priority?: number;
+    disabled?: boolean;
+  } = {};
+
+  export let retrieveElement: boolean = false;
+  export let onIntersected: (() => void) | undefined = undefined;
+  export let onClick: ((asset: AssetResponseDto) => void) | undefined = undefined;
+  export let onRetrieveElement: ((elment: HTMLElement) => void) | undefined = undefined;
+  export let onSelect: ((asset: AssetResponseDto) => void) | undefined = undefined;
+  export let onMouseEvent: ((event: { isMouseOver: boolean; selectedGroupIndex: number }) => void) | undefined =
+    undefined;
 
   let className = '';
   export { className as class };
 
+  let {
+    IMAGE_THUMBNAIL: { THUMBHASH_FADE_DURATION },
+  } = TUNABLES;
+
+  const componentId = generateId();
+  let element: HTMLElement | undefined;
   let mouseOver = false;
+  let intersecting = false;
+  let lastRetrievedElement: HTMLElement | undefined;
+  let loaded = false;
 
-  $: dispatch('mouse-event', { isMouseOver: mouseOver, selectedGroupIndex: groupIndex });
+  $: if (!retrieveElement) {
+    lastRetrievedElement = undefined;
+  }
+  $: if (retrieveElement && element && lastRetrievedElement !== element) {
+    lastRetrievedElement = element;
+    onRetrieveElement?.(element);
+  }
 
-  $: [width, height] = ((): [number, number] => {
-    if (thumbnailSize) {
-      return [thumbnailSize, thumbnailSize];
-    }
+  $: width = thumbnailSize || thumbnailWidth || 235;
+  $: height = thumbnailSize || thumbnailHeight || 235;
+  $: display = intersecting;
 
-    if (thumbnailWidth && thumbnailHeight) {
-      return [thumbnailWidth, thumbnailHeight];
-    }
-
-    return [235, 235];
-  })();
-
-  const onIconClickedHandler = (e: MouseEvent) => {
-    e.stopPropagation();
-    e.preventDefault();
+  const onIconClickedHandler = (e?: MouseEvent) => {
+    e?.stopPropagation();
+    e?.preventDefault();
     if (!disabled) {
-      dispatch('select', { asset });
+      onSelect?.(asset);
     }
   };
 
+  const callClickHandlers = () => {
+    if (selected) {
+      onIconClickedHandler();
+      return;
+    }
+    onClick?.(asset);
+  };
   const handleClick = (e: MouseEvent) => {
     if (e.ctrlKey || e.metaKey) {
       return;
     }
+    e.stopPropagation();
+    e.preventDefault();
+    callClickHandlers();
+  };
 
-    if (selected) {
-      onIconClickedHandler(e);
-      return;
-    }
-
-    onClick?.(asset, e);
+  const _onMouseEnter = () => {
+    mouseOver = true;
+    onMouseEvent?.({ isMouseOver: true, selectedGroupIndex: groupIndex });
   };
 
   const onMouseEnter = () => {
-    mouseOver = true;
+    if (dateGroup && assetStore) {
+      assetStore.taskManager.queueScrollSensitiveTask({ componentId, task: () => _onMouseEnter() });
+    } else {
+      _onMouseEnter();
+    }
   };
 
   const onMouseLeave = () => {
-    mouseOver = false;
+    if (dateGroup && assetStore) {
+      assetStore.taskManager.queueScrollSensitiveTask({ componentId, task: () => (mouseOver = false) });
+    } else {
+      mouseOver = false;
+    }
   };
+
+  const _onIntersect = () => {
+    intersecting = true;
+    onIntersected?.();
+  };
+
+  const onIntersect = () => {
+    if (intersecting === true) {
+      return;
+    }
+    if (dateGroup && assetStore) {
+      assetStore.taskManager.intersectedThumbnail(componentId, dateGroup, asset, () => void _onIntersect());
+    } else {
+      void _onIntersect();
+    }
+  };
+
+  const onSeparate = () => {
+    if (intersecting === false) {
+      return;
+    }
+    if (dateGroup && assetStore) {
+      assetStore.taskManager.seperatedThumbnail(componentId, dateGroup, asset, () => (intersecting = false));
+    } else {
+      intersecting = false;
+    }
+  };
+
+  onDestroy(() => {
+    assetStore?.taskManager.removeAllTasksForComponent(componentId);
+  });
 </script>
 
-<IntersectionObserver once={false} on:intersected let:intersecting>
-  <a
-    href={currentUrlReplaceAssetId(asset.id)}
-    style:width="{width}px"
-    style:height="{height}px"
-    class="group focus-visible:outline-none flex overflow-hidden {disabled
-      ? 'bg-gray-300'
-      : 'bg-immich-primary/20 dark:bg-immich-dark-primary/20'}"
-    class:cursor-not-allowed={disabled}
-    on:mouseenter={onMouseEnter}
-    on:mouseleave={onMouseLeave}
-    tabindex={0}
-    on:click={handleClick}
-  >
-    {#if intersecting}
+<div
+  bind:this={element}
+  use:intersectionObserver={{
+    ...intersectionConfig,
+    onIntersect,
+    onSeparate,
+  }}
+  data-asset={asset.id}
+  data-int={intersecting}
+  style:width="{width}px"
+  style:height="{height}px"
+  class="group focus-visible:outline-none flex overflow-hidden {disabled
+    ? 'bg-gray-300'
+    : 'bg-immich-primary/20 dark:bg-immich-dark-primary/20'}"
+>
+  {#if !loaded && asset.thumbhash}
+    <canvas
+      use:thumbhash={{ base64ThumbHash: asset.thumbhash }}
+      class="absolute object-cover z-10"
+      style:width="{width}px"
+      style:height="{height}px"
+      out:fade={{ duration: THUMBHASH_FADE_DURATION }}
+    ></canvas>
+  {/if}
+
+  {#if display}
+    <!-- svelte queries for all links on afterNavigate, leading to performance problems in asset-grid which updates
+     the navigation url on scroll. Replace this with button for now. -->
+    <div
+      class:cursor-not-allowed={disabled}
+      class:cursor-pointer={!disabled}
+      on:mouseenter={onMouseEnter}
+      on:mouseleave={onMouseLeave}
+      on:keypress={(evt) => {
+        if (evt.key === 'Enter') {
+          callClickHandlers();
+        }
+      }}
+      tabindex={0}
+      on:click={handleClick}
+      role="link"
+    >
+      {#if mouseOver}
+        <!-- lazy show the url on mouse over-->
+        <a
+          class="absolute z-30 {className} top-[41px]"
+          style:cursor="unset"
+          style:width="{width}px"
+          style:height="{height}px"
+          href={currentUrlReplaceAssetId(asset.id)}
+          on:click={(evt) => evt.preventDefault()}
+          tabindex={0}
+        >
+        </a>
+      {/if}
       <div class="absolute z-20 {className}" style:width="{width}px" style:height="{height}px">
         <!-- Select asset button  -->
         {#if !readonly && (mouseOver || selected || selectionCandidate)}
@@ -189,11 +303,11 @@
             altText={$getAltText(asset)}
             widthStyle="{width}px"
             heightStyle="{height}px"
-            thumbhash={asset.thumbhash}
             curve={selected}
+            onComplete={() => (loaded = true)}
           />
         {:else}
-          <div class="flex h-full w-full items-center justify-center p-4">
+          <div class="absolute flex h-full w-full items-center justify-center p-4 z-10">
             <Icon path={mdiImageBrokenVariant} size="48" />
           </div>
         {/if}
@@ -201,6 +315,7 @@
         {#if asset.type === AssetTypeEnum.Video}
           <div class="absolute top-0 h-full w-full">
             <VideoThumbnail
+              {assetStore}
               url={getAssetPlaybackUrl({ id: asset.id, checksum: asset.checksum })}
               enablePlayback={mouseOver && $playVideoThumbnailOnHover}
               curve={selected}
@@ -213,6 +328,7 @@
         {#if asset.type === AssetTypeEnum.Image && asset.livePhotoVideoId}
           <div class="absolute top-0 h-full w-full">
             <VideoThumbnail
+              {assetStore}
               url={getAssetPlaybackUrl({ id: asset.livePhotoVideoId, checksum: asset.checksum })}
               pauseIcon={mdiMotionPauseOutline}
               playIcon={mdiMotionPlayOutline}
@@ -230,6 +346,6 @@
           out:fade={{ duration: 100 }}
         />
       {/if}
-    {/if}
-  </a>
-</IntersectionObserver>
+    </div>
+  {/if}
+</div>
diff --git a/web/src/lib/components/assets/thumbnail/video-thumbnail.svelte b/web/src/lib/components/assets/thumbnail/video-thumbnail.svelte
index 5c4196e54b..5cac0b1945 100644
--- a/web/src/lib/components/assets/thumbnail/video-thumbnail.svelte
+++ b/web/src/lib/components/assets/thumbnail/video-thumbnail.svelte
@@ -3,7 +3,11 @@
   import LoadingSpinner from '$lib/components/shared-components/loading-spinner.svelte';
   import { mdiAlertCircleOutline, mdiPauseCircleOutline, mdiPlayCircleOutline } from '@mdi/js';
   import Icon from '$lib/components/elements/icon.svelte';
+  import { AssetStore } from '$lib/stores/assets.store';
+  import { generateId } from '$lib/utils/generate-id';
+  import { onDestroy } from 'svelte';
 
+  export let assetStore: AssetStore | undefined = undefined;
   export let url: string;
   export let durationInSeconds = 0;
   export let enablePlayback = false;
@@ -13,6 +17,7 @@
   export let playIcon = mdiPlayCircleOutline;
   export let pauseIcon = mdiPauseCircleOutline;
 
+  const componentId = generateId();
   let remainingSeconds = durationInSeconds;
   let loading = true;
   let error = false;
@@ -27,6 +32,43 @@
       player.src = '';
     }
   }
+  const onMouseEnter = () => {
+    if (assetStore) {
+      assetStore.taskManager.queueScrollSensitiveTask({
+        componentId,
+        task: () => {
+          if (playbackOnIconHover) {
+            enablePlayback = true;
+          }
+        },
+      });
+    } else {
+      if (playbackOnIconHover) {
+        enablePlayback = true;
+      }
+    }
+  };
+
+  const onMouseLeave = () => {
+    if (assetStore) {
+      assetStore.taskManager.queueScrollSensitiveTask({
+        componentId,
+        task: () => {
+          if (playbackOnIconHover) {
+            enablePlayback = false;
+          }
+        },
+      });
+    } else {
+      if (playbackOnIconHover) {
+        enablePlayback = false;
+      }
+    }
+  };
+
+  onDestroy(() => {
+    assetStore?.taskManager.removeAllTasksForComponent(componentId);
+  });
 </script>
 
 <div class="absolute right-0 top-0 z-20 flex place-items-center gap-1 text-xs font-medium text-white">
@@ -37,19 +79,7 @@
   {/if}
 
   <!-- svelte-ignore a11y-no-static-element-interactions -->
-  <span
-    class="pr-2 pt-2"
-    on:mouseenter={() => {
-      if (playbackOnIconHover) {
-        enablePlayback = true;
-      }
-    }}
-    on:mouseleave={() => {
-      if (playbackOnIconHover) {
-        enablePlayback = false;
-      }
-    }}
-  >
+  <span class="pr-2 pt-2" on:mouseenter={onMouseEnter} on:mouseleave={onMouseLeave}>
     {#if enablePlayback}
       {#if loading}
         <LoadingSpinner />
diff --git a/web/src/lib/components/faces-page/assign-face-side-panel.svelte b/web/src/lib/components/faces-page/assign-face-side-panel.svelte
index 0dd4251dab..eba26e6e61 100644
--- a/web/src/lib/components/faces-page/assign-face-side-panel.svelte
+++ b/web/src/lib/components/faces-page/assign-face-side-panel.svelte
@@ -113,7 +113,6 @@
                   title={$getPersonNameWithHiddenValue(person.name, person.isHidden)}
                   widthStyle="90px"
                   heightStyle="90px"
-                  thumbhash={null}
                   hidden={person.isHidden}
                 />
               </div>
diff --git a/web/src/lib/components/faces-page/person-side-panel.svelte b/web/src/lib/components/faces-page/person-side-panel.svelte
index 712100763c..fd4fbdf964 100644
--- a/web/src/lib/components/faces-page/person-side-panel.svelte
+++ b/web/src/lib/components/faces-page/person-side-panel.svelte
@@ -265,8 +265,6 @@
                       title={$t('face_unassigned')}
                       widthStyle="90px"
                       heightStyle="90px"
-                      thumbhash={null}
-                      hidden={false}
                     />
                   {:then data}
                     <ImageThumbnail
@@ -277,8 +275,6 @@
                       title={$t('face_unassigned')}
                       widthStyle="90px"
                       heightStyle="90px"
-                      thumbhash={null}
-                      hidden={false}
                     />
                   {/await}
                 {/if}
diff --git a/web/src/lib/components/memory-page/memory-viewer.svelte b/web/src/lib/components/memory-page/memory-viewer.svelte
index 2a6e121c3a..250cb379cc 100644
--- a/web/src/lib/components/memory-page/memory-viewer.svelte
+++ b/web/src/lib/components/memory-page/memory-viewer.svelte
@@ -2,7 +2,7 @@
   import { goto } from '$app/navigation';
   import { page } from '$app/stores';
   import { shortcuts } from '$lib/actions/shortcut';
-  import IntersectionObserver from '$lib/components/asset-viewer/intersection-observer.svelte';
+
   import CircleIconButton from '$lib/components/elements/buttons/circle-icon-button.svelte';
   import AddToAlbum from '$lib/components/photos-page/actions/add-to-album.svelte';
   import ArchiveAction from '$lib/components/photos-page/actions/archive-action.svelte';
@@ -38,6 +38,8 @@
   import { tweened } from 'svelte/motion';
   import { fade } from 'svelte/transition';
   import { t } from 'svelte-i18n';
+  import { intersectionObserver } from '$lib/actions/intersection-observer';
+  import { resizeObserver } from '$lib/actions/resize-observer';
   import { locale } from '$lib/stores/preferences.store';
 
   const parseIndex = (s: string | null, max: number | null) =>
@@ -383,21 +385,18 @@
         />
       </div>
 
-      <IntersectionObserver
-        once={false}
-        on:intersected={() => (galleryInView = true)}
-        on:hidden={() => (galleryInView = false)}
-        bottom={-200}
+      <div
+        id="gallery-memory"
+        use:intersectionObserver={{
+          onIntersect: () => (galleryInView = true),
+          onSeparate: () => (galleryInView = false),
+          bottom: '-200px',
+        }}
+        use:resizeObserver={({ height, width }) => ((viewport.height = height), (viewport.width = width))}
+        bind:this={memoryGallery}
       >
-        <div
-          id="gallery-memory"
-          bind:this={memoryGallery}
-          bind:clientHeight={viewport.height}
-          bind:clientWidth={viewport.width}
-        >
-          <GalleryViewer assets={currentMemory.assets} {viewport} bind:selectedAssets />
-        </div>
-      </IntersectionObserver>
+        <GalleryViewer assets={currentMemory.assets} {viewport} bind:selectedAssets />
+      </div>
     </section>
   {/if}
 </section>
diff --git a/web/src/lib/components/photos-page/asset-date-group.svelte b/web/src/lib/components/photos-page/asset-date-group.svelte
index dd57160fb4..5ca29967fe 100644
--- a/web/src/lib/components/photos-page/asset-date-group.svelte
+++ b/web/src/lib/components/photos-page/asset-date-group.svelte
@@ -1,84 +1,69 @@
 <script lang="ts">
+  import { intersectionObserver } from '$lib/actions/intersection-observer';
   import Icon from '$lib/components/elements/icon.svelte';
+  import Skeleton from '$lib/components/photos-page/skeleton.svelte';
   import type { AssetInteractionStore } from '$lib/stores/asset-interaction.store';
-  import { assetViewingStore } from '$lib/stores/asset-viewing.store';
-  import type { AssetStore, Viewport } from '$lib/stores/assets.store';
-  import { locale } from '$lib/stores/preferences.store';
-  import { getAssetRatio } from '$lib/utils/asset-utils';
-  import {
-    calculateWidth,
-    formatGroupTitle,
-    fromLocalDateTime,
-    splitBucketIntoDateGroups,
-  } from '$lib/utils/timeline-util';
+  import { AssetBucket, type AssetStore, type Viewport } from '$lib/stores/assets.store';
+  import { navigate } from '$lib/utils/navigation';
+  import { findTotalOffset, type DateGroup, type ScrollTargetListener } from '$lib/utils/timeline-util';
   import type { AssetResponseDto } from '@immich/sdk';
   import { mdiCheckCircle, mdiCircleOutline } from '@mdi/js';
-  import justifiedLayout from 'justified-layout';
-  import { createEventDispatcher } from 'svelte';
+  import { createEventDispatcher, onDestroy } from 'svelte';
   import { fly } from 'svelte/transition';
   import Thumbnail from '../assets/thumbnail/thumbnail.svelte';
+  import { TUNABLES } from '$lib/utils/tunables';
+  import { generateId } from '$lib/utils/generate-id';
 
-  export let assets: AssetResponseDto[];
-  export let bucketDate: string;
-  export let bucketHeight: number;
+  export let element: HTMLElement | undefined = undefined;
   export let isSelectionMode = false;
   export let viewport: Viewport;
   export let singleSelect = false;
   export let withStacked = false;
   export let showArchiveIcon = false;
-
+  export let assetGridElement: HTMLElement | undefined = undefined;
+  export let renderThumbsAtBottomMargin: string | undefined = undefined;
+  export let renderThumbsAtTopMargin: string | undefined = undefined;
   export let assetStore: AssetStore;
+  export let bucket: AssetBucket;
   export let assetInteractionStore: AssetInteractionStore;
 
+  export let onScrollTarget: ScrollTargetListener | undefined = undefined;
+  export let onAssetInGrid: ((asset: AssetResponseDto) => void) | undefined = undefined;
+
+  const componentId = generateId();
+  $: bucketDate = bucket.bucketDate;
+  $: dateGroups = bucket.dateGroups;
+
+  const {
+    DATEGROUP: { INTERSECTION_DISABLED, INTERSECTION_ROOT_TOP, INTERSECTION_ROOT_BOTTOM },
+  } = TUNABLES;
+  /* TODO figure out a way to calculate this*/
+  const TITLE_HEIGHT = 51;
+
   const { selectedGroup, selectedAssets, assetSelectionCandidates, isMultiSelectState } = assetInteractionStore;
   const dispatch = createEventDispatcher<{
     select: { title: string; assets: AssetResponseDto[] };
     selectAssets: AssetResponseDto;
     selectAssetCandidates: AssetResponseDto | null;
-    shift: { heightDelta: number };
   }>();
 
   let isMouseOverGroup = false;
-  let actualBucketHeight: number;
   let hoveredDateGroup = '';
 
-  $: assetsGroupByDate = splitBucketIntoDateGroups(assets, $locale);
-
-  $: geometry = (() => {
-    const geometry = [];
-    for (let group of assetsGroupByDate) {
-      const justifiedLayoutResult = justifiedLayout(
-        group.map((assetGroup) => getAssetRatio(assetGroup)),
-        {
-          boxSpacing: 2,
-          containerWidth: Math.floor(viewport.width),
-          containerPadding: 0,
-          targetRowHeightTolerance: 0.15,
-          targetRowHeight: 235,
-        },
-      );
-      geometry.push({
-        ...justifiedLayoutResult,
-        containerWidth: calculateWidth(justifiedLayoutResult.boxes),
-      });
+  const onClick = (assets: AssetResponseDto[], groupTitle: string, asset: AssetResponseDto) => {
+    if (isSelectionMode || $isMultiSelectState) {
+      assetSelectHandler(asset, assets, groupTitle);
+      return;
     }
-    return geometry;
-  })();
+    void navigate({ targetRoute: 'current', assetId: asset.id });
+  };
 
-  $: {
-    if (actualBucketHeight && actualBucketHeight !== 0 && actualBucketHeight != bucketHeight) {
-      const heightDelta = assetStore.updateBucket(bucketDate, actualBucketHeight);
-      if (heightDelta !== 0) {
-        scrollTimeline(heightDelta);
-      }
+  const onRetrieveElement = (dateGroup: DateGroup, asset: AssetResponseDto, element: HTMLElement) => {
+    if (assetGridElement && onScrollTarget) {
+      const offset = findTotalOffset(element, assetGridElement) - TITLE_HEIGHT;
+      onScrollTarget({ bucket, dateGroup, asset, offset });
     }
-  }
-
-  function scrollTimeline(heightDelta: number) {
-    dispatch('shift', {
-      heightDelta,
-    });
-  }
+  };
 
   const handleSelectGroup = (title: string, assets: AssetResponseDto[]) => dispatch('select', { title, assets });
 
@@ -104,93 +89,149 @@
       dispatch('selectAssetCandidates', asset);
     }
   };
+
+  onDestroy(() => {
+    $assetStore.taskManager.removeAllTasksForComponent(componentId);
+  });
 </script>
 
-<section id="asset-group-by-date" class="flex flex-wrap gap-x-12" bind:clientHeight={actualBucketHeight}>
-  {#each assetsGroupByDate as groupAssets, groupIndex (groupAssets[0].id)}
-    {@const asset = groupAssets[0]}
-    {@const groupTitle = formatGroupTitle(fromLocalDateTime(asset.localDateTime).startOf('day'))}
-    <!-- Asset Group By Date -->
+<section id="asset-group-by-date" class="flex flex-wrap gap-x-12" data-bucket-date={bucketDate} bind:this={element}>
+  {#each dateGroups as dateGroup, groupIndex (dateGroup.date)}
+    {@const display =
+      dateGroup.intersecting || !!dateGroup.assets.some((asset) => asset.id === $assetStore.pendingScrollAssetId)}
 
-    <!-- svelte-ignore a11y-no-static-element-interactions -->
     <div
-      class="flex flex-col"
-      on:mouseenter={() => {
-        isMouseOverGroup = true;
-        assetMouseEventHandler(groupTitle, null);
-      }}
-      on:mouseleave={() => {
-        isMouseOverGroup = false;
-        assetMouseEventHandler(groupTitle, null);
+      id="date-group"
+      use:intersectionObserver={{
+        onIntersect: () => {
+          $assetStore.taskManager.intersectedDateGroup(componentId, dateGroup, () =>
+            assetStore.updateBucketDateGroup(bucket, dateGroup, { intersecting: true }),
+          );
+        },
+        onSeparate: () => {
+          $assetStore.taskManager.seperatedDateGroup(componentId, dateGroup, () =>
+            assetStore.updateBucketDateGroup(bucket, dateGroup, { intersecting: false }),
+          );
+        },
+        top: INTERSECTION_ROOT_TOP,
+        bottom: INTERSECTION_ROOT_BOTTOM,
+        root: assetGridElement,
+        disabled: INTERSECTION_DISABLED,
       }}
+      data-display={display}
+      data-date-group={dateGroup.date}
+      style:height={dateGroup.height + 'px'}
+      style:width={dateGroup.geometry.containerWidth + 'px'}
+      style:overflow={'clip'}
     >
-      <!-- Date group title -->
-      <div
-        class="flex z-[100] sticky top-0 pt-7 pb-5 h-6 place-items-center text-xs font-medium text-immich-fg bg-immich-bg dark:bg-immich-dark-bg dark:text-immich-dark-fg md:text-sm"
-        style="width: {geometry[groupIndex].containerWidth}px"
-      >
-        {#if !singleSelect && ((hoveredDateGroup == groupTitle && isMouseOverGroup) || $selectedGroup.has(groupTitle))}
+      {#if !display}
+        <Skeleton height={dateGroup.height + 'px'} title={dateGroup.groupTitle} />
+      {/if}
+      {#if display}
+        <!-- Asset Group By Date -->
+        <!-- svelte-ignore a11y-no-static-element-interactions -->
+        <div
+          on:mouseenter={() =>
+            $assetStore.taskManager.queueScrollSensitiveTask({
+              componentId,
+              task: () => {
+                isMouseOverGroup = true;
+                assetMouseEventHandler(dateGroup.groupTitle, null);
+              },
+            })}
+          on:mouseleave={() => {
+            $assetStore.taskManager.queueScrollSensitiveTask({
+              componentId,
+              task: () => {
+                isMouseOverGroup = false;
+                assetMouseEventHandler(dateGroup.groupTitle, null);
+              },
+            });
+          }}
+        >
+          <!-- Date group title -->
           <div
-            transition:fly={{ x: -24, duration: 200, opacity: 0.5 }}
-            class="inline-block px-2 hover:cursor-pointer"
-            on:click={() => handleSelectGroup(groupTitle, groupAssets)}
-            on:keydown={() => handleSelectGroup(groupTitle, groupAssets)}
+            class="flex z-[100] sticky top-[-1px] pt-[calc(1.75rem+1px)] pb-5 h-6 place-items-center text-xs font-medium text-immich-fg bg-immich-bg dark:bg-immich-dark-bg dark:text-immich-dark-fg md:text-sm"
+            style:width={dateGroup.geometry.containerWidth + 'px'}
           >
-            {#if $selectedGroup.has(groupTitle)}
-              <Icon path={mdiCheckCircle} size="24" color="#4250af" />
-            {:else}
-              <Icon path={mdiCircleOutline} size="24" color="#757575" />
+            {#if !singleSelect && ((hoveredDateGroup == dateGroup.groupTitle && isMouseOverGroup) || $selectedGroup.has(dateGroup.groupTitle))}
+              <div
+                transition:fly={{ x: -24, duration: 200, opacity: 0.5 }}
+                class="inline-block px-2 hover:cursor-pointer"
+                on:click={() => handleSelectGroup(dateGroup.groupTitle, dateGroup.assets)}
+                on:keydown={() => handleSelectGroup(dateGroup.groupTitle, dateGroup.assets)}
+              >
+                {#if $selectedGroup.has(dateGroup.groupTitle)}
+                  <Icon path={mdiCheckCircle} size="24" color="#4250af" />
+                {:else}
+                  <Icon path={mdiCircleOutline} size="24" color="#757575" />
+                {/if}
+              </div>
             {/if}
+
+            <span class="w-full truncate first-letter:capitalize" title={dateGroup.groupTitle}>
+              {dateGroup.groupTitle}
+            </span>
           </div>
-        {/if}
 
-        <span class="w-full truncate first-letter:capitalize" title={groupTitle}>
-          {groupTitle}
-        </span>
-      </div>
-
-      <!-- Image grid -->
-      <div
-        class="relative"
-        style="height: {geometry[groupIndex].containerHeight}px;width: {geometry[groupIndex].containerWidth}px"
-      >
-        {#each groupAssets as asset, index (asset.id)}
-          {@const box = geometry[groupIndex].boxes[index]}
+          <!-- Image grid -->
           <div
-            class="absolute"
-            style="width: {box.width}px; height: {box.height}px; top: {box.top}px; left: {box.left}px"
+            class="relative overflow-clip"
+            style:height={dateGroup.geometry.containerHeight + 'px'}
+            style:width={dateGroup.geometry.containerWidth + 'px'}
           >
-            <Thumbnail
-              showStackedIcon={withStacked}
-              {showArchiveIcon}
-              {asset}
-              {groupIndex}
-              onClick={(asset, event) => {
-                if (isSelectionMode || $isMultiSelectState) {
-                  event.preventDefault();
-                  assetSelectHandler(asset, groupAssets, groupTitle);
-                  return;
-                }
-
-                assetViewingStore.setAsset(asset);
-              }}
-              on:select={() => assetSelectHandler(asset, groupAssets, groupTitle)}
-              on:mouse-event={() => assetMouseEventHandler(groupTitle, asset)}
-              selected={$selectedAssets.has(asset) || $assetStore.albumAssets.has(asset.id)}
-              selectionCandidate={$assetSelectionCandidates.has(asset)}
-              disabled={$assetStore.albumAssets.has(asset.id)}
-              thumbnailWidth={box.width}
-              thumbnailHeight={box.height}
-            />
+            {#each dateGroup.assets as asset, index (asset.id)}
+              {@const box = dateGroup.geometry.boxes[index]}
+              <!-- update ASSET_GRID_PADDING-->
+              <div
+                use:intersectionObserver={{
+                  onIntersect: () => onAssetInGrid?.(asset),
+                  top: `-${TITLE_HEIGHT}px`,
+                  bottom: `-${viewport.height - TITLE_HEIGHT - 1}px`,
+                  right: `-${viewport.width - 1}px`,
+                  root: assetGridElement,
+                }}
+                data-asset-id={asset.id}
+                class="absolute"
+                style:width={box.width + 'px'}
+                style:height={box.height + 'px'}
+                style:top={box.top + 'px'}
+                style:left={box.left + 'px'}
+              >
+                <Thumbnail
+                  {dateGroup}
+                  {assetStore}
+                  intersectionConfig={{
+                    root: assetGridElement,
+                    bottom: renderThumbsAtBottomMargin,
+                    top: renderThumbsAtTopMargin,
+                  }}
+                  retrieveElement={$assetStore.pendingScrollAssetId === asset.id}
+                  onRetrieveElement={(element) => onRetrieveElement(dateGroup, asset, element)}
+                  showStackedIcon={withStacked}
+                  {showArchiveIcon}
+                  {asset}
+                  {groupIndex}
+                  onClick={(asset) => onClick(dateGroup.assets, dateGroup.groupTitle, asset)}
+                  onSelect={(asset) => assetSelectHandler(asset, dateGroup.assets, dateGroup.groupTitle)}
+                  onMouseEvent={() => assetMouseEventHandler(dateGroup.groupTitle, asset)}
+                  selected={$selectedAssets.has(asset) || $assetStore.albumAssets.has(asset.id)}
+                  selectionCandidate={$assetSelectionCandidates.has(asset)}
+                  disabled={$assetStore.albumAssets.has(asset.id)}
+                  thumbnailWidth={box.width}
+                  thumbnailHeight={box.height}
+                />
+              </div>
+            {/each}
           </div>
-        {/each}
-      </div>
+        </div>
+      {/if}
     </div>
   {/each}
 </section>
 
 <style>
   #asset-group-by-date {
-    contain: layout;
+    contain: layout paint style;
   }
 </style>
diff --git a/web/src/lib/components/photos-page/asset-grid.svelte b/web/src/lib/components/photos-page/asset-grid.svelte
index 3e0935d938..db030ed14c 100644
--- a/web/src/lib/components/photos-page/asset-grid.svelte
+++ b/web/src/lib/components/photos-page/asset-grid.svelte
@@ -1,11 +1,17 @@
 <script lang="ts">
-  import { goto } from '$app/navigation';
+  import { afterNavigate, beforeNavigate, goto } from '$app/navigation';
   import { shortcuts, type ShortcutOptions } from '$lib/actions/shortcut';
   import type { Action } from '$lib/components/asset-viewer/actions/action';
   import { AppRoute, AssetAction } from '$lib/constants';
   import type { AssetInteractionStore } from '$lib/stores/asset-interaction.store';
   import { assetViewingStore } from '$lib/stores/asset-viewing.store';
-  import { BucketPosition, isSelectingAllAssets, type AssetStore, type Viewport } from '$lib/stores/assets.store';
+  import {
+    AssetBucket,
+    AssetStore,
+    isSelectingAllAssets,
+    type BucketListener,
+    type ViewportXY,
+  } from '$lib/stores/assets.store';
   import { locale, showDeleteModal } from '$lib/stores/preferences.store';
   import { isSearchEnabled } from '$lib/stores/search.store';
   import { featureFlags } from '$lib/stores/server-config.store';
@@ -13,19 +19,38 @@
   import { deleteAssets } from '$lib/utils/actions';
   import { archiveAssets, selectAllAssets, stackAssets } from '$lib/utils/asset-utils';
   import { navigate } from '$lib/utils/navigation';
-  import { formatGroupTitle, splitBucketIntoDateGroups } from '$lib/utils/timeline-util';
+  import {
+    formatGroupTitle,
+    splitBucketIntoDateGroups,
+    type ScrubberListener,
+    type ScrollTargetListener,
+  } from '$lib/utils/timeline-util';
+  import { TUNABLES } from '$lib/utils/tunables';
   import type { AlbumResponseDto, AssetResponseDto } from '@immich/sdk';
-  import { DateTime } from 'luxon';
+  import { throttle } from 'lodash-es';
   import { createEventDispatcher, onDestroy, onMount } from 'svelte';
-  import IntersectionObserver from '../asset-viewer/intersection-observer.svelte';
   import Portal from '../shared-components/portal/portal.svelte';
-  import Scrollbar from '../shared-components/scrollbar/scrollbar.svelte';
+  import Scrubber from '../shared-components/scrubber/scrubber.svelte';
   import ShowShortcuts from '../shared-components/show-shortcuts.svelte';
   import AssetDateGroup from './asset-date-group.svelte';
   import DeleteAssetDialog from './delete-asset-dialog.svelte';
 
+  import { resizeObserver } from '$lib/actions/resize-observer';
+  import MeasureDateGroup from '$lib/components/photos-page/measure-date-group.svelte';
+  import { intersectionObserver } from '$lib/actions/intersection-observer';
+  import Skeleton from '$lib/components/photos-page/skeleton.svelte';
+  import { page } from '$app/stores';
+  import type { UpdatePayload } from 'vite';
+  import { generateId } from '$lib/utils/generate-id';
+
   export let isSelectionMode = false;
   export let singleSelect = false;
+
+  /** `true` if this asset grid is responds to navigation events; if `true`, then look at the
+   `AssetViewingStore.gridScrollTarget` and load and scroll to the asset specified, and
+   additionally, update the page location/url with the asset as the asset-grid is scrolled */
+  export let enableRouting: boolean;
+
   export let assetStore: AssetStore;
   export let assetInteractionStore: AssetInteractionStore;
   export let removeAction:
@@ -40,17 +65,32 @@
   export let album: AlbumResponseDto | null = null;
   export let isShowDeleteConfirmation = false;
 
-  $: isTrashEnabled = $featureFlags.loaded && $featureFlags.trash;
-
+  let { isViewing: showAssetViewer, asset: viewingAsset, preloadAssets, gridScrollTarget } = assetViewingStore;
   const { assetSelectionCandidates, assetSelectionStart, selectedGroup, selectedAssets, isMultiSelectState } =
     assetInteractionStore;
-  const viewport: Viewport = { width: 0, height: 0 };
-  let { isViewing: showAssetViewer, asset: viewingAsset, preloadAssets } = assetViewingStore;
+
+  const viewport: ViewportXY = { width: 0, height: 0, x: 0, y: 0 };
+  const safeViewport: ViewportXY = { width: 0, height: 0, x: 0, y: 0 };
+
+  const componentId = generateId();
   let element: HTMLElement;
+  let timelineElement: HTMLElement;
   let showShortcuts = false;
   let showSkeleton = true;
+  let internalScroll = false;
+  let navigating = false;
+  let preMeasure: AssetBucket[] = [];
+  let lastIntersectedBucketDate: string | undefined;
+  let scrubBucketPercent = 0;
+  let scrubBucket: { bucketDate: string | undefined } | undefined;
+  let scrubOverallPercent: number = 0;
+  let topSectionHeight = 0;
+  let topSectionOffset = 0;
+  // 60 is the bottom spacer element at 60px
+  let bottomSectionHeight = 60;
+  let leadout = false;
 
-  $: timelineY = element?.scrollTop || 0;
+  $: isTrashEnabled = $featureFlags.loaded && $featureFlags.trash;
   $: isEmpty = $assetStore.initialized && $assetStore.buckets.length === 0;
   $: idsSelectedAssets = [...$selectedAssets].map(({ id }) => id);
   $: isAllArchived = [...$selectedAssets].every((asset) => asset.isArchived);
@@ -59,30 +99,329 @@
       assetInteractionStore.clearMultiselect();
     }
   }
-
   $: {
-    void assetStore.updateViewport(viewport);
+    if (element && isViewportOrigin()) {
+      const rect = element.getBoundingClientRect();
+      viewport.height = rect.height;
+      viewport.width = rect.width;
+      viewport.x = rect.x;
+      viewport.y = rect.y;
+    }
+    if (!isViewportOrigin() && !isEqual(viewport, safeViewport)) {
+      safeViewport.height = viewport.height;
+      safeViewport.width = viewport.width;
+      safeViewport.x = viewport.x;
+      safeViewport.y = viewport.y;
+      updateViewport();
+    }
   }
+  const {
+    ASSET_GRID: { NAVIGATE_ON_ASSET_IN_VIEW },
+    BUCKET: {
+      INTERSECTION_ROOT_TOP: BUCKET_INTERSECTION_ROOT_TOP,
+      INTERSECTION_ROOT_BOTTOM: BUCKET_INTERSECTION_ROOT_BOTTOM,
+    },
+    THUMBNAIL: {
+      INTERSECTION_ROOT_TOP: THUMBNAIL_INTERSECTION_ROOT_TOP,
+      INTERSECTION_ROOT_BOTTOM: THUMBNAIL_INTERSECTION_ROOT_BOTTOM,
+    },
+  } = TUNABLES;
 
   const dispatch = createEventDispatcher<{ select: AssetResponseDto; escape: void }>();
 
-  onMount(async () => {
-    showSkeleton = false;
-    assetStore.connect();
-    await assetStore.init(viewport);
-  });
+  const isViewportOrigin = () => {
+    return viewport.height === 0 && viewport.width === 0;
+  };
 
-  onDestroy(() => {
-    if ($showAssetViewer) {
-      $showAssetViewer = false;
+  const isEqual = (a: ViewportXY, b: ViewportXY) => {
+    return a.height == b.height && a.width == b.width && a.x === b.x && a.y === b.y;
+  };
+
+  const completeNav = () => {
+    navigating = false;
+    if (internalScroll) {
+      internalScroll = false;
+      return;
     }
 
-    assetStore.disconnect();
+    if ($gridScrollTarget?.at) {
+      void $assetStore.scheduleScrollToAssetId($gridScrollTarget, () => {
+        element.scrollTo({ top: 0 });
+        showSkeleton = false;
+      });
+    } else {
+      element.scrollTo({ top: 0 });
+      showSkeleton = false;
+    }
+  };
+
+  afterNavigate((nav) => {
+    const { complete, type } = nav;
+    if (type === 'enter') {
+      return;
+    }
+    complete.then(completeNav, completeNav);
   });
 
+  beforeNavigate(() => {
+    navigating = true;
+  });
+
+  const hmrSupport = () => {
+    // when hmr happens, skeleton is initialized to true by default
+    // normally, loading asset-grid is part of a navigation event, and the completion of
+    // that event triggers a scroll-to-asset, if necessary, when then clears the skeleton.
+    // this handler will run the navigation/scroll-to-asset handler when hmr is performed,
+    // preventing skeleton from showing after hmr
+    if (import.meta && import.meta.hot) {
+      const afterApdate = (payload: UpdatePayload) => {
+        const assetGridUpdate = payload.updates.some(
+          (update) => update.path.endsWith('asset-grid.svelte') || update.path.endsWith('assets-store.ts'),
+        );
+
+        if (assetGridUpdate) {
+          setTimeout(() => {
+            void $assetStore.updateViewport(safeViewport, true);
+            const asset = $page.url.searchParams.get('at');
+            if (asset) {
+              $gridScrollTarget = { at: asset };
+              void navigate(
+                { targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget },
+                { replaceState: true, forceNavigate: true },
+              );
+            } else {
+              element.scrollTo({ top: 0 });
+              showSkeleton = false;
+            }
+          }, 500);
+        }
+      };
+      import.meta.hot?.on('vite:afterUpdate', afterApdate);
+      import.meta.hot?.on('vite:beforeUpdate', (payload) => {
+        const assetGridUpdate = payload.updates.some((update) => update.path.endsWith('asset-grid.svelte'));
+        if (assetGridUpdate) {
+          assetStore.destroy();
+        }
+      });
+
+      return () => import.meta.hot?.off('vite:afterUpdate', afterApdate);
+    }
+    return () => void 0;
+  };
+
+  const _updateLastIntersectedBucketDate = () => {
+    let elem = document.elementFromPoint(safeViewport.x + 1, safeViewport.y + 1);
+
+    while (elem != null) {
+      if (elem.id === 'bucket') {
+        break;
+      }
+      elem = elem.parentElement;
+    }
+    if (elem) {
+      lastIntersectedBucketDate = (elem as HTMLElement).dataset.bucketDate;
+    }
+  };
+  const updateLastIntersectedBucketDate = throttle(_updateLastIntersectedBucketDate, 16, {
+    leading: false,
+    trailing: true,
+  });
+
+  const scrollTolastIntersectedBucket = (adjustedBucket: AssetBucket, delta: number) => {
+    if (!lastIntersectedBucketDate) {
+      _updateLastIntersectedBucketDate();
+    }
+    if (lastIntersectedBucketDate) {
+      const currentIndex = $assetStore.buckets.findIndex((b) => b.bucketDate === lastIntersectedBucketDate);
+      const deltaIndex = $assetStore.buckets.indexOf(adjustedBucket);
+
+      if (deltaIndex < currentIndex) {
+        element?.scrollBy(0, delta);
+      }
+    }
+  };
+
+  const bucketListener: BucketListener = (event) => {
+    const { type } = event;
+    if (type === 'bucket-height') {
+      const { bucket, delta } = event;
+      scrollTolastIntersectedBucket(bucket, delta);
+    }
+  };
+
+  onMount(() => {
+    void $assetStore
+      .init({ bucketListener })
+      .then(() => ($assetStore.connect(), $assetStore.updateViewport(safeViewport)));
+    if (!enableRouting) {
+      showSkeleton = false;
+    }
+    const dispose = hmrSupport();
+    return () => {
+      $assetStore.disconnect();
+      $assetStore.destroy();
+      dispose();
+    };
+  });
+
+  function getOffset(bucketDate: string) {
+    let offset = 0;
+    for (let a = 0; a < assetStore.buckets.length; a++) {
+      if (assetStore.buckets[a].bucketDate === bucketDate) {
+        break;
+      }
+      offset += assetStore.buckets[a].bucketHeight;
+    }
+    return offset;
+  }
+  const _updateViewport = () => void $assetStore.updateViewport(safeViewport);
+  const updateViewport = throttle(_updateViewport, 16);
+
+  const getMaxScrollPercent = () =>
+    ($assetStore.timelineHeight + bottomSectionHeight + topSectionHeight - safeViewport.height) /
+    ($assetStore.timelineHeight + bottomSectionHeight + topSectionHeight);
+
+  const getMaxScroll = () =>
+    topSectionHeight + bottomSectionHeight + (timelineElement.clientHeight - element.clientHeight);
+
+  const scrollToBucketAndOffset = (bucket: AssetBucket, bucketScrollPercent: number) => {
+    const topOffset = getOffset(bucket.bucketDate) + topSectionHeight + topSectionOffset;
+    const maxScrollPercent = getMaxScrollPercent();
+    const delta = bucket.bucketHeight * bucketScrollPercent;
+    const scrollTop = (topOffset + delta) * maxScrollPercent;
+    element.scrollTop = scrollTop;
+  };
+
+  const _onScrub: ScrubberListener = (
+    bucketDate: string | undefined,
+    scrollPercent: number,
+    bucketScrollPercent: number,
+  ) => {
+    if (!bucketDate || $assetStore.timelineHeight < safeViewport.height * 2) {
+      // edge case - scroll limited due to size of content, must adjust - use use the overall percent instead
+
+      const maxScroll = getMaxScroll();
+      const offset = maxScroll * scrollPercent;
+      element.scrollTop = offset;
+    } else {
+      const bucket = assetStore.buckets.find((b) => b.bucketDate === bucketDate);
+      if (!bucket) {
+        return;
+      }
+      scrollToBucketAndOffset(bucket, bucketScrollPercent);
+    }
+  };
+  const onScrub = throttle(_onScrub, 16, { leading: false, trailing: true });
+
+  const stopScrub: ScrubberListener = async (
+    bucketDate: string | undefined,
+    _scrollPercent: number,
+    bucketScrollPercent: number,
+  ) => {
+    if (!bucketDate || $assetStore.timelineHeight < safeViewport.height * 2) {
+      // edge case - scroll limited due to size of content, must adjust - use use the overall percent instead
+      return;
+    }
+    const bucket = assetStore.buckets.find((b) => b.bucketDate === bucketDate);
+    if (!bucket) {
+      return;
+    }
+    if (bucket && !bucket.measured) {
+      preMeasure.push(bucket);
+      if (!bucket.loaded) {
+        await assetStore.loadBucket(bucket.bucketDate);
+      }
+      // Wait here, and collect the deltas that are above offset, which affect offset position
+      await bucket.measuredPromise;
+      scrollToBucketAndOffset(bucket, bucketScrollPercent);
+    }
+  };
+
+  const _handleTimelineScroll = () => {
+    leadout = false;
+    if ($assetStore.timelineHeight < safeViewport.height * 2) {
+      // edge case - scroll limited due to size of content, must adjust -  use the overall percent instead
+      const maxScroll = getMaxScroll();
+      scrubOverallPercent = Math.min(1, element.scrollTop / maxScroll);
+
+      scrubBucket = undefined;
+      scrubBucketPercent = 0;
+    } else {
+      let top = element?.scrollTop;
+      if (top < topSectionHeight) {
+        // in the lead-in area
+        scrubBucket = undefined;
+        scrubBucketPercent = 0;
+        const maxScroll = getMaxScroll();
+
+        scrubOverallPercent = Math.min(1, element.scrollTop / maxScroll);
+        return;
+      }
+
+      let maxScrollPercent = getMaxScrollPercent();
+      let found = false;
+
+      // create virtual buckets....
+      const vbuckets = [
+        { bucketHeight: topSectionHeight, bucketDate: undefined },
+        ...assetStore.buckets,
+        { bucketHeight: bottomSectionHeight, bucketDate: undefined },
+      ];
+
+      for (const bucket of vbuckets) {
+        let next = top - bucket.bucketHeight * maxScrollPercent;
+        if (next < 0) {
+          scrubBucket = bucket;
+          scrubBucketPercent = top / (bucket.bucketHeight * maxScrollPercent);
+          found = true;
+          break;
+        }
+        top = next;
+      }
+      if (!found) {
+        leadout = true;
+        scrubBucket = undefined;
+        scrubBucketPercent = 0;
+        scrubOverallPercent = 1;
+      }
+    }
+  };
+  const handleTimelineScroll = throttle(_handleTimelineScroll, 16, { leading: false, trailing: true });
+
+  const _onAssetInGrid = async (asset: AssetResponseDto) => {
+    if (!enableRouting || navigating || internalScroll) {
+      return;
+    }
+    $gridScrollTarget = { at: asset.id };
+    internalScroll = true;
+    await navigate(
+      { targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget },
+      { replaceState: true, forceNavigate: true },
+    );
+  };
+  const onAssetInGrid = NAVIGATE_ON_ASSET_IN_VIEW
+    ? throttle(_onAssetInGrid, 16, { leading: false, trailing: true })
+    : () => void 0;
+
+  const onScrollTarget: ScrollTargetListener = ({ bucket, offset }) => {
+    element.scrollTo({ top: offset });
+    if (!bucket.measured) {
+      preMeasure.push(bucket);
+    }
+    showSkeleton = false;
+    $assetStore.clearPendingScroll();
+    // set intersecting true manually here, to reduce flicker that happens when
+    // clearing pending scroll, but the intersection observer hadn't yet had time to run
+    $assetStore.updateBucket(bucket.bucketDate, { intersecting: true });
+  };
+
   const trashOrDelete = async (force: boolean = false) => {
     isShowDeleteConfirmation = false;
-    await deleteAssets(!(isTrashEnabled && !force), (assetIds) => assetStore.removeAssets(assetIds), idsSelectedAssets);
+    await deleteAssets(
+      !(isTrashEnabled && !force),
+      (assetIds) => $assetStore.removeAssets(assetIds),
+      idsSelectedAssets,
+    );
     assetInteractionStore.clearMultiselect();
   };
 
@@ -107,7 +446,7 @@
   const onStackAssets = async () => {
     const ids = await stackAssets(Array.from($selectedAssets));
     if (ids) {
-      assetStore.removeAssets(ids);
+      $assetStore.removeAssets(ids);
       dispatch('escape');
     }
   };
@@ -115,7 +454,7 @@
   const toggleArchive = async () => {
     const ids = await archiveAssets(Array.from($selectedAssets), !isAllArchived);
     if (ids) {
-      assetStore.removeAssets(ids);
+      $assetStore.removeAssets(ids);
       deselectAllAssets();
     }
   };
@@ -135,7 +474,7 @@
       { shortcut: { key: 'Escape' }, onShortcut: () => dispatch('escape') },
       { shortcut: { key: '?', shift: true }, onShortcut: () => (showShortcuts = !showShortcuts) },
       { shortcut: { key: '/' }, onShortcut: () => goto(AppRoute.EXPLORE) },
-      { shortcut: { key: 'A', ctrl: true }, onShortcut: () => selectAllAssets(assetStore, assetInteractionStore) },
+      { shortcut: { key: 'A', ctrl: true }, onShortcut: () => selectAllAssets($assetStore, assetInteractionStore) },
       { shortcut: { key: 'PageDown' }, preventDefault: false, onShortcut: focusElement },
       { shortcut: { key: 'PageUp' }, preventDefault: false, onShortcut: focusElement },
     ];
@@ -154,29 +493,33 @@
   })();
 
   const handleSelectAsset = (asset: AssetResponseDto) => {
-    if (!assetStore.albumAssets.has(asset.id)) {
+    if (!$assetStore.albumAssets.has(asset.id)) {
       assetInteractionStore.selectAsset(asset);
     }
   };
 
-  async function intersectedHandler(event: CustomEvent) {
-    const element_ = event.detail.container as HTMLElement;
-    const target = element_.firstChild as HTMLElement;
-    if (target) {
-      const bucketDate = target.id.split('_')[1];
-      await assetStore.loadBucket(bucketDate, event.detail.position);
-    }
+  function intersectedHandler(bucket: AssetBucket) {
+    updateLastIntersectedBucketDate();
+    const intersectedTask = () => {
+      $assetStore.updateBucket(bucket.bucketDate, { intersecting: true });
+      void $assetStore.loadBucket(bucket.bucketDate);
+    };
+    $assetStore.taskManager.intersectedBucket(componentId, bucket, intersectedTask);
   }
 
-  function handleScrollTimeline(event: CustomEvent) {
-    element.scrollBy(0, event.detail.heightDelta);
+  function seperatedHandler(bucket: AssetBucket) {
+    const seperatedTask = () => {
+      $assetStore.updateBucket(bucket.bucketDate, { intersecting: false });
+      bucket.cancel();
+    };
+    $assetStore.taskManager.seperatedBucket(componentId, bucket, seperatedTask);
   }
 
   const handlePrevious = async () => {
-    const previousAsset = await assetStore.getPreviousAsset($viewingAsset);
+    const previousAsset = await $assetStore.getPreviousAsset($viewingAsset);
 
     if (previousAsset) {
-      const preloadAsset = await assetStore.getPreviousAsset(previousAsset);
+      const preloadAsset = await $assetStore.getPreviousAsset(previousAsset);
       assetViewingStore.setAsset(previousAsset, preloadAsset ? [preloadAsset] : []);
       await navigate({ targetRoute: 'current', assetId: previousAsset.id });
     }
@@ -185,10 +528,10 @@
   };
 
   const handleNext = async () => {
-    const nextAsset = await assetStore.getNextAsset($viewingAsset);
+    const nextAsset = await $assetStore.getNextAsset($viewingAsset);
 
     if (nextAsset) {
-      const preloadAsset = await assetStore.getNextAsset(nextAsset);
+      const preloadAsset = await $assetStore.getNextAsset(nextAsset);
       assetViewingStore.setAsset(nextAsset, preloadAsset ? [preloadAsset] : []);
       await navigate({ targetRoute: 'current', assetId: nextAsset.id });
     }
@@ -196,7 +539,12 @@
     return !!nextAsset;
   };
 
-  const handleClose = () => assetViewingStore.showAssetViewer(false);
+  const handleClose = async ({ detail: { asset } }: { detail: { asset: AssetResponseDto } }) => {
+    assetViewingStore.showAssetViewer(false);
+    showSkeleton = true;
+    $gridScrollTarget = { at: asset.id };
+    await navigate({ targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget });
+  };
 
   const handleAction = async (action: Action) => {
     switch (action.type) {
@@ -206,7 +554,7 @@
       case AssetAction.DELETE: {
         // find the next asset to show or close the viewer
         // eslint-disable-next-line @typescript-eslint/no-unused-expressions
-        (await handleNext()) || (await handlePrevious()) || handleClose();
+        (await handleNext()) || (await handlePrevious()) || (await handleClose({ detail: { asset: action.asset } }));
 
         // delete after find the next one
         assetStore.removeAssets([action.asset.id]);
@@ -232,20 +580,6 @@
     }
   };
 
-  let animationTick = false;
-
-  const handleTimelineScroll = () => {
-    if (animationTick) {
-      return;
-    }
-
-    animationTick = true;
-    window.requestAnimationFrame(() => {
-      timelineY = element?.scrollTop || 0;
-      animationTick = false;
-    });
-  };
-
   let lastAssetMouseEvent: AssetResponseDto | null = null;
 
   $: if (!lastAssetMouseEvent) {
@@ -355,7 +689,7 @@
       // Select/deselect assets in all intermediate buckets
       for (let bucketIndex = startBucketIndex + 1; bucketIndex < endBucketIndex; bucketIndex++) {
         const bucket = $assetStore.buckets[bucketIndex];
-        await assetStore.loadBucket(bucket.bucketDate, BucketPosition.Unknown);
+        await $assetStore.loadBucket(bucket.bucketDate);
         for (const asset of bucket.assets) {
           if (deselect) {
             assetInteractionStore.removeAssetFromMultiselectGroup(asset);
@@ -370,11 +704,10 @@
         const bucket = $assetStore.buckets[bucketIndex];
 
         // Split bucket into date groups and check each group
-        const assetsGroupByDate = splitBucketIntoDateGroups(bucket.assets, $locale);
-
+        const assetsGroupByDate = splitBucketIntoDateGroups(bucket, $locale);
         for (const dateGroup of assetsGroupByDate) {
-          const dateGroupTitle = formatGroupTitle(DateTime.fromISO(dateGroup[0].fileCreatedAt).startOf('day'));
-          if (dateGroup.every((a) => $selectedAssets.has(a))) {
+          const dateGroupTitle = formatGroupTitle(dateGroup.date);
+          if (dateGroup.assets.every((a) => $selectedAssets.has(a))) {
             assetInteractionStore.addGroupToMultiselectGroup(dateGroupTitle);
           } else {
             assetInteractionStore.removeGroupFromMultiselectGroup(dateGroupTitle);
@@ -411,6 +744,9 @@
       e.preventDefault();
     }
   };
+  onDestroy(() => {
+    assetStore.taskManager.removeAllTasksForComponent(componentId);
+  });
 </script>
 
 <svelte:window on:keydown={onKeyDown} on:keyup={onKeyUp} on:selectstart={onSelectStart} use:shortcuts={shortcutList} />
@@ -427,78 +763,97 @@
   <ShowShortcuts on:close={() => (showShortcuts = !showShortcuts)} />
 {/if}
 
-<Scrollbar
+<Scrubber
+  invisible={showSkeleton}
   {assetStore}
-  height={viewport.height}
-  {timelineY}
-  on:scrollTimeline={({ detail }) => (element.scrollTop = detail)}
+  height={safeViewport.height}
+  timelineTopOffset={topSectionHeight}
+  timelineBottomOffset={bottomSectionHeight}
+  {leadout}
+  {scrubOverallPercent}
+  {scrubBucketPercent}
+  {scrubBucket}
+  {onScrub}
+  {stopScrub}
 />
 
 <!-- Right margin MUST be equal to the width of immich-scrubbable-scrollbar -->
 <section
   id="asset-grid"
-  class="scrollbar-hidden h-full overflow-y-auto outline-none pb-[60px] {isEmpty
-    ? 'm-0'
-    : 'ml-4 tall:ml-0 md:mr-[60px]'}"
+  class="scrollbar-hidden h-full overflow-y-auto outline-none {isEmpty ? 'm-0' : 'ml-4 tall:ml-0 mr-[60px]'}"
   tabindex="-1"
-  bind:clientHeight={viewport.height}
-  bind:clientWidth={viewport.width}
+  use:resizeObserver={({ height, width }) => ((viewport.width = width), (viewport.height = height))}
   bind:this={element}
-  on:scroll={handleTimelineScroll}
+  on:scroll={() => ((assetStore.lastScrollTime = Date.now()), handleTimelineScroll())}
 >
-  <!-- skeleton -->
-  {#if showSkeleton}
-    <div class="mt-8 animate-pulse">
-      <div class="mb-2 h-4 w-24 rounded-full bg-immich-primary/20 dark:bg-immich-dark-primary/20" />
-      <div class="flex w-[120%] flex-wrap">
-        {#each Array.from({ length: 100 }) as _}
-          <div class="m-[1px] h-[10em] w-[16em] bg-immich-primary/20 dark:bg-immich-dark-primary/20" />
-        {/each}
-      </div>
-    </div>
-  {/if}
-
-  {#if element}
+  <section
+    use:resizeObserver={({ target, height }) => ((topSectionHeight = height), (topSectionOffset = target.offsetTop))}
+    class:invisible={showSkeleton}
+  >
     <slot />
-
-    <!-- (optional) empty placeholder -->
     {#if isEmpty}
+      <!-- (optional) empty placeholder -->
       <slot name="empty" />
     {/if}
-    <section id="virtual-timeline" style:height={$assetStore.timelineHeight + 'px'}>
-      {#each $assetStore.buckets as bucket (bucket.bucketDate)}
-        <IntersectionObserver
-          on:intersected={intersectedHandler}
-          on:hidden={() => assetStore.cancelBucket(bucket)}
-          let:intersecting
-          top={750}
-          bottom={750}
-          root={element}
-        >
-          <div id={'bucket_' + bucket.bucketDate} style:height={bucket.bucketHeight + 'px'}>
-            {#if intersecting}
-              <AssetDateGroup
-                {withStacked}
-                {showArchiveIcon}
-                {assetStore}
-                {assetInteractionStore}
-                {isSelectionMode}
-                {singleSelect}
-                on:select={({ detail: group }) => handleGroupSelect(group.title, group.assets)}
-                on:shift={handleScrollTimeline}
-                on:selectAssetCandidates={({ detail: asset }) => handleSelectAssetCandidates(asset)}
-                on:selectAssets={({ detail: asset }) => handleSelectAssets(asset)}
-                assets={bucket.assets}
-                bucketDate={bucket.bucketDate}
-                bucketHeight={bucket.bucketHeight}
-                {viewport}
-              />
-            {/if}
-          </div>
-        </IntersectionObserver>
-      {/each}
-    </section>
-  {/if}
+  </section>
+
+  <section
+    bind:this={timelineElement}
+    id="virtual-timeline"
+    class:invisible={showSkeleton}
+    style:height={$assetStore.timelineHeight + 'px'}
+  >
+    {#each $assetStore.buckets as bucket (bucket.bucketDate)}
+      {@const isPremeasure = preMeasure.includes(bucket)}
+      {@const display = bucket.intersecting || bucket === $assetStore.pendingScrollBucket || isPremeasure}
+      <div
+        id="bucket"
+        use:intersectionObserver={{
+          onIntersect: () => intersectedHandler(bucket),
+          onSeparate: () => seperatedHandler(bucket),
+          top: BUCKET_INTERSECTION_ROOT_TOP,
+          bottom: BUCKET_INTERSECTION_ROOT_BOTTOM,
+          root: element,
+        }}
+        data-bucket-display={bucket.intersecting}
+        data-bucket-date={bucket.bucketDate}
+        style:height={bucket.bucketHeight + 'px'}
+      >
+        {#if display && !bucket.measured}
+          <MeasureDateGroup
+            {bucket}
+            {assetStore}
+            onMeasured={() => (preMeasure = preMeasure.filter((b) => b !== bucket))}
+          ></MeasureDateGroup>
+        {/if}
+
+        {#if !display || !bucket.measured}
+          <Skeleton height={bucket.bucketHeight + 'px'} title={`${bucket.bucketDateFormattted}`} />
+        {/if}
+        {#if display && bucket.measured}
+          <AssetDateGroup
+            assetGridElement={element}
+            renderThumbsAtTopMargin={THUMBNAIL_INTERSECTION_ROOT_TOP}
+            renderThumbsAtBottomMargin={THUMBNAIL_INTERSECTION_ROOT_BOTTOM}
+            {withStacked}
+            {showArchiveIcon}
+            {assetStore}
+            {assetInteractionStore}
+            {isSelectionMode}
+            {singleSelect}
+            {onScrollTarget}
+            {onAssetInGrid}
+            {bucket}
+            viewport={safeViewport}
+            on:select={({ detail: group }) => handleGroupSelect(group.title, group.assets)}
+            on:selectAssetCandidates={({ detail: asset }) => handleSelectAssetCandidates(asset)}
+            on:selectAssets={({ detail: asset }) => handleSelectAssets(asset)}
+          />
+        {/if}
+      </div>
+    {/each}
+    <div class="h-[60px]"></div>
+  </section>
 </section>
 
 <Portal target="body">
@@ -522,7 +877,7 @@
 
 <style>
   #asset-grid {
-    contain: layout;
+    contain: strict;
     scrollbar-width: none;
   }
 </style>
diff --git a/web/src/lib/components/photos-page/measure-date-group.svelte b/web/src/lib/components/photos-page/measure-date-group.svelte
new file mode 100644
index 0000000000..98e423ae94
--- /dev/null
+++ b/web/src/lib/components/photos-page/measure-date-group.svelte
@@ -0,0 +1,89 @@
+<script lang="ts" context="module">
+  const recentTimes: number[] = [];
+  // TODO: track average time to measure, and use this to populate TUNABLES.ASSETS_STORE.CHECK_INTERVAL_MS
+  // eslint-disable-next-line @typescript-eslint/no-unused-vars
+  function adjustTunables(avg: number) {}
+  function addMeasure(time: number) {
+    recentTimes.push(time);
+    if (recentTimes.length > 10) {
+      recentTimes.shift();
+    }
+    const sum = recentTimes.reduce((acc: number, val: number) => {
+      return acc + val;
+    }, 0);
+    const avg = sum / recentTimes.length;
+    adjustTunables(avg);
+  }
+</script>
+
+<script lang="ts">
+  import { resizeObserver } from '$lib/actions/resize-observer';
+  import type { AssetBucket, AssetStore, BucketListener } from '$lib/stores/assets.store';
+
+  export let assetStore: AssetStore;
+  export let bucket: AssetBucket;
+  export let onMeasured: () => void;
+
+  async function _measure(element: Element) {
+    try {
+      await bucket.complete;
+      const t1 = Date.now();
+      let heightPending = bucket.dateGroups.some((group) => !group.heightActual);
+      if (heightPending) {
+        const listener: BucketListener = (event) => {
+          const { type } = event;
+          if (type === 'height') {
+            const { bucket: changedBucket } = event;
+            if (changedBucket === bucket && type === 'height') {
+              heightPending = bucket.dateGroups.some((group) => !group.heightActual);
+              if (!heightPending) {
+                const height = element.getBoundingClientRect().height;
+                if (height !== 0) {
+                  $assetStore.updateBucket(bucket.bucketDate, { height: height, measured: true });
+                }
+
+                onMeasured();
+                $assetStore.removeListener(listener);
+                const t2 = Date.now();
+
+                addMeasure((t2 - t1) / bucket.bucketCount);
+              }
+            }
+          }
+        };
+        assetStore.addListener(listener);
+      }
+    } catch {
+      // ignore if complete rejects (canceled load)
+    }
+  }
+  function measure(element: Element) {
+    void _measure(element);
+  }
+</script>
+
+<section id="measure-asset-group-by-date" class="flex flex-wrap gap-x-12" use:measure>
+  {#each bucket.dateGroups as dateGroup}
+    <div id="date-group" data-date-group={dateGroup.date}>
+      <div
+        use:resizeObserver={({ height }) => $assetStore.updateBucketDateGroup(bucket, dateGroup, { height: height })}
+      >
+        <div
+          class="flex z-[100] sticky top-[-1px] pt-7 pb-5 h-6 place-items-center text-xs font-medium text-immich-fg bg-immich-bg dark:bg-immich-dark-bg dark:text-immich-dark-fg md:text-sm"
+          style:width={dateGroup.geometry.containerWidth + 'px'}
+        >
+          <span class="w-full truncate first-letter:capitalize">
+            {dateGroup.groupTitle}
+          </span>
+        </div>
+
+        <div
+          class="relative overflow-clip"
+          style:height={dateGroup.geometry.containerHeight + 'px'}
+          style:width={dateGroup.geometry.containerWidth + 'px'}
+          style:visibility={'hidden'}
+        ></div>
+      </div>
+    </div>
+  {/each}
+</section>
diff --git a/web/src/lib/components/photos-page/memory-lane.svelte b/web/src/lib/components/photos-page/memory-lane.svelte
index 43c2958944..5bc55796ae 100644
--- a/web/src/lib/components/photos-page/memory-lane.svelte
+++ b/web/src/lib/components/photos-page/memory-lane.svelte
@@ -1,4 +1,5 @@
 <script lang="ts">
+  import { resizeObserver } from '$lib/actions/resize-observer';
   import Icon from '$lib/components/elements/icon.svelte';
   import { AppRoute, QueryParameter } from '$lib/constants';
   import { memoryStore } from '$lib/stores/memory.store';
@@ -38,7 +39,7 @@
     id="memory-lane"
     bind:this={memoryLaneElement}
     class="relative mt-5 overflow-x-hidden whitespace-nowrap transition-all"
-    bind:offsetWidth
+    use:resizeObserver={({ width }) => (offsetWidth = width)}
     on:scroll={onScroll}
   >
     {#if canScrollLeft || canScrollRight}
@@ -67,7 +68,7 @@
         {/if}
       </div>
     {/if}
-    <div class="inline-block" bind:offsetWidth={innerWidth}>
+    <div class="inline-block" use:resizeObserver={({ width }) => (innerWidth = width)}>
       {#each $memoryStore as memory, index (memory.yearsAgo)}
         {#if memory.assets.length > 0}
           <a
diff --git a/web/src/lib/components/photos-page/skeleton.svelte b/web/src/lib/components/photos-page/skeleton.svelte
new file mode 100644
index 0000000000..07836eb4db
--- /dev/null
+++ b/web/src/lib/components/photos-page/skeleton.svelte
@@ -0,0 +1,35 @@
+<script lang="ts">
+  export let title: string | null = null;
+  export let height: string | null = null;
+</script>
+
+<div class="overflow-clip" style={`height: ${height}`}>
+  {#if title}
+    <div
+      class="flex z-[100] sticky top-0 pt-7 pb-5 h-6 place-items-center text-xs font-medium text-immich-fg bg-immich-bg dark:bg-immich-dark-bg dark:text-immich-dark-fg md:text-sm"
+    >
+      <span class="w-full truncate first-letter:capitalize">{title}</span>
+    </div>
+  {/if}
+  <div id="skeleton" style={`height: ${height}`}></div>
+</div>
+
+<style>
+  #skeleton {
+    background-image: url('/light_skeleton.png');
+    background-repeat: repeat;
+    background-size: 235px, 235px;
+  }
+  :global(.dark) #skeleton {
+    background-image: url('/dark_skeleton.png');
+  }
+  @keyframes delayedVisibility {
+    to {
+      visibility: visible;
+    }
+  }
+  #skeleton {
+    visibility: hidden;
+    animation: 0s linear 0.1s forwards delayedVisibility;
+  }
+</style>
diff --git a/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte b/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte
index f977d91a99..c7b49f6012 100644
--- a/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte
+++ b/web/src/lib/components/shared-components/gallery-viewer/gallery-viewer.svelte
@@ -4,25 +4,25 @@
   import Thumbnail from '$lib/components/assets/thumbnail/thumbnail.svelte';
   import { AppRoute, AssetAction } from '$lib/constants';
   import { assetViewingStore } from '$lib/stores/asset-viewing.store';
-  import type { BucketPosition, Viewport } from '$lib/stores/assets.store';
+  import type { Viewport } from '$lib/stores/assets.store';
   import { getAssetRatio } from '$lib/utils/asset-utils';
   import { handleError } from '$lib/utils/handle-error';
   import { navigate } from '$lib/utils/navigation';
   import { calculateWidth } from '$lib/utils/timeline-util';
   import { type AssetResponseDto } from '@immich/sdk';
   import justifiedLayout from 'justified-layout';
-  import { createEventDispatcher, onDestroy } from 'svelte';
+  import { onDestroy } from 'svelte';
   import { t } from 'svelte-i18n';
   import AssetViewer from '../../asset-viewer/asset-viewer.svelte';
   import Portal from '../portal/portal.svelte';
-
-  const dispatch = createEventDispatcher<{ intersected: { container: HTMLDivElement; position: BucketPosition } }>();
+  import { handlePromiseError } from '$lib/utils';
 
   export let assets: AssetResponseDto[];
   export let selectedAssets: Set<AssetResponseDto> = new Set();
   export let disableAssetSelect = false;
   export let showArchiveIcon = false;
   export let viewport: Viewport;
+  export let onIntersected: (() => void) | undefined = undefined;
   export let showAssetName = false;
 
   let { isViewing: isViewerOpen, asset: viewingAsset, setAsset } = assetViewingStore;
@@ -127,18 +127,15 @@
         <Thumbnail
           {asset}
           readonly={disableAssetSelect}
-          onClick={async (asset, e) => {
-            e.preventDefault();
-
+          onClick={(asset) => {
             if (isMultiSelectionMode) {
               selectAssetHandler(asset);
               return;
             }
-            await viewAssetHandler(asset);
+            void viewAssetHandler(asset);
           }}
-          on:select={(e) => selectAssetHandler(e.detail.asset)}
-          on:intersected={(event) =>
-            i === Math.max(1, assets.length - 7) ? dispatch('intersected', event.detail) : undefined}
+          onSelect={(asset) => selectAssetHandler(asset)}
+          onIntersected={() => (i === Math.max(1, assets.length - 7) ? onIntersected?.() : void 0)}
           selected={selectedAssets.has(asset)}
           {showArchiveIcon}
           thumbnailWidth={geometry.boxes[i].width}
@@ -159,6 +156,15 @@
 <!-- Overlay Asset Viewer -->
 {#if $isViewerOpen}
   <Portal target="body">
-    <AssetViewer asset={$viewingAsset} onAction={handleAction} on:previous={handlePrevious} on:next={handleNext} />
+    <AssetViewer
+      asset={$viewingAsset}
+      onAction={handleAction}
+      on:previous={handlePrevious}
+      on:next={handleNext}
+      on:close={() => {
+        assetViewingStore.showAssetViewer(false);
+        handlePromiseError(navigate({ targetRoute: 'current', assetId: null }));
+      }}
+    />
   </Portal>
 {/if}
diff --git a/web/src/lib/components/shared-components/scrollbar/scrollbar.svelte b/web/src/lib/components/shared-components/scrollbar/scrollbar.svelte
deleted file mode 100644
index 9282c760c2..0000000000
--- a/web/src/lib/components/shared-components/scrollbar/scrollbar.svelte
+++ /dev/null
@@ -1,183 +0,0 @@
-<script lang="ts">
-  import type { AssetStore, AssetBucket } from '$lib/stores/assets.store';
-  import type { DateTime } from 'luxon';
-  import { fromLocalDateTime } from '$lib/utils/timeline-util';
-  import { createEventDispatcher } from 'svelte';
-  import { clamp } from 'lodash-es';
-  import { locale } from '$lib/stores/preferences.store';
-
-  export let timelineY = 0;
-  export let height = 0;
-  export let assetStore: AssetStore;
-
-  let isHover = false;
-  let isDragging = false;
-  let isAnimating = false;
-  let hoverLabel = '';
-  let hoverY = 0;
-  let clientY = 0;
-  let windowHeight = 0;
-  let scrollBar: HTMLElement | undefined;
-
-  const toScrollY = (timelineY: number) => (timelineY / $assetStore.timelineHeight) * height;
-  const toTimelineY = (scrollY: number) => Math.round((scrollY * $assetStore.timelineHeight) / height);
-
-  const HOVER_DATE_HEIGHT = 30;
-  const MIN_YEAR_LABEL_DISTANCE = 16;
-
-  $: {
-    hoverY = clamp(height - windowHeight + clientY, 0, height);
-    if (scrollBar) {
-      const rect = scrollBar.getBoundingClientRect();
-      const x = rect.left + rect.width / 2;
-      const y = rect.top + Math.min(hoverY, height - 1);
-      updateLabel(x, y);
-    }
-  }
-
-  $: scrollY = toScrollY(timelineY);
-
-  class Segment {
-    public count = 0;
-    public height = 0;
-    public timeGroup = '';
-    public date!: DateTime;
-    public hasLabel = false;
-  }
-
-  const calculateSegments = (buckets: AssetBucket[]) => {
-    let height = 0;
-    let previous: Segment;
-    return buckets.map((bucket) => {
-      const segment = new Segment();
-      segment.count = bucket.assets.length;
-      segment.height = toScrollY(bucket.bucketHeight);
-      segment.timeGroup = bucket.bucketDate;
-      segment.date = fromLocalDateTime(segment.timeGroup);
-
-      if (previous?.date.year !== segment.date.year && height > MIN_YEAR_LABEL_DISTANCE) {
-        previous.hasLabel = true;
-        height = 0;
-      }
-
-      height += segment.height;
-      previous = segment;
-      return segment;
-    });
-  };
-
-  $: segments = calculateSegments($assetStore.buckets);
-
-  const dispatch = createEventDispatcher<{ scrollTimeline: number }>();
-  const scrollTimeline = () => dispatch('scrollTimeline', toTimelineY(hoverY));
-
-  const updateLabel = (cursorX: number, cursorY: number) => {
-    const segment = document.elementsFromPoint(cursorX, cursorY).find(({ id }) => id === 'time-segment');
-    if (!segment) {
-      return;
-    }
-    const attr = (segment as HTMLElement).dataset.date;
-    if (!attr) {
-      return;
-    }
-    hoverLabel = new Date(attr).toLocaleString($locale, {
-      month: 'short',
-      year: 'numeric',
-      timeZone: 'UTC',
-    });
-  };
-
-  const handleMouseEvent = (event: { clientY: number; isDragging?: boolean }) => {
-    const wasDragging = isDragging;
-
-    isDragging = event.isDragging ?? isDragging;
-    clientY = event.clientY;
-
-    if (wasDragging === false && isDragging) {
-      scrollTimeline();
-    }
-
-    if (!isDragging || isAnimating) {
-      return;
-    }
-
-    isAnimating = true;
-
-    window.requestAnimationFrame(() => {
-      scrollTimeline();
-      isAnimating = false;
-    });
-  };
-</script>
-
-<svelte:window
-  bind:innerHeight={windowHeight}
-  on:mousemove={({ clientY }) => (isDragging || isHover) && handleMouseEvent({ clientY })}
-  on:mousedown={({ clientY }) => isHover && handleMouseEvent({ clientY, isDragging: true })}
-  on:mouseup={({ clientY }) => handleMouseEvent({ clientY, isDragging: false })}
-/>
-
-<!-- svelte-ignore a11y-no-static-element-interactions -->
-
-{#if $assetStore.timelineHeight > height}
-  <div
-    id="immich-scrubbable-scrollbar"
-    class="absolute right-0 z-[1] select-none bg-immich-bg hover:cursor-row-resize"
-    style:width={isDragging ? '100vw' : '60px'}
-    style:height={height + 'px'}
-    style:background-color={isDragging ? 'transparent' : 'transparent'}
-    draggable="false"
-    bind:this={scrollBar}
-    on:mouseenter={() => (isHover = true)}
-    on:mouseleave={() => (isHover = false)}
-  >
-    {#if isHover || isDragging}
-      <div
-        id="time-label"
-        class="pointer-events-none absolute right-0 z-[100] min-w-24 w-fit whitespace-nowrap rounded-tl-md border-b-2 border-immich-primary bg-immich-bg py-1 px-1 text-sm font-medium shadow-[0_0_8px_rgba(0,0,0,0.25)] dark:border-immich-dark-primary dark:bg-immich-dark-gray dark:text-immich-dark-fg"
-        style:top="{clamp(hoverY - HOVER_DATE_HEIGHT, 0, height - HOVER_DATE_HEIGHT - 2)}px"
-      >
-        {hoverLabel}
-      </div>
-    {/if}
-
-    <!-- Scroll Position Indicator Line -->
-    {#if !isDragging}
-      <div
-        class="absolute right-0 h-[2px] w-10 bg-immich-primary dark:bg-immich-dark-primary"
-        style:top="{scrollY}px"
-      />
-    {/if}
-    <!-- Time Segment -->
-    {#each segments as segment}
-      <div
-        id="time-segment"
-        class="relative"
-        data-date={segment.date}
-        style:height={segment.height + 'px'}
-        aria-label={segment.timeGroup + ' ' + segment.count}
-      >
-        {#if segment.hasLabel}
-          <div
-            aria-label={segment.timeGroup + ' ' + segment.count}
-            class="absolute right-0 bottom-0 z-10 pr-5 text-[12px] dark:text-immich-dark-fg font-immich-mono"
-          >
-            {segment.date.year}
-          </div>
-        {:else if segment.height > 5}
-          <div
-            aria-label={segment.timeGroup + ' ' + segment.count}
-            class="absolute right-0 mr-3 block h-[4px] w-[4px] rounded-full bg-gray-300"
-          />
-        {/if}
-      </div>
-    {/each}
-  </div>
-{/if}
-
-<style>
-  #immich-scrubbable-scrollbar,
-  #time-segment {
-    contain: layout;
-  }
-</style>
diff --git a/web/src/lib/components/shared-components/scrubber/scrubber.svelte b/web/src/lib/components/shared-components/scrubber/scrubber.svelte
new file mode 100644
index 0000000000..e2cc638650
--- /dev/null
+++ b/web/src/lib/components/shared-components/scrubber/scrubber.svelte
@@ -0,0 +1,281 @@
+<script lang="ts">
+  import type { AssetStore, AssetBucket, BucketListener } from '$lib/stores/assets.store';
+  import type { DateTime } from 'luxon';
+  import { fromLocalDateTime, type ScrubberListener } from '$lib/utils/timeline-util';
+  import { clamp } from 'lodash-es';
+  import { onMount } from 'svelte';
+
+  export let timelineTopOffset = 0;
+  export let timelineBottomOffset = 0;
+  export let height = 0;
+  export let assetStore: AssetStore;
+  export let invisible = false;
+  export let scrubOverallPercent: number = 0;
+  export let scrubBucketPercent: number = 0;
+  export let scrubBucket: { bucketDate: string | undefined } | undefined = undefined;
+  export let leadout: boolean = false;
+  export let onScrub: ScrubberListener | undefined = undefined;
+  export let startScrub: ScrubberListener | undefined = undefined;
+  export let stopScrub: ScrubberListener | undefined = undefined;
+
+  let isHover = false;
+  let isDragging = false;
+  let hoverLabel: string | undefined;
+  let bucketDate: string | undefined;
+  let hoverY = 0;
+  let clientY = 0;
+  let windowHeight = 0;
+  let scrollBar: HTMLElement | undefined;
+  let segments: Segment[] = [];
+
+  const toScrollY = (percent: number) => percent * (height - HOVER_DATE_HEIGHT * 2);
+  const toTimelineY = (scrollY: number) => scrollY / (height - HOVER_DATE_HEIGHT * 2);
+
+  const HOVER_DATE_HEIGHT = 31.75;
+  const MIN_YEAR_LABEL_DISTANCE = 16;
+  const MIN_DOT_DISTANCE = 8;
+
+  const toScrollFromBucketPercentage = (
+    scrubBucket: { bucketDate: string | undefined } | undefined,
+    scrubBucketPercent: number,
+    scrubOverallPercent: number,
+  ) => {
+    if (scrubBucket) {
+      let offset = relativeTopOffset;
+      let match = false;
+      for (const segment of segments) {
+        if (segment.bucketDate === scrubBucket.bucketDate) {
+          offset += scrubBucketPercent * segment.height;
+          match = true;
+          break;
+        }
+        offset += segment.height;
+      }
+      if (!match) {
+        offset += scrubBucketPercent * relativeBottomOffset;
+      }
+      // 2px is the height of the indicator
+      return offset - 2;
+    } else if (leadout) {
+      let offset = relativeTopOffset;
+      for (const segment of segments) {
+        offset += segment.height;
+      }
+      offset += scrubOverallPercent * relativeBottomOffset;
+      return offset - 2;
+    } else {
+      // 2px is the height of the indicator
+      return scrubOverallPercent * (height - HOVER_DATE_HEIGHT * 2) - 2;
+    }
+  };
+  $: scrollY = toScrollFromBucketPercentage(scrubBucket, scrubBucketPercent, scrubOverallPercent);
+  $: timelineFullHeight = $assetStore.timelineHeight + timelineTopOffset + timelineBottomOffset;
+  $: relativeTopOffset = toScrollY(timelineTopOffset / timelineFullHeight);
+  $: relativeBottomOffset = toScrollY(timelineBottomOffset / timelineFullHeight);
+
+  const listener: BucketListener = (event) => {
+    const { type } = event;
+    if (type === 'viewport') {
+      segments = calculateSegments($assetStore.buckets);
+      scrollY = toScrollFromBucketPercentage(scrubBucket, scrubBucketPercent, scrubOverallPercent);
+    }
+  };
+
+  onMount(() => {
+    assetStore.addListener(listener);
+    return () => assetStore.removeListener(listener);
+  });
+
+  type Segment = {
+    count: number;
+    height: number;
+    dateFormatted: string;
+    bucketDate: string;
+    date: DateTime;
+    hasLabel: boolean;
+    hasDot: boolean;
+  };
+
+  const calculateSegments = (buckets: AssetBucket[]) => {
+    let height = 0;
+    let dotHeight = 0;
+
+    let segments: Segment[] = [];
+    let previousLabeledSegment: Segment | undefined;
+
+    for (const [i, bucket] of buckets.entries()) {
+      const scrollBarPercentage =
+        bucket.bucketHeight / ($assetStore.timelineHeight + timelineTopOffset + timelineBottomOffset);
+
+      const segment = {
+        count: bucket.assets.length,
+        height: toScrollY(scrollBarPercentage),
+        bucketDate: bucket.bucketDate,
+        date: fromLocalDateTime(bucket.bucketDate),
+        dateFormatted: bucket.bucketDateFormattted,
+        hasLabel: false,
+        hasDot: false,
+      };
+
+      if (i === 0) {
+        segment.hasDot = true;
+        segment.hasLabel = true;
+        previousLabeledSegment = segment;
+      } else {
+        if (previousLabeledSegment?.date?.year !== segment.date.year && height > MIN_YEAR_LABEL_DISTANCE) {
+          height = 0;
+          segment.hasLabel = true;
+          previousLabeledSegment = segment;
+        }
+        if (i !== 1 && segment.height > 5 && dotHeight > MIN_DOT_DISTANCE) {
+          segment.hasDot = true;
+          dotHeight = 0;
+        }
+
+        height += segment.height;
+        dotHeight += segment.height;
+      }
+      segments.push(segment);
+    }
+
+    hoverLabel = segments[0]?.dateFormatted;
+    return segments;
+  };
+
+  const updateLabel = (segment: HTMLElement) => {
+    hoverLabel = segment.dataset.label;
+    bucketDate = segment.dataset.timeSegmentBucketDate;
+  };
+
+  const handleMouseEvent = (event: { clientY: number; isDragging?: boolean }) => {
+    const wasDragging = isDragging;
+
+    isDragging = event.isDragging ?? isDragging;
+    clientY = event.clientY;
+
+    if (!scrollBar) {
+      return;
+    }
+
+    const rect = scrollBar.getBoundingClientRect()!;
+    const lower = 0;
+    const upper = rect?.height - HOVER_DATE_HEIGHT * 2;
+    hoverY = clamp(clientY - rect?.top - HOVER_DATE_HEIGHT, lower, upper);
+    const x = rect!.left + rect!.width / 2;
+    const elems = document.elementsFromPoint(x, clientY);
+    const segment = elems.find(({ id }) => id === 'time-segment');
+    let bucketPercentY = 0;
+    if (segment) {
+      updateLabel(segment as HTMLElement);
+      const sr = segment.getBoundingClientRect();
+      const sy = sr.y;
+      const relativeY = clientY - sy;
+      bucketPercentY = relativeY / sr.height;
+    } else {
+      const leadin = elems.find(({ id }) => id === 'lead-in');
+      if (leadin) {
+        updateLabel(leadin as HTMLElement);
+      } else {
+        bucketDate = undefined;
+        bucketPercentY = 0;
+      }
+    }
+
+    const scrollPercent = toTimelineY(hoverY);
+    if (wasDragging === false && isDragging) {
+      void startScrub?.(bucketDate, scrollPercent, bucketPercentY);
+      void onScrub?.(bucketDate, scrollPercent, bucketPercentY);
+    }
+
+    if (wasDragging && !isDragging) {
+      void stopScrub?.(bucketDate, scrollPercent, bucketPercentY);
+      return;
+    }
+
+    if (!isDragging) {
+      return;
+    }
+
+    void onScrub?.(bucketDate, scrollPercent, bucketPercentY);
+  };
+</script>
+
+<svelte:window
+  bind:innerHeight={windowHeight}
+  on:mousemove={({ clientY }) => (isDragging || isHover) && handleMouseEvent({ clientY })}
+  on:mousedown={({ clientY }) => isHover && handleMouseEvent({ clientY, isDragging: true })}
+  on:mouseup={({ clientY }) => handleMouseEvent({ clientY, isDragging: false })}
+/>
+
+<!-- svelte-ignore a11y-no-static-element-interactions -->
+
+<div
+  id="immich-scrubbable-scrollbar"
+  class={`absolute right-0 z-[1] select-none bg-immich-bg hover:cursor-row-resize`}
+  style:padding-top={HOVER_DATE_HEIGHT + 'px'}
+  style:padding-bottom={HOVER_DATE_HEIGHT + 'px'}
+  class:invisible
+  style:width={isDragging ? '100vw' : '60px'}
+  style:height={height + 'px'}
+  style:background-color={isDragging ? 'transparent' : 'transparent'}
+  draggable="false"
+  bind:this={scrollBar}
+  on:mouseenter={() => (isHover = true)}
+  on:mouseleave={() => (isHover = false)}
+>
+  {#if hoverLabel && (isHover || isDragging)}
+    <div
+      id="time-label"
+      class="truncate opacity-85 pointer-events-none absolute right-0 z-[100] min-w-20 max-w-64 w-fit rounded-tl-md border-b-2 border-immich-primary bg-immich-bg py-1 px-1 text-sm font-medium shadow-[0_0_8px_rgba(0,0,0,0.25)] dark:border-immich-dark-primary dark:bg-immich-dark-gray dark:text-immich-dark-fg"
+      style:top="{hoverY + 2}px"
+    >
+      {hoverLabel}
+    </div>
+  {/if}
+  <!-- Scroll Position Indicator Line -->
+  {#if !isDragging}
+    <div
+      class="absolute right-0 h-[2px] w-10 bg-immich-primary dark:bg-immich-dark-primary"
+      style:top="{scrollY + HOVER_DATE_HEIGHT}px"
+    />
+  {/if}
+  <div id="lead-in" class="relative" style:height={relativeTopOffset + 'px'} data-label={segments.at(0)?.dateFormatted}>
+    {#if relativeTopOffset > 6}
+      <div class="absolute right-[0.75rem] h-[4px] w-[4px] rounded-full bg-gray-300" />
+    {/if}
+  </div>
+  <!-- Time Segment -->
+  {#each segments as segment}
+    <div
+      id="time-segment"
+      class="relative"
+      data-time-segment-bucket-date={segment.date}
+      data-label={segment.dateFormatted}
+      style:height={segment.height + 'px'}
+      aria-label={segment.dateFormatted + ' ' + segment.count}
+    >
+      {#if segment.hasLabel}
+        <div
+          aria-label={segment.dateFormatted + ' ' + segment.count}
+          class="absolute right-[1.25rem] top-[-16px] z-10 text-[12px] dark:text-immich-dark-fg font-immich-mono"
+        >
+          {segment.date.year}
+        </div>
+      {/if}
+      {#if segment.hasDot}
+        <div
+          aria-label={segment.dateFormatted + ' ' + segment.count}
+          class="absolute right-[0.75rem] bottom-0 h-[4px] w-[4px] rounded-full bg-gray-300"
+        />
+      {/if}
+    </div>
+  {/each}
+  <div id="lead-out" class="relative" style:height={relativeBottomOffset + 'px'}></div>
+</div>
+
+<style>
+  #immich-scrubbable-scrollbar,
+  #time-segment {
+    contain: layout size style;
+  }
+</style>
diff --git a/web/src/lib/components/utilities-page/duplicates/duplicates-compare-control.svelte b/web/src/lib/components/utilities-page/duplicates/duplicates-compare-control.svelte
index fcf68fdb91..2f1efc487c 100644
--- a/web/src/lib/components/utilities-page/duplicates/duplicates-compare-control.svelte
+++ b/web/src/lib/components/utilities-page/duplicates/duplicates-compare-control.svelte
@@ -4,7 +4,8 @@
   import Portal from '$lib/components/shared-components/portal/portal.svelte';
   import DuplicateAsset from '$lib/components/utilities-page/duplicates/duplicate-asset.svelte';
   import { assetViewingStore } from '$lib/stores/asset-viewing.store';
-  import { suggestDuplicateByFileSize } from '$lib/utils';
+  import { handlePromiseError, suggestDuplicateByFileSize } from '$lib/utils';
+  import { navigate } from '$lib/utils/navigation';
   import { shortcuts } from '$lib/actions/shortcut';
   import { type AssetResponseDto } from '@immich/sdk';
   import { mdiCheck, mdiTrashCanOutline, mdiImageMultipleOutline } from '@mdi/js';
@@ -158,7 +159,10 @@
           const index = getAssetIndex($viewingAsset.id) - 1 + assets.length;
           setAsset(assets[index % assets.length]);
         }}
-        on:close={() => assetViewingStore.showAssetViewer(false)}
+        on:close={() => {
+          assetViewingStore.showAssetViewer(false);
+          handlePromiseError(navigate({ targetRoute: 'current', assetId: null }));
+        }}
       />
     </Portal>
   {/await}
diff --git a/web/src/lib/stores/asset-viewing.store.ts b/web/src/lib/stores/asset-viewing.store.ts
index cabe2e85a1..2e6e44511d 100644
--- a/web/src/lib/stores/asset-viewing.store.ts
+++ b/web/src/lib/stores/asset-viewing.store.ts
@@ -1,4 +1,5 @@
 import { getKey } from '$lib/utils';
+import { type AssetGridRouteSearchParams } from '$lib/utils/navigation';
 import { getAssetInfo, type AssetResponseDto } from '@immich/sdk';
 import { readonly, writable } from 'svelte/store';
 
@@ -6,6 +7,7 @@ function createAssetViewingStore() {
   const viewingAssetStoreState = writable<AssetResponseDto>();
   const preloadAssets = writable<AssetResponseDto[]>([]);
   const viewState = writable<boolean>(false);
+  const gridScrollTarget = writable<AssetGridRouteSearchParams | null | undefined>();
 
   const setAsset = (asset: AssetResponseDto, assetsToPreload: AssetResponseDto[] = []) => {
     preloadAssets.set(assetsToPreload);
@@ -26,6 +28,7 @@ function createAssetViewingStore() {
     asset: readonly(viewingAssetStoreState),
     preloadAssets: readonly(preloadAssets),
     isViewing: viewState,
+    gridScrollTarget,
     setAsset,
     setAssetId,
     showAssetViewer,
diff --git a/web/src/lib/stores/asset.store.spec.ts b/web/src/lib/stores/asset.store.spec.ts
index 3fd9e1e981..7787bf794d 100644
--- a/web/src/lib/stores/asset.store.spec.ts
+++ b/web/src/lib/stores/asset.store.spec.ts
@@ -2,7 +2,7 @@ import { sdkMock } from '$lib/__mocks__/sdk.mock';
 import { AbortError } from '$lib/utils';
 import { TimeBucketSize, type AssetResponseDto } from '@immich/sdk';
 import { assetFactory } from '@test-data/factories/asset-factory';
-import { AssetStore, BucketPosition } from './assets.store';
+import { AssetStore } from './assets.store';
 
 describe('AssetStore', () => {
   beforeEach(() => {
@@ -26,7 +26,8 @@ describe('AssetStore', () => {
       ]);
       sdkMock.getTimeBucket.mockImplementation(({ timeBucket }) => Promise.resolve(bucketAssets[timeBucket]));
 
-      await assetStore.init({ width: 1588, height: 1000 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 1588, height: 1000 });
     });
 
     it('should load buckets in viewport', () => {
@@ -38,15 +39,15 @@ describe('AssetStore', () => {
     it('calculates bucket height', () => {
       expect(assetStore.buckets).toEqual(
         expect.arrayContaining([
-          expect.objectContaining({ bucketDate: '2024-03-01T00:00:00.000Z', bucketHeight: 235 }),
-          expect.objectContaining({ bucketDate: '2024-02-01T00:00:00.000Z', bucketHeight: 3760 }),
-          expect.objectContaining({ bucketDate: '2024-01-01T00:00:00.000Z', bucketHeight: 235 }),
+          expect.objectContaining({ bucketDate: '2024-03-01T00:00:00.000Z', bucketHeight: 286 }),
+          expect.objectContaining({ bucketDate: '2024-02-01T00:00:00.000Z', bucketHeight: 3811 }),
+          expect.objectContaining({ bucketDate: '2024-01-01T00:00:00.000Z', bucketHeight: 286 }),
         ]),
       );
     });
 
     it('calculates timeline height', () => {
-      expect(assetStore.timelineHeight).toBe(4230);
+      expect(assetStore.timelineHeight).toBe(4383);
     });
   });
 
@@ -72,35 +73,28 @@ describe('AssetStore', () => {
 
         return bucketAssets[timeBucket];
       });
-      await assetStore.init({ width: 0, height: 0 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 0, height: 0 });
     });
 
     it('loads a bucket', async () => {
       expect(assetStore.getBucketByDate('2024-01-01T00:00:00.000Z')?.assets.length).toEqual(0);
-      await assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2024-01-01T00:00:00.000Z');
       expect(sdkMock.getTimeBucket).toBeCalledTimes(1);
       expect(assetStore.getBucketByDate('2024-01-01T00:00:00.000Z')?.assets.length).toEqual(3);
     });
 
     it('ignores invalid buckets', async () => {
-      await assetStore.loadBucket('2023-01-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2023-01-01T00:00:00.000Z');
       expect(sdkMock.getTimeBucket).toBeCalledTimes(0);
     });
 
-    it('only updates the position of loaded buckets', async () => {
-      await assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Unknown);
-      expect(assetStore.getBucketByDate('2024-01-01T00:00:00.000Z')?.position).toEqual(BucketPosition.Unknown);
-
-      await assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Visible);
-      expect(assetStore.getBucketByDate('2024-01-01T00:00:00.000Z')?.position).toEqual(BucketPosition.Visible);
-    });
-
     it('cancels bucket loading', async () => {
       const bucket = assetStore.getBucketByDate('2024-01-01T00:00:00.000Z');
-      const loadPromise = assetStore.loadBucket(bucket!.bucketDate, BucketPosition.Unknown);
+      const loadPromise = assetStore.loadBucket(bucket!.bucketDate);
 
       const abortSpy = vi.spyOn(bucket!.cancelToken!, 'abort');
-      assetStore.cancelBucket(bucket!);
+      bucket?.cancel();
       expect(abortSpy).toBeCalledTimes(1);
 
       await loadPromise;
@@ -109,24 +103,24 @@ describe('AssetStore', () => {
 
     it('prevents loading buckets multiple times', async () => {
       await Promise.all([
-        assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Unknown),
-        assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Unknown),
+        assetStore.loadBucket('2024-01-01T00:00:00.000Z'),
+        assetStore.loadBucket('2024-01-01T00:00:00.000Z'),
       ]);
       expect(sdkMock.getTimeBucket).toBeCalledTimes(1);
 
-      await assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Unknown);
+      await assetStore.loadBucket('2024-01-01T00:00:00.000Z');
       expect(sdkMock.getTimeBucket).toBeCalledTimes(1);
     });
 
     it('allows loading a canceled bucket', async () => {
       const bucket = assetStore.getBucketByDate('2024-01-01T00:00:00.000Z');
-      const loadPromise = assetStore.loadBucket(bucket!.bucketDate, BucketPosition.Unknown);
+      const loadPromise = assetStore.loadBucket(bucket!.bucketDate);
 
-      assetStore.cancelBucket(bucket!);
+      bucket?.cancel();
       await loadPromise;
       expect(bucket?.assets.length).toEqual(0);
 
-      await assetStore.loadBucket(bucket!.bucketDate, BucketPosition.Unknown);
+      await assetStore.loadBucket(bucket!.bucketDate);
       expect(bucket!.assets.length).toEqual(3);
     });
   });
@@ -137,7 +131,8 @@ describe('AssetStore', () => {
     beforeEach(async () => {
       assetStore = new AssetStore({});
       sdkMock.getTimeBuckets.mockResolvedValue([]);
-      await assetStore.init({ width: 1588, height: 1000 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 1588, height: 1000 });
     });
 
     it('is empty initially', () => {
@@ -219,7 +214,8 @@ describe('AssetStore', () => {
     beforeEach(async () => {
       assetStore = new AssetStore({});
       sdkMock.getTimeBuckets.mockResolvedValue([]);
-      await assetStore.init({ width: 1588, height: 1000 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 1588, height: 1000 });
     });
 
     it('ignores non-existing assets', () => {
@@ -263,7 +259,8 @@ describe('AssetStore', () => {
     beforeEach(async () => {
       assetStore = new AssetStore({});
       sdkMock.getTimeBuckets.mockResolvedValue([]);
-      await assetStore.init({ width: 1588, height: 1000 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 1588, height: 1000 });
     });
 
     it('ignores invalid IDs', () => {
@@ -312,7 +309,8 @@ describe('AssetStore', () => {
       ]);
       sdkMock.getTimeBucket.mockImplementation(({ timeBucket }) => Promise.resolve(bucketAssets[timeBucket]));
 
-      await assetStore.init({ width: 0, height: 0 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 0, height: 0 });
     });
 
     it('returns null for invalid assetId', async () => {
@@ -321,15 +319,15 @@ describe('AssetStore', () => {
     });
 
     it('returns previous assetId', async () => {
-      await assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2024-01-01T00:00:00.000Z');
       const bucket = assetStore.getBucketByDate('2024-01-01T00:00:00.000Z');
 
       expect(await assetStore.getPreviousAsset(bucket!.assets[1])).toEqual(bucket!.assets[0]);
     });
 
     it('returns previous assetId spanning multiple buckets', async () => {
-      await assetStore.loadBucket('2024-02-01T00:00:00.000Z', BucketPosition.Visible);
-      await assetStore.loadBucket('2024-03-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2024-02-01T00:00:00.000Z');
+      await assetStore.loadBucket('2024-03-01T00:00:00.000Z');
 
       const bucket = assetStore.getBucketByDate('2024-02-01T00:00:00.000Z');
       const previousBucket = assetStore.getBucketByDate('2024-03-01T00:00:00.000Z');
@@ -337,7 +335,7 @@ describe('AssetStore', () => {
     });
 
     it('loads previous bucket', async () => {
-      await assetStore.loadBucket('2024-02-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2024-02-01T00:00:00.000Z');
 
       const loadBucketSpy = vi.spyOn(assetStore, 'loadBucket');
       const bucket = assetStore.getBucketByDate('2024-02-01T00:00:00.000Z');
@@ -347,9 +345,9 @@ describe('AssetStore', () => {
     });
 
     it('skips removed assets', async () => {
-      await assetStore.loadBucket('2024-01-01T00:00:00.000Z', BucketPosition.Visible);
-      await assetStore.loadBucket('2024-02-01T00:00:00.000Z', BucketPosition.Visible);
-      await assetStore.loadBucket('2024-03-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2024-01-01T00:00:00.000Z');
+      await assetStore.loadBucket('2024-02-01T00:00:00.000Z');
+      await assetStore.loadBucket('2024-03-01T00:00:00.000Z');
 
       const [assetOne, assetTwo, assetThree] = assetStore.assets;
       assetStore.removeAssets([assetTwo.id]);
@@ -357,7 +355,7 @@ describe('AssetStore', () => {
     });
 
     it('returns null when no more assets', async () => {
-      await assetStore.loadBucket('2024-03-01T00:00:00.000Z', BucketPosition.Visible);
+      await assetStore.loadBucket('2024-03-01T00:00:00.000Z');
       expect(await assetStore.getPreviousAsset(assetStore.assets[0])).toBeNull();
     });
   });
@@ -368,7 +366,8 @@ describe('AssetStore', () => {
     beforeEach(async () => {
       assetStore = new AssetStore({});
       sdkMock.getTimeBuckets.mockResolvedValue([]);
-      await assetStore.init({ width: 0, height: 0 });
+      await assetStore.init();
+      await assetStore.updateViewport({ width: 0, height: 0 });
     });
 
     it('returns null for invalid buckets', () => {
diff --git a/web/src/lib/stores/assets.store.ts b/web/src/lib/stores/assets.store.ts
index 1022729e91..7fd82b4c3a 100644
--- a/web/src/lib/stores/assets.store.ts
+++ b/web/src/lib/stores/assets.store.ts
@@ -1,6 +1,11 @@
+import { locale } from '$lib/stores/preferences.store';
 import { getKey } from '$lib/utils';
-import { fromLocalDateTime } from '$lib/utils/timeline-util';
-import { TimeBucketSize, getTimeBucket, getTimeBuckets, type AssetResponseDto } from '@immich/sdk';
+import { AssetGridTaskManager } from '$lib/utils/asset-store-task-manager';
+import { getAssetRatio } from '$lib/utils/asset-utils';
+import type { AssetGridRouteSearchParams } from '$lib/utils/navigation';
+import { calculateWidth, fromLocalDateTime, splitBucketIntoDateGroups, type DateGroup } from '$lib/utils/timeline-util';
+import { TimeBucketSize, getAssetInfo, getTimeBucket, getTimeBuckets, type AssetResponseDto } from '@immich/sdk';
+import createJustifiedLayout from 'justified-layout';
 import { throttle } from 'lodash-es';
 import { DateTime } from 'luxon';
 import { t } from 'svelte-i18n';
@@ -8,19 +13,24 @@ import { get, writable, type Unsubscriber } from 'svelte/store';
 import { handleError } from '../utils/handle-error';
 import { websocketEvents } from './websocket';
 
-export enum BucketPosition {
-  Above = 'above',
-  Below = 'below',
-  Visible = 'visible',
-  Unknown = 'unknown',
-}
 type AssetApiGetTimeBucketsRequest = Parameters<typeof getTimeBuckets>[0];
 export type AssetStoreOptions = Omit<AssetApiGetTimeBucketsRequest, 'size'>;
 
+const LAYOUT_OPTIONS = {
+  boxSpacing: 2,
+  containerPadding: 0,
+  targetRowHeightTolerance: 0.15,
+  targetRowHeight: 235,
+};
+
 export interface Viewport {
   width: number;
   height: number;
 }
+export type ViewportXY = Viewport & {
+  x: number;
+  y: number;
+};
 
 interface AssetLookup {
   bucket: AssetBucket;
@@ -29,16 +39,89 @@ interface AssetLookup {
 }
 
 export class AssetBucket {
+  store!: AssetStore;
+  bucketDate!: string;
   /**
    * The DOM height of the bucket in pixel
    * This value is first estimated by the number of asset and later is corrected as the user scroll
    */
-  bucketHeight!: number;
-  bucketDate!: string;
-  bucketCount!: number;
-  assets!: AssetResponseDto[];
-  cancelToken!: AbortController | null;
-  position!: BucketPosition;
+  bucketHeight: number = 0;
+  isBucketHeightActual: boolean = false;
+  bucketDateFormattted!: string;
+  bucketCount: number = 0;
+  assets: AssetResponseDto[] = [];
+  dateGroups: DateGroup[] = [];
+  cancelToken: AbortController | undefined;
+  /**
+   * Prevent this asset's load from being canceled; i.e. to force load of offscreen asset.
+   */
+  isPreventCancel: boolean = false;
+  /**
+   * A promise that resolves once the bucket is loaded, and rejects if bucket is canceled.
+   */
+  complete!: Promise<void>;
+  loading: boolean = false;
+  isLoaded: boolean = false;
+  intersecting: boolean = false;
+  measured: boolean = false;
+  measuredPromise!: Promise<void>;
+
+  constructor(props: Partial<AssetBucket> & { store: AssetStore; bucketDate: string }) {
+    Object.assign(this, props);
+    this.init();
+  }
+
+  private init() {
+    // create a promise, and store its resolve/reject callbacks. The loadedSignal callback
+    // will be incoked when a bucket is loaded, fulfilling the promise. The canceledSignal
+    // callback will be called if the bucket is canceled before it was loaded, rejecting the
+    // promise.
+    this.complete = new Promise((resolve, reject) => {
+      this.loadedSignal = resolve;
+      this.canceledSignal = reject;
+    });
+    // if no-one waits on complete, and its rejected a uncaught rejection message is logged.
+    // We this message with an empty reject handler, since waiting on a bucket is optional.
+    this.complete.catch(() => void 0);
+    this.measuredPromise = new Promise((resolve) => {
+      this.measuredSignal = resolve;
+    });
+
+    this.bucketDateFormattted = fromLocalDateTime(this.bucketDate)
+      .startOf('month')
+      .toJSDate()
+      .toLocaleString(get(locale), {
+        month: 'short',
+        year: 'numeric',
+        timeZone: 'UTC',
+      });
+  }
+
+  private loadedSignal: (() => void) | undefined;
+  private canceledSignal: (() => void) | undefined;
+  measuredSignal: (() => void) | undefined;
+
+  cancel() {
+    if (this.isLoaded) {
+      return;
+    }
+    if (this.isPreventCancel) {
+      return;
+    }
+    this.cancelToken?.abort();
+    this.canceledSignal?.();
+    this.init();
+  }
+
+  loaded() {
+    this.loadedSignal?.();
+    this.isLoaded = true;
+  }
+
+  errored() {
+    this.canceledSignal?.();
+    this.init();
+  }
 }
 
 const isMismatched = (option: boolean | undefined, value: boolean): boolean =>
@@ -65,34 +148,101 @@ interface TrashAssets {
   type: 'trash';
   values: string[];
 }
+interface UpdateStackAssets {
+  type: 'update_stack_assets';
+  values: string[];
+}
 
 export const photoViewer = writable<HTMLImageElement | null>(null);
 
-type PendingChange = AddAsset | UpdateAsset | DeleteAsset | TrashAssets;
+type PendingChange = AddAsset | UpdateAsset | DeleteAsset | TrashAssets | UpdateStackAssets;
+
+export type BucketListener = (
+  event:
+    | ViewPortEvent
+    | BucketLoadEvent
+    | BucketLoadedEvent
+    | BucketCancelEvent
+    | BucketHeightEvent
+    | DateGroupIntersecting
+    | DateGroupHeightEvent,
+) => void;
+
+type ViewPortEvent = {
+  type: 'viewport';
+};
+type BucketLoadEvent = {
+  type: 'load';
+  bucket: AssetBucket;
+};
+type BucketLoadedEvent = {
+  type: 'loaded';
+  bucket: AssetBucket;
+};
+type BucketCancelEvent = {
+  type: 'cancel';
+  bucket: AssetBucket;
+};
+type BucketHeightEvent = {
+  type: 'bucket-height';
+  bucket: AssetBucket;
+  delta: number;
+};
+type DateGroupIntersecting = {
+  type: 'intersecting';
+  bucket: AssetBucket;
+  dateGroup: DateGroup;
+};
+type DateGroupHeightEvent = {
+  type: 'height';
+  bucket: AssetBucket;
+  dateGroup: DateGroup;
+  delta: number;
+  height: number;
+};
 
 export class AssetStore {
-  private store$ = writable(this);
   private assetToBucket: Record<string, AssetLookup> = {};
   private pendingChanges: PendingChange[] = [];
   private unsubscribers: Unsubscriber[] = [];
   private options: AssetApiGetTimeBucketsRequest;
+  private viewport: Viewport = {
+    height: 0,
+    width: 0,
+  };
+  private initializedSignal!: () => void;
+  private store$ = writable(this);
 
+  lastScrollTime: number = 0;
+  subscribe = this.store$.subscribe;
+  /**
+   * A promise that resolves once the store is initialized.
+   */
+  taskManager = new AssetGridTaskManager(this);
+  complete!: Promise<void>;
   initialized = false;
   timelineHeight = 0;
   buckets: AssetBucket[] = [];
   assets: AssetResponseDto[] = [];
   albumAssets: Set<string> = new Set();
+  pendingScrollBucket: AssetBucket | undefined;
+  pendingScrollAssetId: string | undefined;
+
+  listeners: BucketListener[] = [];
 
   constructor(
     options: AssetStoreOptions,
     private albumId?: string,
   ) {
     this.options = { ...options, size: TimeBucketSize.Month };
+    // create a promise, and store its resolve callbacks. The initializedSignal callback
+    // will be invoked when a the assetstore is initialized.
+    this.complete = new Promise((resolve) => {
+      this.initializedSignal = resolve;
+    });
     this.store$.set(this);
   }
 
-  subscribe = this.store$.subscribe;
-
   private addPendingChanges(...changes: PendingChange[]) {
     // prevent websocket events from happening before local client events
     setTimeout(() => {
@@ -182,8 +332,35 @@ export class AssetStore {
     this.emit(true);
   }, 2500);
 
-  async init(viewport: Viewport) {
-    this.initialized = false;
+  addListener(bucketListener: BucketListener) {
+    this.listeners.push(bucketListener);
+  }
+  removeListener(bucketListener: BucketListener) {
+    this.listeners = this.listeners.filter((l) => l != bucketListener);
+  }
+  private notifyListeners(
+    event:
+      | ViewPortEvent
+      | BucketLoadEvent
+      | BucketLoadedEvent
+      | BucketCancelEvent
+      | BucketHeightEvent
+      | DateGroupIntersecting
+      | DateGroupHeightEvent,
+  ) {
+    for (const fn of this.listeners) {
+      fn(event);
+    }
+  }
+  async init({ bucketListener }: { bucketListener?: BucketListener } = {}) {
+    if (this.initialized) {
+      throw 'Can only init once';
+    }
+    if (bucketListener) {
+      this.addListener(bucketListener);
+    }
+    //  uncaught rejection go away
+    this.complete.catch(() => void 0);
     this.timelineHeight = 0;
     this.buckets = [];
     this.assets = [];
@@ -194,65 +371,118 @@ export class AssetStore {
       ...this.options,
       key: getKey(),
     });
-
+    this.buckets = timebuckets.map(
+      (bucket) => new AssetBucket({ store: this, bucketDate: bucket.timeBucket, bucketCount: bucket.count }),
+    );
+    this.initializedSignal();
     this.initialized = true;
-
-    this.buckets = timebuckets.map((bucket) => ({
-      bucketDate: bucket.timeBucket,
-      bucketHeight: 0,
-      bucketCount: bucket.count,
-      assets: [],
-      cancelToken: null,
-      position: BucketPosition.Unknown,
-    }));
-
-    // if loading an asset, the grid-view may be hidden, which means
-    // it has 0 width and height. No need to update bucket or timeline
-    // heights in this case. Later, updateViewport will be called to
-    // update the heights.
-    if (viewport.height !== 0 && viewport.width !== 0) {
-      await this.updateViewport(viewport);
-    }
   }
 
-  async updateViewport(viewport: Viewport) {
+  public destroy() {
+    this.taskManager.destroy();
+    this.listeners = [];
+    this.initialized = false;
+  }
+
+  async updateViewport(viewport: Viewport, force?: boolean) {
+    if (!this.initialized) {
+      return;
+    }
+    if (viewport.height === 0 && viewport.width === 0) {
+      return;
+    }
+
+    if (!force && this.viewport.height === viewport.height && this.viewport.width === viewport.width) {
+      return;
+    }
+
+    // changing width invalidates the actual height, and needs to be remeasured, since width changes causes
+    // layout reflows.
+    const changedWidth = this.viewport.width != viewport.width;
+    this.viewport = { ...viewport };
+
     for (const bucket of this.buckets) {
-      const unwrappedWidth = (3 / 2) * bucket.bucketCount * THUMBNAIL_HEIGHT * (7 / 10);
-      const rows = Math.ceil(unwrappedWidth / viewport.width);
-      const height = rows * THUMBNAIL_HEIGHT;
-      bucket.bucketHeight = height;
+      this.updateGeometry(bucket, changedWidth);
     }
     this.timelineHeight = this.buckets.reduce((accumulator, b) => accumulator + b.bucketHeight, 0);
 
-    let height = 0;
     const loaders = [];
+    let height = 0;
     for (const bucket of this.buckets) {
-      if (height < viewport.height) {
-        height += bucket.bucketHeight;
-        loaders.push(this.loadBucket(bucket.bucketDate, BucketPosition.Visible));
-        continue;
+      if (height >= viewport.height) {
+        break;
       }
-      break;
+      height += bucket.bucketHeight;
+      loaders.push(this.loadBucket(bucket.bucketDate));
     }
     await Promise.all(loaders);
+    this.notifyListeners({ type: 'viewport' });
     this.emit(false);
   }
 
-  async loadBucket(bucketDate: string, position: BucketPosition): Promise<void> {
+  private updateGeometry(bucket: AssetBucket, invalidateHeight: boolean) {
+    if (invalidateHeight) {
+      bucket.isBucketHeightActual = false;
+      bucket.measured = false;
+      for (const assetGroup of bucket.dateGroups) {
+        assetGroup.heightActual = false;
+      }
+    }
+    if (!bucket.isBucketHeightActual) {
+      const unwrappedWidth = (3 / 2) * bucket.bucketCount * THUMBNAIL_HEIGHT * (7 / 10);
+      const rows = Math.ceil(unwrappedWidth / this.viewport.width);
+      const height = 51 + rows * THUMBNAIL_HEIGHT;
+      bucket.bucketHeight = height;
+    }
+
+    for (const assetGroup of bucket.dateGroups) {
+      if (!assetGroup.heightActual) {
+        const unwrappedWidth = (3 / 2) * assetGroup.assets.length * THUMBNAIL_HEIGHT * (7 / 10);
+        const rows = Math.ceil(unwrappedWidth / this.viewport.width);
+        const height = rows * THUMBNAIL_HEIGHT;
+        assetGroup.height = height;
+      }
+
+      const layoutResult = createJustifiedLayout(
+        assetGroup.assets.map((g) => getAssetRatio(g)),
+        {
+          ...LAYOUT_OPTIONS,
+          containerWidth: Math.floor(this.viewport.width),
+        },
+      );
+      assetGroup.geometry = {
+        ...layoutResult,
+        containerWidth: calculateWidth(layoutResult.boxes),
+      };
+    }
+  }
+
+  async loadBucket(bucketDate: string, options: { preventCancel?: boolean; pending?: boolean } = {}): Promise<void> {
     const bucket = this.getBucketByDate(bucketDate);
     if (!bucket) {
       return;
     }
-
-    bucket.position = position;
-
-    if (bucket.cancelToken || bucket.assets.length > 0) {
-      this.emit(false);
+    if (bucket.bucketCount === bucket.assets.length) {
+      // already loaded
       return;
     }
 
-    bucket.cancelToken = new AbortController();
+    if (bucket.cancelToken != null && bucket.bucketCount !== bucket.assets.length) {
+      // if promise is pending, and preventCancel is requested, then don't overwrite it
+      if (!bucket.isPreventCancel && options.preventCancel) {
+        bucket.isPreventCancel = options.preventCancel;
+      }
+      await bucket.complete;
+      return;
+    }
 
+    if (options.pending) {
+      this.pendingScrollBucket = bucket;
+    }
+    this.notifyListeners({ type: 'load', bucket });
+    bucket.isPreventCancel = !!options.preventCancel;
+
+    const cancelToken = (bucket.cancelToken = new AbortController());
     try {
       const assets = await getTimeBucket(
         {
@@ -260,9 +490,14 @@ export class AssetStore {
           timeBucket: bucketDate,
           key: getKey(),
         },
-        { signal: bucket.cancelToken.signal },
+        { signal: cancelToken.signal },
       );
 
+      if (cancelToken.signal.aborted) {
+        this.notifyListeners({ type: 'cancel', bucket });
+        return;
+      }
+
       if (this.albumId) {
         const albumAssets = await getTimeBucket(
           {
@@ -271,50 +506,87 @@ export class AssetStore {
             size: this.options.size,
             key: getKey(),
           },
-          { signal: bucket.cancelToken.signal },
+          { signal: cancelToken.signal },
         );
-
+        if (cancelToken.signal.aborted) {
+          this.notifyListeners({ type: 'cancel', bucket });
+          return;
+        }
         for (const asset of albumAssets) {
           this.albumAssets.add(asset.id);
         }
       }
 
-      if (bucket.cancelToken.signal.aborted) {
+      bucket.assets = assets;
+      bucket.dateGroups = splitBucketIntoDateGroups(bucket, get(locale));
+      this.updateGeometry(bucket, true);
+      this.timelineHeight = this.buckets.reduce((accumulator, b) => accumulator + b.bucketHeight, 0);
+      bucket.loaded();
+      this.notifyListeners({ type: 'loaded', bucket });
+    } catch (error) {
+      /* eslint-disable-next-line  @typescript-eslint/no-explicit-any */
+      if ((error as any).name === 'AbortError') {
         return;
       }
-
-      bucket.assets = assets;
-
-      this.emit(true);
-    } catch (error) {
       const $t = get(t);
       handleError(error, $t('errors.failed_to_load_assets'));
+      bucket.errored();
     } finally {
-      bucket.cancelToken = null;
+      bucket.cancelToken = undefined;
+      this.emit(true);
     }
   }
 
-  cancelBucket(bucket: AssetBucket) {
-    bucket.cancelToken?.abort();
-  }
-
-  updateBucket(bucketDate: string, height: number) {
+  updateBucket(bucketDate: string, properties: { height?: number; intersecting?: boolean; measured?: boolean }) {
     const bucket = this.getBucketByDate(bucketDate);
     if (!bucket) {
-      return 0;
+      return {};
+    }
+    let delta = 0;
+    if ('height' in properties) {
+      const height = properties.height!;
+      delta = height - bucket.bucketHeight;
+      bucket.isBucketHeightActual = true;
+      bucket.bucketHeight = height;
+      this.timelineHeight += delta;
+      this.notifyListeners({ type: 'bucket-height', bucket, delta });
+    }
+    if ('intersecting' in properties) {
+      bucket.intersecting = properties.intersecting!;
+    }
+    if ('measured' in properties) {
+      if (properties.measured) {
+        bucket.measuredSignal?.();
+      }
+      bucket.measured = properties.measured!;
     }
-
-    const delta = height - bucket.bucketHeight;
-    const scrollTimeline = bucket.position == BucketPosition.Above;
-
-    bucket.bucketHeight = height;
-    bucket.position = BucketPosition.Unknown;
-
-    this.timelineHeight += delta;
-
     this.emit(false);
+    return { delta };
+  }
 
-    return scrollTimeline ? delta : 0;
+  updateBucketDateGroup(
+    bucket: AssetBucket,
+    dateGroup: DateGroup,
+    properties: { height?: number; intersecting?: boolean },
+  ) {
+    let delta = 0;
+    if ('height' in properties) {
+      const height = properties.height!;
+      if (height > 0) {
+        delta = height - dateGroup.height;
+        dateGroup.heightActual = true;
+        dateGroup.height = height;
+        this.notifyListeners({ type: 'height', bucket, dateGroup, delta, height });
+      }
+    }
+    if ('intersecting' in properties) {
+      dateGroup.intersecting = properties.intersecting!;
+      if (dateGroup.intersecting) {
+        this.notifyListeners({ type: 'intersecting', bucket, dateGroup });
+      }
+    }
+    this.emit(false);
+    return { delta };
   }
 
   addAssets(assets: AssetResponseDto[]) {
@@ -354,15 +626,7 @@ export class AssetStore {
       let bucket = this.getBucketByDate(timeBucket);
 
       if (!bucket) {
-        bucket = {
-          bucketDate: timeBucket,
-          bucketHeight: THUMBNAIL_HEIGHT,
-          bucketCount: 0,
-          assets: [],
-          cancelToken: null,
-          position: BucketPosition.Unknown,
-        };
-
+        bucket = new AssetBucket({ store: this, bucketDate: timeBucket, bucketHeight: THUMBNAIL_HEIGHT });
         this.buckets.push(bucket);
       }
 
@@ -383,6 +647,8 @@ export class AssetStore {
         const bDate = DateTime.fromISO(b.fileCreatedAt).toUTC();
         return bDate.diff(aDate).milliseconds;
       });
+      bucket.dateGroups = splitBucketIntoDateGroups(bucket, get(locale));
+      this.updateGeometry(bucket, true);
     }
 
     this.emit(true);
@@ -392,18 +658,73 @@ export class AssetStore {
     return this.buckets.find((bucket) => bucket.bucketDate === bucketDate) || null;
   }
 
-  async getBucketInfoForAssetId({ id, localDateTime }: Pick<AssetResponseDto, 'id' | 'localDateTime'>) {
+  async findAndLoadBucketAsPending(id: string) {
     const bucketInfo = this.assetToBucket[id];
     if (bucketInfo) {
-      return bucketInfo;
+      const bucket = bucketInfo.bucket;
+      this.pendingScrollBucket = bucket;
+      this.pendingScrollAssetId = id;
+      this.emit(false);
+      return bucket;
     }
+    const asset = await getAssetInfo({ id });
+    if (asset) {
+      if (this.options.isArchived !== asset.isArchived) {
+        return;
+      }
+      const bucket = await this.loadBucketAtTime(asset.localDateTime, { preventCancel: true, pending: true });
+      if (bucket) {
+        this.pendingScrollBucket = bucket;
+        this.pendingScrollAssetId = asset.id;
+        this.emit(false);
+      }
+      return bucket;
+    }
+  }
+
+  /* Must be paired with matching clearPendingScroll() call */
+  async scheduleScrollToAssetId(scrollTarget: AssetGridRouteSearchParams, onFailure: () => void) {
+    try {
+      const { at: assetId } = scrollTarget;
+      if (assetId) {
+        await this.complete;
+        const bucket = await this.findAndLoadBucketAsPending(assetId);
+        if (bucket) {
+          return;
+        }
+      }
+    } catch {
+      // failure
+    }
+    onFailure();
+  }
+
+  clearPendingScroll() {
+    this.pendingScrollBucket = undefined;
+    this.pendingScrollAssetId = undefined;
+  }
+
+  private async loadBucketAtTime(localDateTime: string, options: { preventCancel?: boolean; pending?: boolean }) {
     let date = fromLocalDateTime(localDateTime);
     if (this.options.size == TimeBucketSize.Month) {
       date = date.set({ day: 1, hour: 0, minute: 0, second: 0, millisecond: 0 });
     } else if (this.options.size == TimeBucketSize.Day) {
       date = date.set({ hour: 0, minute: 0, second: 0, millisecond: 0 });
     }
-    await this.loadBucket(date.toISO()!, BucketPosition.Unknown);
+    const iso = date.toISO()!;
+    await this.loadBucket(iso, options);
+    return this.getBucketByDate(iso);
+  }
+
+  private async getBucketInfoForAsset(
+    { id, localDateTime }: Pick<AssetResponseDto, 'id' | 'localDateTime'>,
+    options: { preventCancel?: boolean; pending?: boolean } = {},
+  ) {
+    const bucketInfo = this.assetToBucket[id];
+    if (bucketInfo) {
+      return bucketInfo;
+    }
+    await this.loadBucketAtTime(localDateTime, options);
     return this.assetToBucket[id] || null;
   }
 
@@ -417,7 +738,7 @@ export class AssetStore {
     );
     for (const bucket of this.buckets) {
       if (index < bucket.bucketCount) {
-        await this.loadBucket(bucket.bucketDate, BucketPosition.Unknown);
+        await this.loadBucket(bucket.bucketDate);
         return bucket.assets[index] || null;
       }
 
@@ -458,6 +779,7 @@ export class AssetStore {
     // Iterate in reverse to allow array splicing.
     for (let index = this.buckets.length - 1; index >= 0; index--) {
       const bucket = this.buckets[index];
+      let changed = false;
       for (let index_ = bucket.assets.length - 1; index_ >= 0; index_--) {
         const asset = bucket.assets[index_];
         if (!idSet.has(asset.id)) {
@@ -465,17 +787,22 @@ export class AssetStore {
         }
 
         bucket.assets.splice(index_, 1);
+        changed = true;
         if (bucket.assets.length === 0) {
           this.buckets.splice(index, 1);
         }
       }
+      if (changed) {
+        bucket.dateGroups = splitBucketIntoDateGroups(bucket, get(locale));
+        this.updateGeometry(bucket, true);
+      }
     }
 
     this.emit(true);
   }
 
   async getPreviousAsset(asset: AssetResponseDto): Promise<AssetResponseDto | null> {
-    const info = await this.getBucketInfoForAssetId(asset);
+    const info = await this.getBucketInfoForAsset(asset);
     if (!info) {
       return null;
     }
@@ -491,12 +818,12 @@ export class AssetStore {
     }
 
     const previousBucket = this.buckets[bucketIndex - 1];
-    await this.loadBucket(previousBucket.bucketDate, BucketPosition.Unknown);
+    await this.loadBucket(previousBucket.bucketDate);
     return previousBucket.assets.at(-1) || null;
   }
 
   async getNextAsset(asset: AssetResponseDto): Promise<AssetResponseDto | null> {
-    const info = await this.getBucketInfoForAssetId(asset);
+    const info = await this.getBucketInfoForAsset(asset);
     if (!info) {
       return null;
     }
@@ -512,7 +839,7 @@ export class AssetStore {
     }
 
     const nextBucket = this.buckets[bucketIndex + 1];
-    await this.loadBucket(nextBucket.bucketDate, BucketPosition.Unknown);
+    await this.loadBucket(nextBucket.bucketDate);
     return nextBucket.assets[0] || null;
   }
 
@@ -537,8 +864,7 @@ export class AssetStore {
       }
       this.assetToBucket = assetToBucket;
     }
-
-    this.store$.update(() => this);
+    this.store$.set(this);
   }
 }
 
diff --git a/web/src/lib/utils/asset-store-task-manager.ts b/web/src/lib/utils/asset-store-task-manager.ts
new file mode 100644
index 0000000000..6ece1327c4
--- /dev/null
+++ b/web/src/lib/utils/asset-store-task-manager.ts
@@ -0,0 +1,465 @@
+import type { AssetBucket, AssetStore } from '$lib/stores/assets.store';
+import { generateId } from '$lib/utils/generate-id';
+import { cancelIdleCB, idleCB } from '$lib/utils/idle-callback-support';
+import { KeyedPriorityQueue } from '$lib/utils/keyed-priority-queue';
+import { type DateGroup } from '$lib/utils/timeline-util';
+import { TUNABLES } from '$lib/utils/tunables';
+import { type AssetResponseDto } from '@immich/sdk';
+import { clamp } from 'lodash-es';
+
+type Task = () => void;
+
+class InternalTaskManager {
+  assetStore: AssetStore;
+  componentTasks = new Map<string, Set<string>>();
+  priorityQueue = new KeyedPriorityQueue<string, Task>();
+  idleQueue = new Map<string, Task>();
+  taskCleaners = new Map<string, Task>();
+
+  queueTimer: ReturnType<typeof setTimeout> | undefined;
+  lastIdle: number | undefined;
+
+  constructor(assetStore: AssetStore) {
+    this.assetStore = assetStore;
+  }
+  destroy() {
+    this.componentTasks.clear();
+    this.priorityQueue.clear();
+    this.idleQueue.clear();
+    this.taskCleaners.clear();
+    clearTimeout(this.queueTimer);
+    if (this.lastIdle) {
+      cancelIdleCB(this.lastIdle);
+    }
+  }
+  getOrCreateComponentTasks(componentId: string) {
+    let componentTaskSet = this.componentTasks.get(componentId);
+    if (!componentTaskSet) {
+      componentTaskSet = new Set<string>();
+      this.componentTasks.set(componentId, componentTaskSet);
+    }
+
+    return componentTaskSet;
+  }
+  deleteFromComponentTasks(componentId: string, taskId: string) {
+    if (this.componentTasks.has(componentId)) {
+      const componentTaskSet = this.componentTasks.get(componentId);
+      componentTaskSet?.delete(taskId);
+      if (componentTaskSet?.size === 0) {
+        this.componentTasks.delete(componentId);
+      }
+    }
+  }
+
+  drainIntersectedQueue() {
+    let count = 0;
+    for (let t = this.priorityQueue.shift(); t; t = this.priorityQueue.shift()) {
+      t.value();
+      if (this.taskCleaners.has(t.key)) {
+        this.taskCleaners.get(t.key)!();
+        this.taskCleaners.delete(t.key);
+      }
+      if (count++ >= TUNABLES.SCROLL_TASK_QUEUE.DRAIN_MAX_TASKS) {
+        this.scheduleDrainIntersectedQueue(TUNABLES.SCROLL_TASK_QUEUE.DRAIN_MAX_TASKS_DELAY_MS);
+        break;
+      }
+    }
+  }
+
+  scheduleDrainIntersectedQueue(delay: number = TUNABLES.SCROLL_TASK_QUEUE.CHECK_INTERVAL_MS) {
+    clearTimeout(this.queueTimer);
+    this.queueTimer = setTimeout(() => {
+      const delta = Date.now() - this.assetStore.lastScrollTime;
+      if (delta < TUNABLES.SCROLL_TASK_QUEUE.MIN_DELAY_MS) {
+        let amount = clamp(
+          1 + Math.round(this.priorityQueue.length / TUNABLES.SCROLL_TASK_QUEUE.TRICKLE_BONUS_FACTOR),
+          1,
+          TUNABLES.SCROLL_TASK_QUEUE.DRAIN_MAX_TASKS * 2,
+        );
+
+        const nextDelay = clamp(
+          amount > 1
+            ? Math.round(delay / TUNABLES.SCROLL_TASK_QUEUE.TRICKLE_ACCELERATION_FACTOR)
+            : TUNABLES.SCROLL_TASK_QUEUE.CHECK_INTERVAL_MS,
+          TUNABLES.SCROLL_TASK_QUEUE.TRICKLE_ACCELERATED_MIN_DELAY,
+          TUNABLES.SCROLL_TASK_QUEUE.TRICKLE_ACCELERATED_MAX_DELAY,
+        );
+
+        while (amount > 0) {
+          this.priorityQueue.shift()?.value();
+          amount--;
+        }
+        if (this.priorityQueue.length > 0) {
+          this.scheduleDrainIntersectedQueue(nextDelay);
+        }
+      } else {
+        this.drainIntersectedQueue();
+      }
+    }, delay);
+  }
+
+  removeAllTasksForComponent(componentId: string) {
+    if (this.componentTasks.has(componentId)) {
+      const tasksIds = this.componentTasks.get(componentId) || [];
+      for (const taskId of tasksIds) {
+        this.priorityQueue.remove(taskId);
+        this.idleQueue.delete(taskId);
+        if (this.taskCleaners.has(taskId)) {
+          const cleanup = this.taskCleaners.get(taskId);
+          this.taskCleaners.delete(taskId);
+          cleanup!();
+        }
+      }
+    }
+    this.componentTasks.delete(componentId);
+  }
+
+  queueScrollSensitiveTask({
+    task,
+    cleanup,
+    componentId,
+    priority = 10,
+    taskId = generateId(),
+  }: {
+    task: Task;
+    cleanup?: Task;
+    componentId: string;
+    priority?: number;
+    taskId?: string;
+  }) {
+    this.priorityQueue.push(taskId, task, priority);
+    if (cleanup) {
+      this.taskCleaners.set(taskId, cleanup);
+    }
+    this.getOrCreateComponentTasks(componentId).add(taskId);
+    const lastTime = this.assetStore.lastScrollTime;
+    const delta = Date.now() - lastTime;
+    if (lastTime != 0 && delta < TUNABLES.SCROLL_TASK_QUEUE.MIN_DELAY_MS) {
+      this.scheduleDrainIntersectedQueue();
+    } else {
+      // flush the queue early
+      clearTimeout(this.queueTimer);
+      this.drainIntersectedQueue();
+    }
+  }
+
+  scheduleDrainSeparatedQueue() {
+    if (this.lastIdle) {
+      cancelIdleCB(this.lastIdle);
+    }
+    this.lastIdle = idleCB(
+      () => {
+        let count = 0;
+        let entry = this.idleQueue.entries().next().value;
+        while (entry) {
+          const [taskId, task] = entry;
+          this.idleQueue.delete(taskId);
+          task();
+          if (count++ >= TUNABLES.SCROLL_TASK_QUEUE.DRAIN_MAX_TASKS) {
+            break;
+          }
+          entry = this.idleQueue.entries().next().value;
+        }
+        if (this.idleQueue.size > 0) {
+          this.scheduleDrainSeparatedQueue();
+        }
+      },
+      { timeout: 1000 },
+    );
+  }
+  queueSeparateTask({
+    task,
+    cleanup,
+    componentId,
+    taskId,
+  }: {
+    task: Task;
+    cleanup: Task;
+    componentId: string;
+    taskId: string;
+  }) {
+    this.idleQueue.set(taskId, task);
+    this.taskCleaners.set(taskId, cleanup);
+    this.getOrCreateComponentTasks(componentId).add(taskId);
+    this.scheduleDrainSeparatedQueue();
+  }
+
+  removeIntersectedTask(taskId: string) {
+    const removed = this.priorityQueue.remove(taskId);
+    if (this.taskCleaners.has(taskId)) {
+      const cleanup = this.taskCleaners.get(taskId);
+      this.taskCleaners.delete(taskId);
+      cleanup!();
+    }
+    return removed;
+  }
+
+  removeSeparateTask(taskId: string) {
+    const removed = this.idleQueue.delete(taskId);
+    if (this.taskCleaners.has(taskId)) {
+      const cleanup = this.taskCleaners.get(taskId);
+      this.taskCleaners.delete(taskId);
+      cleanup!();
+    }
+    return removed;
+  }
+}
+
+export class AssetGridTaskManager {
+  private internalManager: InternalTaskManager;
+  constructor(assetStore: AssetStore) {
+    this.internalManager = new InternalTaskManager(assetStore);
+  }
+
+  tasks: Map<AssetBucket, BucketTask> = new Map();
+
+  queueScrollSensitiveTask({
+    task,
+    cleanup,
+    componentId,
+    priority = 10,
+    taskId = generateId(),
+  }: {
+    task: Task;
+    cleanup?: Task;
+    componentId: string;
+    priority?: number;
+    taskId?: string;
+  }) {
+    return this.internalManager.queueScrollSensitiveTask({ task, cleanup, componentId, priority, taskId });
+  }
+
+  removeAllTasksForComponent(componentId: string) {
+    return this.internalManager.removeAllTasksForComponent(componentId);
+  }
+
+  destroy() {
+    return this.internalManager.destroy();
+  }
+
+  private getOrCreateBucketTask(bucket: AssetBucket) {
+    let bucketTask = this.tasks.get(bucket);
+    if (!bucketTask) {
+      bucketTask = this.createBucketTask(bucket);
+    }
+    return bucketTask;
+  }
+
+  private createBucketTask(bucket: AssetBucket) {
+    const bucketTask = new BucketTask(this.internalManager, this, bucket);
+    this.tasks.set(bucket, bucketTask);
+    return bucketTask;
+  }
+
+  intersectedBucket(componentId: string, bucket: AssetBucket, task: Task) {
+    const bucketTask = this.getOrCreateBucketTask(bucket);
+    bucketTask.scheduleIntersected(componentId, task);
+  }
+
+  seperatedBucket(componentId: string, bucket: AssetBucket, seperated: Task) {
+    const bucketTask = this.getOrCreateBucketTask(bucket);
+    bucketTask.scheduleSeparated(componentId, seperated);
+  }
+
+  intersectedDateGroup(componentId: string, dateGroup: DateGroup, intersected: Task) {
+    const bucketTask = this.getOrCreateBucketTask(dateGroup.bucket);
+    bucketTask.intersectedDateGroup(componentId, dateGroup, intersected);
+  }
+
+  seperatedDateGroup(componentId: string, dateGroup: DateGroup, seperated: Task) {
+    const bucketTask = this.getOrCreateBucketTask(dateGroup.bucket);
+    bucketTask.separatedDateGroup(componentId, dateGroup, seperated);
+  }
+
+  intersectedThumbnail(componentId: string, dateGroup: DateGroup, asset: AssetResponseDto, intersected: Task) {
+    const bucketTask = this.getOrCreateBucketTask(dateGroup.bucket);
+    const dateGroupTask = bucketTask.getOrCreateDateGroupTask(dateGroup);
+    dateGroupTask.intersectedThumbnail(componentId, asset, intersected);
+  }
+
+  seperatedThumbnail(componentId: string, dateGroup: DateGroup, asset: AssetResponseDto, seperated: Task) {
+    const bucketTask = this.getOrCreateBucketTask(dateGroup.bucket);
+    const dateGroupTask = bucketTask.getOrCreateDateGroupTask(dateGroup);
+    dateGroupTask.separatedThumbnail(componentId, asset, seperated);
+  }
+}
+
+class IntersectionTask {
+  internalTaskManager: InternalTaskManager;
+  seperatedKey;
+  intersectedKey;
+  priority;
+
+  intersected: Task | undefined;
+  separated: Task | undefined;
+
+  constructor(internalTaskManager: InternalTaskManager, keyPrefix: string, key: string, priority: number) {
+    this.internalTaskManager = internalTaskManager;
+    this.seperatedKey = keyPrefix + ':s:' + key;
+    this.intersectedKey = keyPrefix + ':i:' + key;
+    this.priority = priority;
+  }
+
+  trackIntersectedTask(componentId: string, task: Task) {
+    const execTask = () => {
+      if (this.separated) {
+        return;
+      }
+      task?.();
+    };
+    this.intersected = execTask;
+    const cleanup = () => {
+      this.intersected = undefined;
+      this.internalTaskManager.deleteFromComponentTasks(componentId, this.intersectedKey);
+    };
+    return { task: execTask, cleanup };
+  }
+
+  trackSeperatedTask(componentId: string, task: Task) {
+    const execTask = () => {
+      if (this.intersected) {
+        return;
+      }
+      task?.();
+    };
+    this.separated = execTask;
+    const cleanup = () => {
+      this.separated = undefined;
+      this.internalTaskManager.deleteFromComponentTasks(componentId, this.seperatedKey);
+    };
+    return { task: execTask, cleanup };
+  }
+
+  removePendingSeparated() {
+    if (this.separated) {
+      this.internalTaskManager.removeSeparateTask(this.seperatedKey);
+    }
+  }
+  removePendingIntersected() {
+    if (this.intersected) {
+      this.internalTaskManager.removeIntersectedTask(this.intersectedKey);
+    }
+  }
+
+  scheduleIntersected(componentId: string, intersected: Task) {
+    this.removePendingSeparated();
+    if (this.intersected) {
+      return;
+    }
+    const { task, cleanup } = this.trackIntersectedTask(componentId, intersected);
+    this.internalTaskManager.queueScrollSensitiveTask({
+      task,
+      cleanup,
+      componentId: componentId,
+      priority: this.priority,
+      taskId: this.intersectedKey,
+    });
+  }
+
+  scheduleSeparated(componentId: string, separated: Task) {
+    this.removePendingIntersected();
+
+    if (this.separated) {
+      return;
+    }
+
+    const { task, cleanup } = this.trackSeperatedTask(componentId, separated);
+    this.internalTaskManager.queueSeparateTask({
+      task,
+      cleanup,
+      componentId: componentId,
+      taskId: this.seperatedKey,
+    });
+  }
+}
+class BucketTask extends IntersectionTask {
+  assetBucket: AssetBucket;
+  assetGridTaskManager: AssetGridTaskManager;
+  // indexed by dateGroup's date
+  dateTasks: Map<DateGroup, DateGroupTask> = new Map();
+
+  constructor(internalTaskManager: InternalTaskManager, parent: AssetGridTaskManager, assetBucket: AssetBucket) {
+    super(internalTaskManager, 'b', assetBucket.bucketDate, TUNABLES.BUCKET.PRIORITY);
+    this.assetBucket = assetBucket;
+    this.assetGridTaskManager = parent;
+  }
+
+  getOrCreateDateGroupTask(dateGroup: DateGroup) {
+    let dateGroupTask = this.dateTasks.get(dateGroup);
+    if (!dateGroupTask) {
+      dateGroupTask = this.createDateGroupTask(dateGroup);
+    }
+    return dateGroupTask;
+  }
+
+  createDateGroupTask(dateGroup: DateGroup) {
+    const dateGroupTask = new DateGroupTask(this.internalTaskManager, this, dateGroup);
+    this.dateTasks.set(dateGroup, dateGroupTask);
+    return dateGroupTask;
+  }
+
+  removePendingSeparated() {
+    super.removePendingSeparated();
+    for (const dateGroupTask of this.dateTasks.values()) {
+      dateGroupTask.removePendingSeparated();
+    }
+  }
+
+  intersectedDateGroup(componentId: string, dateGroup: DateGroup, intersected: Task) {
+    const dateGroupTask = this.getOrCreateDateGroupTask(dateGroup);
+    dateGroupTask.scheduleIntersected(componentId, intersected);
+  }
+
+  separatedDateGroup(componentId: string, dateGroup: DateGroup, separated: Task) {
+    const dateGroupTask = this.getOrCreateDateGroupTask(dateGroup);
+    dateGroupTask.scheduleSeparated(componentId, separated);
+  }
+}
+class DateGroupTask extends IntersectionTask {
+  dateGroup: DateGroup;
+  bucketTask: BucketTask;
+  // indexed by thumbnail's asset
+  thumbnailTasks: Map<AssetResponseDto, ThumbnailTask> = new Map();
+
+  constructor(internalTaskManager: InternalTaskManager, parent: BucketTask, dateGroup: DateGroup) {
+    super(internalTaskManager, 'dg', dateGroup.date.toString(), TUNABLES.DATEGROUP.PRIORITY);
+    this.dateGroup = dateGroup;
+    this.bucketTask = parent;
+  }
+
+  removePendingSeparated() {
+    super.removePendingSeparated();
+    for (const thumbnailTask of this.thumbnailTasks.values()) {
+      thumbnailTask.removePendingSeparated();
+    }
+  }
+
+  getOrCreateThumbnailTask(asset: AssetResponseDto) {
+    let thumbnailTask = this.thumbnailTasks.get(asset);
+    if (!thumbnailTask) {
+      thumbnailTask = new ThumbnailTask(this.internalTaskManager, this, asset);
+      this.thumbnailTasks.set(asset, thumbnailTask);
+    }
+    return thumbnailTask;
+  }
+
+  intersectedThumbnail(componentId: string, asset: AssetResponseDto, intersected: Task) {
+    const thumbnailTask = this.getOrCreateThumbnailTask(asset);
+    thumbnailTask.scheduleIntersected(componentId, intersected);
+  }
+
+  separatedThumbnail(componentId: string, asset: AssetResponseDto, seperated: Task) {
+    const thumbnailTask = this.getOrCreateThumbnailTask(asset);
+    thumbnailTask.scheduleSeparated(componentId, seperated);
+  }
+}
+class ThumbnailTask extends IntersectionTask {
+  asset: AssetResponseDto;
+  dateGroupTask: DateGroupTask;
+
+  constructor(internalTaskManager: InternalTaskManager, parent: DateGroupTask, asset: AssetResponseDto) {
+    super(internalTaskManager, 't', asset.id, TUNABLES.THUMBNAIL.PRIORITY);
+    this.asset = asset;
+    this.dateGroupTask = parent;
+  }
+}
diff --git a/web/src/lib/utils/asset-utils.ts b/web/src/lib/utils/asset-utils.ts
index 2722745317..576b14b201 100644
--- a/web/src/lib/utils/asset-utils.ts
+++ b/web/src/lib/utils/asset-utils.ts
@@ -4,7 +4,7 @@ import { NotificationType, notificationController } from '$lib/components/shared
 import { AppRoute } from '$lib/constants';
 import type { AssetInteractionStore } from '$lib/stores/asset-interaction.store';
 import { assetViewingStore } from '$lib/stores/asset-viewing.store';
-import { BucketPosition, isSelectingAllAssets, type AssetStore } from '$lib/stores/assets.store';
+import { isSelectingAllAssets, type AssetStore } from '$lib/stores/assets.store';
 import { downloadManager } from '$lib/stores/download';
 import { preferences } from '$lib/stores/user.store';
 import { downloadRequest, getKey, withError } from '$lib/utils';
@@ -403,7 +403,7 @@ export const selectAllAssets = async (assetStore: AssetStore, assetInteractionSt
 
   try {
     for (const bucket of assetStore.buckets) {
-      await assetStore.loadBucket(bucket.bucketDate, BucketPosition.Unknown);
+      await assetStore.loadBucket(bucket.bucketDate);
 
       if (!get(isSelectingAllAssets)) {
         break; // Cancelled
diff --git a/web/src/lib/utils/idle-callback-support.ts b/web/src/lib/utils/idle-callback-support.ts
new file mode 100644
index 0000000000..0f7f060084
--- /dev/null
+++ b/web/src/lib/utils/idle-callback-support.ts
@@ -0,0 +1,20 @@
+interface RequestIdleCallback {
+  didTimeout?: boolean;
+  timeRemaining?(): DOMHighResTimeStamp;
+}
+interface RequestIdleCallbackOptions {
+  timeout?: number;
+}
+
+// eslint-disable-next-line @typescript-eslint/no-explicit-any
+function fake_requestIdleCallback(cb: (deadline: RequestIdleCallback) => any, _?: RequestIdleCallbackOptions) {
+  const start = Date.now();
+  return setTimeout(cb({ didTimeout: false, timeRemaining: () => Math.max(0, 50 - (Date.now() - start)) }), 100);
+}
+
+function fake_cancelIdleCallback(id: number) {
+  return clearTimeout(id);
+}
+
+export const idleCB = window.requestIdleCallback || fake_requestIdleCallback;
+export const cancelIdleCB = window.cancelIdleCallback || fake_cancelIdleCallback;
diff --git a/web/src/lib/utils/keyed-priority-queue.ts b/web/src/lib/utils/keyed-priority-queue.ts
new file mode 100644
index 0000000000..2483b22c6d
--- /dev/null
+++ b/web/src/lib/utils/keyed-priority-queue.ts
@@ -0,0 +1,50 @@
+export class KeyedPriorityQueue<K, T> {
+  private items: { key: K; value: T; priority: number }[] = [];
+  private set: Set<K> = new Set();
+
+  clear() {
+    this.items = [];
+    this.set.clear();
+  }
+
+  remove(key: K) {
+    const removed = this.set.delete(key);
+    if (removed) {
+      const idx = this.items.findIndex((i) => i.key === key);
+      if (idx >= 0) {
+        this.items.splice(idx, 1);
+      }
+    }
+    return removed;
+  }
+
+  push(key: K, value: T, priority: number) {
+    if (this.set.has(key)) {
+      return this.length;
+    }
+    for (let i = 0; i < this.items.length; i++) {
+      if (this.items[i].priority > priority) {
+        this.set.add(key);
+        this.items.splice(i, 0, { key, value, priority });
+        return this.length;
+      }
+    }
+    this.set.add(key);
+    return this.items.push({ key, value, priority });
+  }
+
+  shift() {
+    let item = this.items.shift();
+    while (item) {
+      if (this.set.has(item.key)) {
+        this.set.delete(item.key);
+        return item;
+      }
+      item = this.items.shift();
+    }
+  }
+
+  get length() {
+    return this.set.size;
+  }
+}
diff --git a/web/src/lib/utils/navigation.ts b/web/src/lib/utils/navigation.ts
index 4d5660f173..304376b347 100644
--- a/web/src/lib/utils/navigation.ts
+++ b/web/src/lib/utils/navigation.ts
@@ -5,6 +5,9 @@ import { getAssetInfo } from '@immich/sdk';
 import type { NavigationTarget } from '@sveltejs/kit';
 import { get } from 'svelte/store';
 
+export type AssetGridRouteSearchParams = {
+  at: string | null | undefined;
+};
 export const isExternalUrl = (url: string): boolean => {
   return new URL(url, window.location.href).origin !== window.location.origin;
 };
@@ -33,17 +36,38 @@ function currentUrlWithoutAsset() {
 
 export function currentUrlReplaceAssetId(assetId: string) {
   const $page = get(page);
+  const params = new URLSearchParams($page.url.search);
+  // always remove the assetGridScrollTargetParams
+  params.delete('at');
+  const searchparams = params.size > 0 ? '?' + params.toString() : '';
   // this contains special casing for the /photos/:assetId photos route, which hangs directly
   // off / instead of a subpath, unlike every other asset-containing route.
   return isPhotosRoute($page.route.id)
-    ? `${AppRoute.PHOTOS}/${assetId}${$page.url.search}`
-    : `${$page.url.pathname.replace(/(\/photos.*)$/, '')}/photos/${assetId}${$page.url.search}`;
+    ? `${AppRoute.PHOTOS}/${assetId}${searchparams}`
+    : `${$page.url.pathname.replace(/(\/photos.*)$/, '')}/photos/${assetId}${searchparams}`;
+}
+
+function replaceScrollTarget(url: string, searchParams?: AssetGridRouteSearchParams | null) {
+  const $page = get(page);
+  const parsed = new URL(url, $page.url);
+
+  const { at: assetId } = searchParams || { at: null };
+
+  if (!assetId) {
+    return parsed.pathname;
+  }
+
+  const params = new URLSearchParams($page.url.search);
+  if (assetId) {
+    params.set('at', assetId);
+  }
+  return parsed.pathname + '?' + params.toString();
 }
 
 function currentUrl() {
   const $page = get(page);
   const current = $page.url;
-  return current.pathname + current.search;
+  return current.pathname + current.search + current.hash;
 }
 
 interface Route {
@@ -55,24 +79,58 @@ interface Route {
 
 interface AssetRoute extends Route {
   targetRoute: 'current';
-  assetId: string | null;
+  assetId: string | null | undefined;
 }
+interface AssetGridRoute extends Route {
+  targetRoute: 'current';
+  assetId: string | null | undefined;
+  assetGridRouteSearchParams: AssetGridRouteSearchParams | null | undefined;
+}
+
+type ImmichRoute = AssetRoute | AssetGridRoute;
+
+type NavOptions = {
+  /* navigate even if url is the same */
+  forceNavigate?: boolean | undefined;
+  replaceState?: boolean | undefined;
+  noScroll?: boolean | undefined;
+  keepFocus?: boolean | undefined;
+  invalidateAll?: boolean | undefined;
+  state?: App.PageState | undefined;
+};
 
 function isAssetRoute(route: Route): route is AssetRoute {
   return route.targetRoute === 'current' && 'assetId' in route;
 }
 
-async function navigateAssetRoute(route: AssetRoute) {
+function isAssetGridRoute(route: Route): route is AssetGridRoute {
+  return route.targetRoute === 'current' && 'assetId' in route && 'assetGridRouteSearchParams' in route;
+}
+
+async function navigateAssetRoute(route: AssetRoute, options?: NavOptions) {
   const { assetId } = route;
   const next = assetId ? currentUrlReplaceAssetId(assetId) : currentUrlWithoutAsset();
-  if (next !== currentUrl()) {
-    await goto(next, { replaceState: false });
+  const current = currentUrl();
+  if (next !== current || options?.forceNavigate) {
+    await goto(next, options);
   }
 }
 
-export function navigate<T extends Route>(change: T): Promise<void> {
-  if (isAssetRoute(change)) {
-    return navigateAssetRoute(change);
+async function navigateAssetGridRoute(route: AssetGridRoute, options?: NavOptions) {
+  const { assetId, assetGridRouteSearchParams: assetGridScrollTarget } = route;
+  const assetUrl = assetId ? currentUrlReplaceAssetId(assetId) : currentUrlWithoutAsset();
+  const next = replaceScrollTarget(assetUrl, assetGridScrollTarget);
+  const current = currentUrl();
+  if (next !== current || options?.forceNavigate) {
+    await goto(next, options);
+  }
+}
+
+export function navigate(change: ImmichRoute, options?: NavOptions): Promise<void> {
+  if (isAssetGridRoute(change)) {
+    return navigateAssetGridRoute(change, options);
+  } else if (isAssetRoute(change)) {
+    return navigateAssetRoute(change, options);
   }
   // future navigation requests here
   throw `Invalid navigation: ${JSON.stringify(change)}`;
diff --git a/web/src/lib/utils/priority-queue.ts b/web/src/lib/utils/priority-queue.ts
new file mode 100644
index 0000000000..6b08ffe7ad
--- /dev/null
+++ b/web/src/lib/utils/priority-queue.ts
@@ -0,0 +1,21 @@
+export class PriorityQueue<T> {
+  private items: { value: T; priority: number }[] = [];
+
+  push(value: T, priority: number) {
+    for (let i = 0; i < this.items.length; i++) {
+      if (this.items[i].priority > priority) {
+        this.items.splice(i, 0, { value, priority });
+        return this.length;
+      }
+    }
+    return this.items.push({ value, priority });
+  }
+
+  shift() {
+    return this.items.shift();
+  }
+
+  get length() {
+    return this.items.length;
+  }
+}
diff --git a/web/src/lib/utils/timeline-util.ts b/web/src/lib/utils/timeline-util.ts
index 76a0d1b5cb..3a8f66ee08 100644
--- a/web/src/lib/utils/timeline-util.ts
+++ b/web/src/lib/utils/timeline-util.ts
@@ -1,9 +1,38 @@
+import type { AssetBucket } from '$lib/stores/assets.store';
 import { locale } from '$lib/stores/preferences.store';
 import type { AssetResponseDto } from '@immich/sdk';
-import { groupBy, sortBy } from 'lodash-es';
+import type createJustifiedLayout from 'justified-layout';
+import { groupBy, memoize, sortBy } from 'lodash-es';
 import { DateTime } from 'luxon';
 import { get } from 'svelte/store';
 
+export type DateGroup = {
+  date: DateTime;
+  groupTitle: string;
+  assets: AssetResponseDto[];
+  height: number;
+  heightActual: boolean;
+  intersecting: boolean;
+  geometry: Geometry;
+  bucket: AssetBucket;
+};
+export type ScrubberListener = (
+  bucketDate: string | undefined,
+  overallScrollPercent: number,
+  bucketScrollPercent: number,
+) => void | Promise<void>;
+export type ScrollTargetListener = ({
+  bucket,
+  dateGroup,
+  asset,
+  offset,
+}: {
+  bucket: AssetBucket;
+  dateGroup: DateGroup;
+  asset: AssetResponseDto;
+  offset: number;
+}) => void;
+
 export const fromLocalDateTime = (localDateTime: string) =>
   DateTime.fromISO(localDateTime, { zone: 'UTC', locale: get(locale) });
 
@@ -48,20 +77,48 @@ export function formatGroupTitle(_date: DateTime): string {
   return date.toLocaleString(groupDateFormat);
 }
 
-export function splitBucketIntoDateGroups(
-  assets: AssetResponseDto[],
-  locale: string | undefined,
-): AssetResponseDto[][] {
-  const grouped = groupBy(assets, (asset) =>
+type Geometry = ReturnType<typeof createJustifiedLayout> & {
+  containerWidth: number;
+};
+
+function emptyGeometry() {
+  return {
+    containerWidth: 0,
+    containerHeight: 0,
+    widowCount: 0,
+    boxes: [],
+  };
+}
+
+const formatDateGroupTitle = memoize(formatGroupTitle);
+
+export function splitBucketIntoDateGroups(bucket: AssetBucket, locale: string | undefined): DateGroup[] {
+  const grouped = groupBy(bucket.assets, (asset) =>
     fromLocalDateTime(asset.localDateTime).toLocaleString(groupDateFormat, { locale }),
   );
-  return sortBy(grouped, (group) => assets.indexOf(group[0]));
+  const sorted = sortBy(grouped, (group) => bucket.assets.indexOf(group[0]));
+  return sorted.map((group) => {
+    const date = fromLocalDateTime(group[0].localDateTime).startOf('day');
+    return {
+      date,
+      groupTitle: formatDateGroupTitle(date),
+      assets: group,
+      height: 0,
+      heightActual: false,
+      intersecting: false,
+      geometry: emptyGeometry(),
+      bucket: bucket,
+    };
+  });
 }
 
 export type LayoutBox = {
+  aspectRatio: number;
   top: number;
-  left: number;
   width: number;
+  height: number;
+  left: number;
+  forcedAspectRatio?: boolean;
 };
 
 export function calculateWidth(boxes: LayoutBox[]): number {
@@ -71,6 +128,14 @@ export function calculateWidth(boxes: LayoutBox[]): number {
       width = box.left + box.width;
     }
   }
-
   return width;
 }
+
+export function findTotalOffset(element: HTMLElement, stop: HTMLElement) {
+  let offset = 0;
+  while (element.offsetParent && element !== stop) {
+    offset += element.offsetTop;
+    element = element.offsetParent as HTMLElement;
+  }
+  return offset;
+}
diff --git a/web/src/lib/utils/tunables.ts b/web/src/lib/utils/tunables.ts
new file mode 100644
index 0000000000..e21c30de77
--- /dev/null
+++ b/web/src/lib/utils/tunables.ts
@@ -0,0 +1,63 @@
+function getBoolean(string: string | null, fallback: boolean) {
+  if (string === null) {
+    return fallback;
+  }
+  return 'true' === string;
+}
+function getNumber(string: string | null, fallback: number) {
+  if (string === null) {
+    return fallback;
+  }
+  return Number.parseInt(string);
+}
+function getFloat(string: string | null, fallback: number) {
+  if (string === null) {
+    return fallback;
+  }
+  return Number.parseFloat(string);
+}
+export const TUNABLES = {
+  SCROLL_TASK_QUEUE: {
+    TRICKLE_BONUS_FACTOR: getNumber(localStorage.getItem('SCROLL_TASK_QUEUE.TRICKLE_BONUS_FACTOR'), 25),
+    TRICKLE_ACCELERATION_FACTOR: getFloat(localStorage.getItem('SCROLL_TASK_QUEUE.TRICKLE_ACCELERATION_FACTOR'), 1.5),
+    TRICKLE_ACCELERATED_MIN_DELAY: getNumber(
+      localStorage.getItem('SCROLL_TASK_QUEUE.TRICKLE_ACCELERATED_MIN_DELAY'),
+      8,
+    ),
+    TRICKLE_ACCELERATED_MAX_DELAY: getNumber(
+      localStorage.getItem('SCROLL_TASK_QUEUE.TRICKLE_ACCELERATED_MAX_DELAY'),
+      2000,
+    ),
+    DRAIN_MAX_TASKS: getNumber(localStorage.getItem('SCROLL_TASK_QUEUE.DRAIN_MAX_TASKS'), 15),
+    DRAIN_MAX_TASKS_DELAY_MS: getNumber(localStorage.getItem('SCROLL_TASK_QUEUE.DRAIN_MAX_TASKS_DELAY_MS'), 16),
+    MIN_DELAY_MS: getNumber(localStorage.getItem('SCROLL_TASK_QUEUE.MIN_DELAY_MS')!, 200),
+    CHECK_INTERVAL_MS: getNumber(localStorage.getItem('SCROLL_TASK_QUEUE.CHECK_INTERVAL_MS'), 16),
+  },
+  INTERSECTION_OBSERVER_QUEUE: {
+    DRAIN_MAX_TASKS: getNumber(localStorage.getItem('INTERSECTION_OBSERVER_QUEUE.DRAIN_MAX_TASKS'), 15),
+    THROTTLE_MS: getNumber(localStorage.getItem('INTERSECTION_OBSERVER_QUEUE.THROTTLE_MS'), 16),
+    THROTTLE: getBoolean(localStorage.getItem('INTERSECTION_OBSERVER_QUEUE.THROTTLE'), true),
+  },
+  ASSET_GRID: {
+    NAVIGATE_ON_ASSET_IN_VIEW: getBoolean(localStorage.getItem('ASSET_GRID.NAVIGATE_ON_ASSET_IN_VIEW'), false),
+  },
+  BUCKET: {
+    PRIORITY: getNumber(localStorage.getItem('BUCKET.PRIORITY'), 2),
+    INTERSECTION_ROOT_TOP: localStorage.getItem('BUCKET.INTERSECTION_ROOT_TOP') || '300%',
+    INTERSECTION_ROOT_BOTTOM: localStorage.getItem('BUCKET.INTERSECTION_ROOT_BOTTOM') || '300%',
+  },
+  DATEGROUP: {
+    PRIORITY: getNumber(localStorage.getItem('DATEGROUP.PRIORITY'), 4),
+    INTERSECTION_DISABLED: getBoolean(localStorage.getItem('DATEGROUP.INTERSECTION_DISABLED'), false),
+    INTERSECTION_ROOT_TOP: localStorage.getItem('DATEGROUP.INTERSECTION_ROOT_TOP') || '150%',
+    INTERSECTION_ROOT_BOTTOM: localStorage.getItem('DATEGROUP.INTERSECTION_ROOT_BOTTOM') || '150%',
+  },
+  THUMBNAIL: {
+    PRIORITY: getNumber(localStorage.getItem('THUMBNAIL.PRIORITY'), 8),
+    INTERSECTION_ROOT_TOP: localStorage.getItem('THUMBNAIL.INTERSECTION_ROOT_TOP') || '250%',
+    INTERSECTION_ROOT_BOTTOM: localStorage.getItem('THUMBNAIL.INTERSECTION_ROOT_BOTTOM') || '250%',
+  },
+  IMAGE_THUMBNAIL: {
+    THUMBHASH_FADE_DURATION: getNumber(localStorage.getItem('THUMBHASH_FADE_DURATION'), 150),
+  },
+};
diff --git a/web/src/routes/(user)/+layout.svelte b/web/src/routes/(user)/+layout.svelte
index 23f38b86f4..bf24d0e7e4 100644
--- a/web/src/routes/(user)/+layout.svelte
+++ b/web/src/routes/(user)/+layout.svelte
@@ -1,10 +1,10 @@
 <script lang="ts">
   import UploadCover from '$lib/components/shared-components/drag-and-drop-upload-overlay.svelte';
   import { page } from '$app/stores';
-  import { assetViewingStore } from '$lib/stores/asset-viewing.store';
-  let { isViewing: showAssetViewer, setAsset } = assetViewingStore;
 
-  // This block takes care of opening the viewer.
+  import { assetViewingStore } from '$lib/stores/asset-viewing.store';
+  let { isViewing: showAssetViewer, setAsset, gridScrollTarget } = assetViewingStore;
+
   // $page.data.asset is loaded by route specific +page.ts loaders if that
   // route contains the assetId path.
   $: {
@@ -13,6 +13,8 @@
     } else {
       $showAssetViewer = false;
     }
+    const asset = $page.url.searchParams.get('at');
+    $gridScrollTarget = { at: asset };
   }
 </script>
 
diff --git a/web/src/routes/(user)/albums/[albumId=id]/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/albums/[albumId=id]/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 9e670f714c..ff5709df99 100644
--- a/web/src/routes/(user)/albums/[albumId=id]/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/albums/[albumId=id]/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -43,7 +43,13 @@
   import { downloadAlbum } from '$lib/utils/asset-utils';
   import { openFileUploadDialog } from '$lib/utils/file-uploader';
   import { handleError } from '$lib/utils/handle-error';
-  import { isAlbumsRoute, isPeopleRoute, isSearchRoute } from '$lib/utils/navigation';
+  import {
+    isAlbumsRoute,
+    isPeopleRoute,
+    isSearchRoute,
+    navigate,
+    type AssetGridRouteSearchParams,
+  } from '$lib/utils/navigation';
   import {
     AlbumUserRole,
     AssetOrder,
@@ -78,12 +84,15 @@
   import type { PageData } from './$types';
   import { dialogController } from '$lib/components/shared-components/dialog/dialog';
   import { t } from 'svelte-i18n';
+  import { onDestroy } from 'svelte';
 
   export let data: PageData;
 
-  let { isViewing: showAssetViewer, setAsset } = assetViewingStore;
+  let { isViewing: showAssetViewer, setAsset, gridScrollTarget } = assetViewingStore;
   let { slideshowState, slideshowNavigation } = slideshowStore;
 
+  let oldAt: AssetGridRouteSearchParams | null | undefined;
+
   $: album = data.album;
   $: albumId = album.id;
   $: albumKey = `${albumId}_${albumOrder}`;
@@ -244,7 +253,7 @@
     }
 
     if (viewMode === ViewMode.SELECT_ASSETS) {
-      handleCloseSelectAssets();
+      await handleCloseSelectAssets();
       return;
     }
     if (viewMode === ViewMode.LINK_SHARING) {
@@ -289,20 +298,37 @@
 
       timelineInteractionStore.clearMultiselect();
       viewMode = ViewMode.VIEW;
+      await navigate(
+        { targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget },
+        { replaceState: true, forceNavigate: true },
+      );
     } catch (error) {
       handleError(error, $t('errors.error_adding_assets_to_album'));
     }
   };
 
-  const handleCloseSelectAssets = () => {
+  const setModeToView = async () => {
     viewMode = ViewMode.VIEW;
+    assetStore.destroy();
+    assetStore = new AssetStore({ albumId, order: albumOrder });
+    timelineStore.destroy();
+    timelineStore = new AssetStore({ isArchived: false }, albumId);
+    await navigate(
+      { targetRoute: 'current', assetId: null, assetGridRouteSearchParams: { at: oldAt?.at } },
+      { replaceState: true, forceNavigate: true },
+    );
+    oldAt = null;
+  };
+
+  const handleCloseSelectAssets = async () => {
     timelineInteractionStore.clearMultiselect();
+    await setModeToView();
   };
 
   const handleSelectFromComputer = async () => {
     await openFileUploadDialog({ albumId: album.id });
     timelineInteractionStore.clearMultiselect();
-    viewMode = ViewMode.VIEW;
+    await setModeToView();
   };
 
   const handleAddUsers = async (albumUsers: AlbumUserAddDto[]) => {
@@ -400,6 +426,11 @@
       await deleteAlbum(album);
     }
   });
+
+  onDestroy(() => {
+    assetStore.destroy();
+    timelineStore.destroy();
+  });
 </script>
 
 <div class="flex overflow-hidden" bind:clientWidth={globalWidth}>
@@ -444,7 +475,14 @@
             {#if isEditor}
               <CircleIconButton
                 title={$t('add_photos')}
-                on:click={() => (viewMode = ViewMode.SELECT_ASSETS)}
+                on:click={async () => {
+                  viewMode = ViewMode.SELECT_ASSETS;
+                  oldAt = { at: $gridScrollTarget?.at };
+                  await navigate(
+                    { targetRoute: 'current', assetId: null, assetGridRouteSearchParams: { at: null } },
+                    { replaceState: true },
+                  );
+                }}
                 icon={mdiImagePlusOutline}
               />
             {/if}
@@ -530,12 +568,14 @@
       {#key albumKey}
         {#if viewMode === ViewMode.SELECT_ASSETS}
           <AssetGrid
+            enableRouting={false}
             assetStore={timelineStore}
             assetInteractionStore={timelineInteractionStore}
             isSelectionMode={true}
           />
         {:else}
           <AssetGrid
+            enableRouting={true}
             {album}
             {assetStore}
             {assetInteractionStore}
diff --git a/web/src/routes/(user)/archive/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/archive/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 6e3fb4cb28..2ce1309351 100644
--- a/web/src/routes/(user)/archive/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/archive/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -17,6 +17,7 @@
   import type { PageData } from './$types';
   import { mdiPlus, mdiDotsVertical } from '@mdi/js';
   import { t } from 'svelte-i18n';
+  import { onDestroy } from 'svelte';
 
   export let data: PageData;
 
@@ -25,6 +26,10 @@
   const { isMultiSelectState, selectedAssets } = assetInteractionStore;
 
   $: isAllFavorite = [...$selectedAssets].every((asset) => asset.isFavorite);
+
+  onDestroy(() => {
+    assetStore.destroy();
+  });
 </script>
 
 {#if $isMultiSelectState}
@@ -45,7 +50,7 @@
 {/if}
 
 <UserPageLayout hideNavbar={$isMultiSelectState} title={data.meta.title} scrollbar={false}>
-  <AssetGrid {assetStore} {assetInteractionStore} removeAction={AssetAction.UNARCHIVE}>
+  <AssetGrid enableRouting={true} {assetStore} {assetInteractionStore} removeAction={AssetAction.UNARCHIVE}>
     <EmptyPlaceholder text={$t('no_archived_assets_message')} slot="empty" />
   </AssetGrid>
 </UserPageLayout>
diff --git a/web/src/routes/(user)/favorites/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/favorites/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 49af165ac9..13e70c9161 100644
--- a/web/src/routes/(user)/favorites/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/favorites/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -19,6 +19,7 @@
   import type { PageData } from './$types';
   import { mdiDotsVertical, mdiPlus } from '@mdi/js';
   import { t } from 'svelte-i18n';
+  import { onDestroy } from 'svelte';
 
   export let data: PageData;
 
@@ -27,6 +28,10 @@
   const { isMultiSelectState, selectedAssets } = assetInteractionStore;
 
   $: isAllArchive = [...$selectedAssets].every((asset) => asset.isArchived);
+
+  onDestroy(() => {
+    assetStore.destroy();
+  });
 </script>
 
 <!-- Multiselection mode app bar -->
@@ -50,7 +55,7 @@
 {/if}
 
 <UserPageLayout hideNavbar={$isMultiSelectState} title={data.meta.title} scrollbar={false}>
-  <AssetGrid {assetStore} {assetInteractionStore} removeAction={AssetAction.UNFAVORITE}>
+  <AssetGrid enableRouting={true} {assetStore} {assetInteractionStore} removeAction={AssetAction.UNFAVORITE}>
     <EmptyPlaceholder text={$t('no_favorites_message')} slot="empty" />
   </AssetGrid>
 </UserPageLayout>
diff --git a/web/src/routes/(user)/map/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/map/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 3eb65ca1bd..0ea0ed18bb 100644
--- a/web/src/routes/(user)/map/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/map/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -124,7 +124,10 @@
           showNavigation={viewingAssets.length > 1}
           on:next={navigateNext}
           on:previous={navigatePrevious}
-          on:close={() => assetViewingStore.showAssetViewer(false)}
+          on:close={() => {
+            assetViewingStore.showAssetViewer(false);
+            handlePromiseError(navigate({ targetRoute: 'current', assetId: null }));
+          }}
           isShared={false}
         />
       {/await}
diff --git a/web/src/routes/(user)/partners/[userId]/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/partners/[userId]/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 83e2ba3c1f..b580c4faa5 100644
--- a/web/src/routes/(user)/partners/[userId]/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/partners/[userId]/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -23,6 +23,7 @@
 
   onDestroy(() => {
     assetInteractionStore.clearMultiselect();
+    assetStore.destroy();
   });
 </script>
 
@@ -45,5 +46,5 @@
       </svelte:fragment>
     </ControlAppBar>
   {/if}
-  <AssetGrid {assetStore} {assetInteractionStore} />
+  <AssetGrid enableRouting={true} {assetStore} {assetInteractionStore} />
 </main>
diff --git a/web/src/routes/(user)/people/[personId]/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/people/[personId]/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 02afe7f610..26e803deb6 100644
--- a/web/src/routes/(user)/people/[personId]/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/people/[personId]/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -52,7 +52,7 @@
     mdiEyeOutline,
     mdiPlus,
   } from '@mdi/js';
-  import { onMount } from 'svelte';
+  import { onDestroy, onMount } from 'svelte';
   import type { PageData } from './$types';
   import { listNavigation } from '$lib/actions/list-navigation';
   import { t } from 'svelte-i18n';
@@ -155,6 +155,7 @@
     }
     if (previousPersonId !== data.person.id) {
       handlePromiseError(updateAssetCount());
+      assetStore.destroy();
       assetStore = new AssetStore({
         isArchived: false,
         personId: data.person.id,
@@ -344,6 +345,10 @@
       await goto($page.url);
     }
   };
+
+  onDestroy(() => {
+    assetStore.destroy();
+  });
 </script>
 
 {#if viewMode === ViewMode.UNASSIGN_ASSETS}
@@ -442,6 +447,7 @@
 <main class="relative h-screen overflow-hidden bg-immich-bg tall:ml-4 pt-[var(--navbar-height)] dark:bg-immich-dark-bg">
   {#key refreshAssetGrid}
     <AssetGrid
+      enableRouting={true}
       {assetStore}
       {assetInteractionStore}
       isSelectionMode={viewMode === ViewMode.SELECT_PERSON}
diff --git a/web/src/routes/(user)/photos/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/photos/[[assetId=id]]/+page.svelte
index 85a497fa99..9bcbdbeea0 100644
--- a/web/src/routes/(user)/photos/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/photos/[[assetId=id]]/+page.svelte
@@ -24,6 +24,7 @@
   import { mdiDotsVertical, mdiPlus } from '@mdi/js';
   import { preferences, user } from '$lib/stores/user.store';
   import { t } from 'svelte-i18n';
+  import { onDestroy } from 'svelte';
 
   let { isViewing: showAssetViewer } = assetViewingStore;
   const assetStore = new AssetStore({ isArchived: false, withStacked: true, withPartners: true });
@@ -48,6 +49,10 @@
       return;
     }
   };
+
+  onDestroy(() => {
+    assetStore.destroy();
+  });
 </script>
 
 {#if $isMultiSelectState}
@@ -84,6 +89,7 @@
 
 <UserPageLayout hideNavbar={$isMultiSelectState} showUploadButton scrollbar={false}>
   <AssetGrid
+    enableRouting={true}
     {assetStore}
     {assetInteractionStore}
     removeAction={AssetAction.ARCHIVE}
diff --git a/web/src/routes/(user)/search/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/search/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 7e5362ba0f..cd4def1765 100644
--- a/web/src/routes/(user)/search/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/search/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -291,7 +291,7 @@
         <GalleryViewer
           assets={searchResultAssets}
           bind:selectedAssets
-          on:intersected={loadNextPage}
+          onIntersected={loadNextPage}
           showArchiveIcon={true}
           {viewport}
         />
diff --git a/web/src/routes/(user)/share/[key]/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/share/[key]/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 5ebb0e294c..f4fac282ba 100644
--- a/web/src/routes/(user)/share/[key]/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/share/[key]/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -11,8 +11,13 @@
   import type { PageData } from './$types';
   import { setSharedLink } from '$lib/utils';
   import { t } from 'svelte-i18n';
+  import { navigate } from '$lib/utils/navigation';
+  import { assetViewingStore } from '$lib/stores/asset-viewing.store';
+  import { tick } from 'svelte';
 
   export let data: PageData;
+
+  let { gridScrollTarget } = assetViewingStore;
   let { sharedLink, passwordRequired, sharedLinkKey: key, meta } = data;
   let { title, description } = meta;
   let isOwned = $user ? $user.id === sharedLink?.userId : false;
@@ -29,6 +34,11 @@
       description =
         sharedLink.description ||
         $t('shared_photos_and_videos_count', { values: { assetCount: sharedLink.assets.length } });
+      await tick();
+      await navigate(
+        { targetRoute: 'current', assetId: null, assetGridRouteSearchParams: $gridScrollTarget },
+        { forceNavigate: true, replaceState: true },
+      );
     } catch (error) {
       handleError(error, $t('errors.unable_to_get_shared_link'));
     }
diff --git a/web/src/routes/(user)/trash/[[photos=photos]]/[[assetId=id]]/+page.svelte b/web/src/routes/(user)/trash/[[photos=photos]]/[[assetId=id]]/+page.svelte
index 2907a542b3..27ad5bb3f0 100644
--- a/web/src/routes/(user)/trash/[[photos=photos]]/[[assetId=id]]/+page.svelte
+++ b/web/src/routes/(user)/trash/[[photos=photos]]/[[assetId=id]]/+page.svelte
@@ -25,6 +25,7 @@
   import { handlePromiseError } from '$lib/utils';
   import { dialogController } from '$lib/components/shared-components/dialog/dialog';
   import { t } from 'svelte-i18n';
+  import { onDestroy } from 'svelte';
 
   export let data: PageData;
 
@@ -84,6 +85,10 @@
       handleError(error, $t('errors.unable_to_restore_trash'));
     }
   };
+
+  onDestroy(() => {
+    assetStore.destroy();
+  });
 </script>
 
 {#if $isMultiSelectState}
@@ -111,7 +116,7 @@
       </LinkButton>
     </div>
 
-    <AssetGrid {assetStore} {assetInteractionStore}>
+    <AssetGrid enableRouting={true} {assetStore} {assetInteractionStore}>
       <p class="font-medium text-gray-500/60 dark:text-gray-300/60 p-4">
         {$t('trashed_items_will_be_permanently_deleted_after', { values: { days: $serverConfig.trashDays } })}
       </p>
diff --git a/web/static/dark_skeleton.png b/web/static/dark_skeleton.png
new file mode 100644
index 0000000000..2a115a8496
Binary files /dev/null and b/web/static/dark_skeleton.png differ
diff --git a/web/static/light_skeleton.png b/web/static/light_skeleton.png
new file mode 100644
index 0000000000..22c7eae754
Binary files /dev/null and b/web/static/light_skeleton.png differ