feat(timeline): start at today + infinite scroll into the past #65

Merged
Hartmut merged 7 commits from feature/timeline-past-scroll into main 2026-05-22 11:43:08 +02:00
Showing only changes of commit 0e9d6ec388 - Show all commits
@@ -689,15 +689,25 @@ function TimelineViewContent({
const pendingLeftCompensationPx = useRef(0); const pendingLeftCompensationPx = useRef(0);
// Flag: scroll viewport to today after the next viewStart-driven re-layout. // Flag: scroll viewport to today after the next viewStart-driven re-layout.
const pendingScrollToTodayRef = useRef(false); const pendingScrollToTodayRef = useRef(false);
// Guard reset on every real unmount (including Strict Mode fake-unmount) so the
// Scroll to today on mount so the viewport opens with today at the left edge. // scroll-to-today fires correctly on remount.
// Empty deps: intentionally runs once (and twice in React Strict Mode dev, both correct). const hasScrolledToTodayOnLoad = useRef(false);
useLayoutEffect(() => { useLayoutEffect(() => {
return () => {
hasScrolledToTodayOnLoad.current = false;
};
}, []);
// Scroll to today the first time the canvas has its full width (after initial data load).
// Depends on totalCanvasWidth so it fires after isInitialLoading → false renders the canvas.
useLayoutEffect(() => {
if (totalCanvasWidth === 0) return;
if (hasScrolledToTodayOnLoad.current) return;
const el = scrollContainerRef.current; const el = scrollContainerRef.current;
if (!el) return; if (!el) return;
el.scrollLeft = toLeft(today); el.scrollLeft = toLeft(today);
}, []); hasScrolledToTodayOnLoad.current = true;
}, [totalCanvasWidth, toLeft, today]);
// Apply scroll compensation synchronously after the canvas grows (left-extend or Today button). // Apply scroll compensation synchronously after the canvas grows (left-extend or Today button).
useLayoutEffect(() => { useLayoutEffect(() => {