Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

[BUG] Setting initial scrolling position is racy #883

Closed
vyivel opened this issue Apr 16, 2023 · 7 comments
Closed

[BUG] Setting initial scrolling position is racy #883

vyivel opened this issue Apr 16, 2023 · 7 comments
Labels
bug Something isn't working released

Comments

@vyivel
Copy link

vyivel commented Apr 16, 2023

Describe the bug

Occasionally initialScrollTop and initialTopMostItemIndex have no effect, presumably due to the scrolling being applied before the scrollHeight of the element is updated. This is easier to reproduce with multiple lists.

Related: #874

Reproduction

https://codesandbox.io/p/sandbox/laughing-jang-cycy4n

To Reproduce

Just load the page. Depending on the PC specs and the browser used, this might take multiple reloads or not work at all.

Expected behavior

Screenshots

Firefox 111.0.1: some of the first lists aren't scrolled.
image

When initialScrollTop is used, none of the lists are scrolled.
image

Chromium 111.0.5563.146: seems to work fine with initialTopMostItemIndex.
image

When initialScrollTop is used, only the first list isn't scrolled; this behavior is consistent across reloads.
image

Desktop

  • OS: Fedora 37
  • Browser: multiple (see above)

Additional context

I can't reproduce this in qutebrowser.


With the following patch:

diff --git a/src/hooks/useChangedChildSizes.ts b/src/hooks/useChangedChildSizes.ts
index a89de52a..9a53faa6 100644
--- a/src/hooks/useChangedChildSizes.ts
+++ b/src/hooks/useChangedChildSizes.ts
@@ -42,6 +42,10 @@ export default function useChangedListContentsSizes(
         ? window.innerHeight
         : scrollableElement.offsetHeight
 
+      console.log(`useChangedListContentsSize:
+        scrollTop = ${scrollTop},
+        scrollHeight = ${scrollHeight},
+        ranges = ${ranges != null ? (ranges.length == 0 ? "<empty>" :ranges.map((sr) => `${sr.startIndex}, ${sr.endIndex}, ${sr.size}`).join("; ")) : "null"}`);
       scrollContainerStateCallback({
         scrollTop: Math.max(scrollTop, 0),
         scrollHeight,
diff --git a/src/hooks/useScrollTop.ts b/src/hooks/useScrollTop.ts
index a415cfce..dbff3204 100644
--- a/src/hooks/useScrollTop.ts
+++ b/src/hooks/useScrollTop.ts
@@ -96,6 +96,13 @@ export default function useScrollTop(
     // avoid system hanging because the DOM never called back
     // with the scrollTop
     // scroller is already at this location
+    console.log(`scrollToCallback:
+      offsetHeight = ${offsetHeight},
+      scrollHeight = ${scrollHeight},
+      approximatelyEqual(offsetHeight, scrollHeight) = ${approximatelyEqual(offsetHeight, scrollHeight)},
+      location.top = ${location.top},
+      scrollTop = ${scrollTop},
+      (location.top === scrollTop) = ${location.top === scrollTop}`)
     if (approximatelyEqual(offsetHeight, scrollHeight) || location.top === scrollTop) {
       scrollContainerStateCallback({ scrollTop, scrollHeight, viewportHeight: offsetHeight })
       if (isSmooth) {

The result is the following; the logs suggest that scrollToCallback() is called too early, before the proper scrollHeight is set.

image


Please tell if you need more information.

@vyivel vyivel added the bug Something isn't working label Apr 16, 2023
@petyosi
Copy link
Owner

petyosi commented Apr 16, 2023

Confirming, I managed to reproduce on my side. Will look into it and update the issue, thanks for looking into the source code for the additional hints.

@github-actions
Copy link
Contributor

🎉 This issue has been resolved in version 4.2.2 🎉

The release is available on:

Your semantic-release bot 📦🚀

@vyivel
Copy link
Author

vyivel commented Apr 18, 2023

initialScrollTop seems to be fixed with 42c8a03, but initialTopMostItemIndex is still racy in Firefox.

@petyosi
Copy link
Owner

petyosi commented Apr 18, 2023

@vyivel so these are actually two issues. initialScrollTop and initialTopMostItemIndex follow a separate implementation logic, and I strongly recommend against using initialScrollTop (discussed in some issues here).

  • the initialScrollTop was consistently broken across browsers. I was blind-sided as the test case scenario had another flag set that made it work there. The fix above identified the problem and should address it.

  • the initialTopMostItemIndex in firefox seems to be some sort of a heisenbug. I see it rarely in the sandbox, but when I took your sandbox and added it to the playground, I can't seem to reproduce the problem locally.

If there's anything that can make the problem consistent and reproducible in the local environment, let me know.

petyosi added a commit that referenced this issue Apr 21, 2023
Addresses a flicker with alignToBottom and React 18. Fixes #883
@github-actions
Copy link
Contributor

🎉 This issue has been resolved in version 4.3.0 🎉

The release is available on:

Your semantic-release bot 📦🚀

@petyosi
Copy link
Owner

petyosi commented Apr 21, 2023

@vyivel I reproduced (and, hopefully, fixed) that while working on another thing. Please let me know if you still experience it.

@vyivel
Copy link
Author

vyivel commented Apr 21, 2023

Tried different cases with 4.3.0, everything works as expected, so the problem seems to be fixed. Thanks for your work!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
bug Something isn't working released
Projects
None yet
Development

No branches or pull requests

2 participants