From 821400e846ec2c2c787245e430bae0769921405f Mon Sep 17 00:00:00 2001 From: Himanshu Aggarwal <77129877+aggarwal-h@users.noreply.github.com> Date: Tue, 31 Dec 2024 22:59:03 -0500 Subject: [PATCH] Fix smooth scroll to index on SSR hydration (#591) * Fix smooth scroll issue on SSR hydration * Update SSR story to include scroll on hydration * Simplify hydration options spacing with flex * Fix issue with smooth scrolling after hydration * Add test to check smooth scrolling after hydration * Fix miscs --------- Co-authored-by: inokawa <48897392+inokawa@users.noreply.github.com> --- e2e/VList.spec.ts | 111 ++++++++++++++---------- src/core/store.ts | 3 +- stories/react/advanced/SSR.stories.tsx | 114 ++++++++++++++++++++++++- 3 files changed, 179 insertions(+), 49 deletions(-) diff --git a/e2e/VList.spec.ts b/e2e/VList.spec.ts index 06f446460..236a79b41 100644 --- a/e2e/VList.spec.ts +++ b/e2e/VList.spec.ts @@ -1240,51 +1240,72 @@ test.describe("RTL", () => { }); }); -test("SSR and hydration", async ({ page }) => { - await page.goto(storyUrl("advanced-ssr--default")); - - const component = await getScrollable(page); - - const first = await getFirstItem(component); - const last = await getLastItem(component); - - // check if SSR suceeded - const itemsSelector = '*[style*="top"]'; - const items = component.locator(itemsSelector); - const initialLength = await items.count(); - expect(initialLength).toBeGreaterThanOrEqual(30); - expect(await items.first().textContent()).toEqual("0"); - expect(await items.last().textContent()).toEqual(String(initialLength - 1)); - // check if items have styles for SSR - expect(await items.first().evaluate((e) => e.style.position)).not.toBe( - "absolute" - ); - - // should not change state with scroll before hydration - await component.evaluate((e) => e.scrollTo({ top: 1000 })); - expect(initialLength).toBe(await component.locator(itemsSelector).count()); - await page.waitForTimeout(500); - await component.evaluate((e) => e.scrollTo({ top: 0 })); - - // hydrate - await page.getByRole("button", { name: "hydrate" }).click(); - - // check if hydration suceeded but state is not changed - const hydratedItemsLength = await component.locator(itemsSelector).count(); - expect(hydratedItemsLength).toBe(initialLength); - expect((await getFirstItem(component)).top).toBe(first.top); - expect((await getLastItem(component)).bottom).toBe(last.bottom); - // check if items do not have styles for SSR - expect(await items.first().evaluate((e) => e.style.position)).toBe( - "absolute" - ); - - // should change state with scroll after hydration - await component.evaluate((e) => e.scrollTo({ top: 1000 })); - await page.waitForTimeout(500); - expect(await component.locator(itemsSelector).count()).not.toBe( - initialLength - ); +test.describe("SSR and hydration", () => { + test("check if hydration works", async ({ page }) => { + await page.goto(storyUrl("advanced-ssr--default")); + + const component = await getScrollable(page); + + const first = await getFirstItem(component); + const last = await getLastItem(component); + + // check if SSR suceeded + const itemsSelector = '*[style*="top"]'; + const items = component.locator(itemsSelector); + const initialLength = await items.count(); + expect(initialLength).toBeGreaterThanOrEqual(30); + expect(await items.first().textContent()).toEqual("0"); + expect(await items.last().textContent()).toEqual(String(initialLength - 1)); + // check if items have styles for SSR + expect(await items.first().evaluate((e) => e.style.position)).not.toBe( + "absolute" + ); + + // should not change state with scroll before hydration + await component.evaluate((e) => e.scrollTo({ top: 1000 })); + expect(initialLength).toBe(await component.locator(itemsSelector).count()); + await page.waitForTimeout(500); + await component.evaluate((e) => e.scrollTo({ top: 0 })); + + // hydrate + await page.getByRole("button", { name: "hydrate" }).click(); + + // check if hydration suceeded but state is not changed + const hydratedItemsLength = await component.locator(itemsSelector).count(); + expect(hydratedItemsLength).toBe(initialLength); + expect((await getFirstItem(component)).top).toBe(first.top); + expect((await getLastItem(component)).bottom).toBe(last.bottom); + // check if items do not have styles for SSR + expect(await items.first().evaluate((e) => e.style.position)).toBe( + "absolute" + ); + + // should change state with scroll after hydration + await component.evaluate((e) => e.scrollTo({ top: 1000 })); + await page.waitForTimeout(500); + expect(await component.locator(itemsSelector).count()).not.toBe( + initialLength + ); + }); + + test("check if smooth scrolling works after hydration", async ({ page }) => { + await page.goto(storyUrl("advanced-ssr--scroll-to")); + + const component = await getScrollable(page); + + // turn scroll to index with smooth on + await page.getByRole("checkbox", { name: "scroll to index" }).check(); + await page.getByRole("checkbox", { name: "smooth" }).check(); + + // set scroll index to 100 + await page.locator("input[type=number]").fill("100"); + + // hydrate + await page.getByRole("button", { name: "hydrate" }).click(); + + await page.waitForTimeout(1000); + expect((await getFirstItem(component)).text).toEqual("100"); + }); }); test.describe("emulated iOS WebKit", () => { diff --git a/src/core/store.ts b/src/core/store.ts index d765be26f..5ceabaca9 100644 --- a/src/core/store.ts +++ b/src/core/store.ts @@ -334,7 +334,8 @@ export const createVirtualStore = ( _scrollMode === SCROLL_BY_SHIFT || (_frozenRange ? // https://github.com/inokawa/virtua/issues/380 - index < _frozenRange[0] + // https://github.com/inokawa/virtua/issues/590 + !isSSR && index < _frozenRange[0] : // Otherwise we should maintain visible position getItemOffset(index) + // https://github.com/inokawa/virtua/issues/385 diff --git a/stories/react/advanced/SSR.stories.tsx b/stories/react/advanced/SSR.stories.tsx index 4444e4e93..5428c8f4a 100644 --- a/stories/react/advanced/SSR.stories.tsx +++ b/stories/react/advanced/SSR.stories.tsx @@ -1,6 +1,6 @@ import { Meta, StoryObj } from "@storybook/react"; -import React, { useLayoutEffect, useRef, useState } from "react"; -import { VList } from "../../../src"; +import React, { useEffect, useLayoutEffect, useRef, useState } from "react"; +import { VList, type VListHandle } from "../../../src"; import { hydrateRoot } from "react-dom/client"; import { renderToString } from "react-dom/server"; @@ -32,7 +32,6 @@ const App = () => { }; export const Default: StoryObj = { - name: "SSR", render: () => { const [hydrated, setHydrated] = useState(false); const ref = useRef(null); @@ -66,3 +65,112 @@ export const Default: StoryObj = { ); }, }; + +const AppScrollOnMount = ({ + scrollOnMount, + scrollToIndex, + smooth, +}: { + scrollOnMount?: boolean; + scrollToIndex?: number; + smooth?: boolean; +}) => { + const ref = useRef(null); + useEffect(() => { + if (!ref.current || !scrollOnMount || !scrollToIndex) return; + + ref.current.scrollToIndex(scrollToIndex, { + smooth: smooth, + }); + }, []); + + const COUNT = 10000; + return ( + <> + + {createRows(COUNT)} + + + ); +}; + +export const ScrollTo: StoryObj = { + render: () => { + const [scrollOnMount, setScrollOnMount] = useState(false); + const [scrollIndex, setScrollIndex] = useState(100); + const [smooth, setSmooth] = useState(true); + const [hydrated, setHydrated] = useState(false); + const ref = useRef(null); + + useLayoutEffect(() => { + if (!ref.current) return; + + if (!hydrated) { + ref.current.innerHTML = renderToString(); + } else { + hydrateRoot( + ref.current, + + ); + } + }, [hydrated]); + + return ( +
+
+ +
+ + + { + setScrollIndex(Number(e.target.value)); + }} + /> + +
+
+
+
+ ); + }, +};