Skip to content
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 3 additions & 2 deletions src/lib/components/Devlog.svelte
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,8 @@
showModifyButtons,
allowDelete = true,
projectName = null,
user = null
user = null,
show3DPreview = true
} = $props();
</script>

Expand Down Expand Up @@ -52,7 +53,7 @@
/>
</div>
</div>
{#if devlog.model}
{#if devlog.model && show3DPreview}
<div
class="relative max-h-100 w-full grow border-3 border-primary-900 lg:w-100 lg:max-w-[60%]"
>
Expand Down
61 changes: 41 additions & 20 deletions src/routes/dashboard/explore/+page.svelte
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
<script lang="ts">
import { onMount } from 'svelte';
import { onMount, tick } from 'svelte';
import Devlog from '$lib/components/Devlog.svelte';
import Head from '$lib/components/Head.svelte';

Expand All @@ -14,6 +14,10 @@
let observer: IntersectionObserver | null = null;
const rootMargin = '320px 0px';

let devlogRefs = $state<(HTMLDivElement | null)[]>([]);
let previewVisibility = $state(new Array(data.devlogs.length).fill(true));
const PREVIEW_UNMOUNT_MARGIN = 1200;

function hydrateDevlogs(rawDevlogs: typeof data.devlogs) {
return rawDevlogs.map((entry) => ({
...entry,
Expand All @@ -24,26 +28,35 @@
}));
}

function updatePreviewVisibility() {
previewVisibility = devlogRefs.map((ref) => {
if (!ref) return false;
const rect = ref.getBoundingClientRect();
return (
rect.bottom > -PREVIEW_UNMOUNT_MARGIN &&
rect.top < window.innerHeight + PREVIEW_UNMOUNT_MARGIN
);
});
}

async function loadMoreDevlogs() {
if (loadingMore || !hasMore) return;

loadingMore = true;
loadError = '';

try {
const params = new URLSearchParams({ offset: `${nextOffset}` });
const response = await fetch(`/dashboard/explore?${params.toString()}`);

if (!response.ok) {
throw new Error('Failed to load more devlogs');
}

const payload = await response.json();
const incoming = hydrateDevlogs(payload.devlogs ?? []);

devlogs = [...devlogs, ...incoming];
previewVisibility = [...previewVisibility, ...new Array(incoming.length).fill(false)];
nextOffset = payload.nextOffset ?? nextOffset + incoming.length;
hasMore = Boolean(payload.hasMore);
await tick();
updatePreviewVisibility();
} catch (error) {
console.error(error);
loadError = 'Could not load more right now.';
Expand All @@ -53,6 +66,11 @@
}

onMount(() => {
const onScroll = () => updatePreviewVisibility();
window.addEventListener('scroll', onScroll, { passive: true });
window.addEventListener('resize', onScroll);
tick().then(updatePreviewVisibility);

if (!hasMore || !sentinel) return;

observer = new IntersectionObserver(
Expand All @@ -68,7 +86,11 @@

observer.observe(sentinel);

return () => observer?.disconnect();
return () => {
window.removeEventListener('scroll', onScroll);
window.removeEventListener('resize', onScroll);
observer?.disconnect();
};
});
</script>

Expand All @@ -87,23 +109,23 @@
/>
</div>
{:else}
{#each devlogs as devlog (devlog.devlog.id)}
<Devlog
devlog={devlog.devlog}
projectId={devlog.project.id}
projectName={devlog.project.name}
user={devlog.user}
showModifyButtons={false}
allowDelete={false}
/>
{#each devlogs as devlog, i (devlog.devlog.id)}
<div bind:this={devlogRefs[i]}>
<Devlog
devlog={devlog.devlog}
projectId={devlog.project.id}
projectName={devlog.project.name}
user={devlog.user}
showModifyButtons={false}
allowDelete={false}
show3DPreview={previewVisibility[i]}
/>
</div>
{/each}

<div bind:this={sentinel} class="h-1 w-full"></div>

{#if loadingMore}
<p class="text-sm opacity-70">Loading more...</p>
{/if}

{#if loadError}
<div class="flex items-center gap-2 text-sm text-red-300">
<span>{loadError}</span>
Expand All @@ -112,7 +134,6 @@
</button>
</div>
{/if}

{#if !hasMore && !loadingMore}
<p class="text-center text-sm opacity-70">You're caught up.</p>
{/if}
Expand Down