v1.1.0:7 — exercise-history popup auto-loads on scroll
The popup HAD an IntersectionObserver-based infinite scroll (since v1.0.0:6 alongside the main workout-history page), but the observer was unreliable inside an `absolute`-positioned scroll container with a small 60px rootMargin. It often didn't fire at all — leaving the user with a popup that scrolled internally but never fetched more data even when hundreds of history entries existed server-side. Fix: replace IntersectionObserver with a plain `scroll` event listener on the popup. Fires whenever the user scrolls within 300px of the bottom (matching WorkoutsList's lookahead on the main page). Also runs once on mount in case the first page doesn't fill the popup. Bottom status row now shows "Loading more..." / "Scroll to load more" / "End of history" so the user has feedback on state. No schema, no API, no data.
This commit is contained in:
@@ -58,44 +58,61 @@ function ExerciseHistoryPopup({
|
||||
};
|
||||
}, [exerciseId]);
|
||||
|
||||
// Infinite scroll — observe a sentinel below the rendered list. The
|
||||
// root is the popup's scroll container (the popup itself), not the
|
||||
// viewport, since the user scrolls inside the popup.
|
||||
// v1.1.0:7 — Infinite scroll via a plain scroll listener on the
|
||||
// popup itself. The previous IntersectionObserver implementation was
|
||||
// unreliable inside an absolute-positioned scroll container (the
|
||||
// popup is `position: absolute` + `overflow-y-auto`, which some
|
||||
// browsers don't observe consistently when the root is the same
|
||||
// element). A `scroll` event on the popup is rock-solid.
|
||||
//
|
||||
// Fires whenever the user scrolls within ~300px of the popup's
|
||||
// bottom edge, mirroring the rootMargin used by the workouts-list
|
||||
// infinite-scroll on the main page.
|
||||
//
|
||||
// Also runs once on first render after history loads — important
|
||||
// because if the user has 100+ history entries and the first page
|
||||
// doesn't fill the popup OR the user opens the popup and immediately
|
||||
// sees content without scrolling, we still want to fetch ahead.
|
||||
useEffect(() => {
|
||||
if (loading || !hasMore || !sentinelRef.current || !popupRef.current) {
|
||||
return;
|
||||
}
|
||||
const sentinel = sentinelRef.current;
|
||||
const root = popupRef.current;
|
||||
const observer = new IntersectionObserver(
|
||||
(entries) => {
|
||||
if (!entries[0]?.isIntersecting) return;
|
||||
if (loadingMore || !hasMore) return;
|
||||
setLoadingMore(true);
|
||||
(async () => {
|
||||
try {
|
||||
const res = await fetch(
|
||||
`/api/exercises/${exerciseId}?offset=${history.length}&limit=${HISTORY_PAGE_SIZE}`,
|
||||
);
|
||||
if (res.ok) {
|
||||
const data = await res.json();
|
||||
setHistory((prev) => [...prev, ...(data.history || [])]);
|
||||
setHasMore(!!data.hasMore);
|
||||
} else {
|
||||
setError(`Failed to load more (${res.status})`);
|
||||
setHasMore(false);
|
||||
}
|
||||
} catch {
|
||||
setError("Failed to load more");
|
||||
setHasMore(false);
|
||||
}
|
||||
setLoadingMore(false);
|
||||
})();
|
||||
},
|
||||
{ root, rootMargin: "60px" },
|
||||
);
|
||||
observer.observe(sentinel);
|
||||
return () => observer.disconnect();
|
||||
if (loading || !hasMore || loadingMore || !popupRef.current) return;
|
||||
const el = popupRef.current;
|
||||
|
||||
const loadMore = async () => {
|
||||
if (loadingMore || !hasMore) return;
|
||||
setLoadingMore(true);
|
||||
try {
|
||||
const res = await fetch(
|
||||
`/api/exercises/${exerciseId}?offset=${history.length}&limit=${HISTORY_PAGE_SIZE}`,
|
||||
);
|
||||
if (res.ok) {
|
||||
const data = await res.json();
|
||||
setHistory((prev) => [...prev, ...(data.history || [])]);
|
||||
setHasMore(!!data.hasMore);
|
||||
} else {
|
||||
setError(`Failed to load more (${res.status})`);
|
||||
setHasMore(false);
|
||||
}
|
||||
} catch {
|
||||
setError("Failed to load more");
|
||||
setHasMore(false);
|
||||
}
|
||||
setLoadingMore(false);
|
||||
};
|
||||
|
||||
const maybeLoad = () => {
|
||||
const { scrollTop, scrollHeight, clientHeight } = el;
|
||||
// 300px lookahead — match WorkoutsList's rootMargin behavior.
|
||||
if (scrollHeight - scrollTop - clientHeight < 300) {
|
||||
loadMore();
|
||||
}
|
||||
};
|
||||
|
||||
el.addEventListener("scroll", maybeLoad, { passive: true });
|
||||
// Initial check: if the loaded page doesn't fill the popup, we
|
||||
// still want to fetch the next page so the user doesn't have to
|
||||
// scroll a near-empty container before more arrives.
|
||||
maybeLoad();
|
||||
return () => el.removeEventListener("scroll", maybeLoad);
|
||||
}, [loading, hasMore, loadingMore, history.length, exerciseId]);
|
||||
|
||||
// Close on outside click
|
||||
@@ -167,10 +184,21 @@ function ExerciseHistoryPopup({
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
{/* Sentinel + status row at the bottom of the list */}
|
||||
{/* Status row at the bottom of the list. The sentinel ref
|
||||
is no longer the load trigger (we use a scroll listener
|
||||
on the popup itself in v1.1.0:7), but the visual marker
|
||||
still tells the user whether more is loading or done. */}
|
||||
<div ref={sentinelRef} className="flex justify-center py-2">
|
||||
{loadingMore && (
|
||||
<Loader className="w-3.5 h-3.5 animate-spin text-zinc-500" />
|
||||
<span className="inline-flex items-center gap-2 text-[10px] text-zinc-500 uppercase tracking-wider">
|
||||
<Loader className="w-3.5 h-3.5 animate-spin" />
|
||||
Loading more...
|
||||
</span>
|
||||
)}
|
||||
{!loadingMore && hasMore && (
|
||||
<span className="text-[10px] text-zinc-600 uppercase tracking-wider">
|
||||
Scroll to load more
|
||||
</span>
|
||||
)}
|
||||
{!loadingMore && !hasMore && history.length >= HISTORY_PAGE_SIZE && (
|
||||
<span className="text-[10px] text-zinc-600 uppercase tracking-wider">
|
||||
|
||||
Reference in New Issue
Block a user