-
-
Save claus/992a5596d6532ac91b24abe24e10ae81 to your computer and use it in GitHub Desktop.
import useScrollRestoration from "utils/hooks/useScrollRestoration"; | |
const App = ({ Component, pageProps, router }) => { | |
useScrollRestoration(router); | |
return <Component {...pageProps} />; | |
}; | |
export default App; |
import { useEffect } from 'react'; | |
import Router from 'next/router'; | |
function saveScrollPos(url) { | |
const scrollPos = { x: window.scrollX, y: window.scrollY }; | |
sessionStorage.setItem(url, JSON.stringify(scrollPos)); | |
} | |
function restoreScrollPos(url) { | |
const scrollPos = JSON.parse(sessionStorage.getItem(url)); | |
if (scrollPos) { | |
window.scrollTo(scrollPos.x, scrollPos.y); | |
} | |
} | |
export default function useScrollRestoration(router) { | |
useEffect(() => { | |
if ('scrollRestoration' in window.history) { | |
let shouldScrollRestore = false; | |
window.history.scrollRestoration = 'manual'; | |
restoreScrollPos(router.asPath); | |
const onBeforeUnload = event => { | |
saveScrollPos(router.asPath); | |
delete event['returnValue']; | |
}; | |
const onRouteChangeStart = () => { | |
saveScrollPos(router.asPath); | |
}; | |
const onRouteChangeComplete = url => { | |
if (shouldScrollRestore) { | |
shouldScrollRestore = false; | |
restoreScrollPos(url); | |
} | |
}; | |
window.addEventListener('beforeunload', onBeforeUnload); | |
Router.events.on('routeChangeStart', onRouteChangeStart); | |
Router.events.on('routeChangeComplete', onRouteChangeComplete); | |
Router.beforePopState(() => { | |
shouldScrollRestore = true; | |
return true; | |
}); | |
return () => { | |
window.removeEventListener('beforeunload', onBeforeUnload); | |
Router.events.off('routeChangeStart', onRouteChangeStart); | |
Router.events.off('routeChangeComplete', onRouteChangeComplete); | |
Router.beforePopState(() => true); | |
}; | |
} | |
}, [router]); | |
} |
this is great, thanks!
It works fine for me
Thank you for this resource
The hook version from @MichalSzorad seems to work fairly nicely.
I'm not sure if there's a possible solution, that would be worth the complexity, but one scenario I don't see addressed is as follows:
- Scroll down page A click on link to page B
- On page B click on a link that goes back to page A
- We land and stay at the top of page A
- Click back button twice, scroll is restored to the scrolled down positions, as it was on first view of page A
- Click forward button twice, scroll is again restored to the scrolled down position, even though we had not originally scrolled this second instance of Page A.
How crazy/possible is it to somehow record the scroll position for each pop state version of a url? If we don't, is it better to reset a saved scroll position after arriving to the page via a regular link again?
I encountered an issue with the latest version of Next.js (v15) where the router events are not supported as expected. If you're experiencing similar problems, you might find this component helpful.
This method uses replaceState to store the position within the history API, avoiding reliance on Next.js APIs. As a result, it may also work on older versions, depending on their implementation.
Check out the component here: gist link.
The hook version from @MichalSzorad seems to work fairly nicely.
I'm not sure if there's a possible solution, that would be worth the complexity, but one scenario I don't see addressed is as follows:
- Scroll down page A click on link to page B
- On page B click on a link that goes back to page A
- We land and stay at the top of page A
- Click back button twice, scroll is restored to the scrolled down positions, as it was on first view of page A
- Click forward button twice, scroll is again restored to the scrolled down position, even though we had not originally scrolled this second instance of Page A.
How crazy/possible is it to somehow record the scroll position for each pop state version of a url? If we don't, is it better to reset a saved scroll position after arriving to the page via a regular link again?
You may want to try the component I provided as it should not have any issues with rapid navigation, or going to urls which were previously visited.
Okay I've just got done doing tons of testing and the solution I've landed on is to use NextJS's experimental support.
experimental: {
scrollRestoration: true,
},
However to fix iOS snapshots you must wrap your app with this:
useEffect(() => {
window.history.scrollRestoration = 'auto'
}, [router])
This only works for pages that are rendered instantly. If you are loading data or the page isn't ready, than you'll need to use one of the solutions above. But make sure you modify it to set scrollRestoration to "auto" so iOS snapshots continue to function normally.
I'm back again. I updated and simplified the above scripts pretty drastically. I'm also using window.history.state.key to track positions, so it supports going to the same page throughout history and always restoring the correct scroll position. This also fixes iOS Snapshots, since I don't set window.scrollRestoration to manual.
I also allow you to set a selector for an element to restore scroll to, for my website I have an "overflow-y-auto" child element that I need scrollRestoration for, so window won't do the trick. Here is my updated, versatile script for 2024 lol.
I also added an option to add a delay if you need to, and I moved router into the options, it will default to useRouter if you don't provide one.
useScrollRestoration.js
import { useRouter } from 'next/router'
import { useEffect, useState } from 'react'
export const saveScrollPos = (key, selector) => {
let scrollPos = { x: window.scrollX, y: window.scrollY }
if (selector) {
const element = document.querySelector(selector)
scrollPos = { x: element.scrollLeft, y: element.scrollTop }
}
sessionStorage.setItem(`scrollPos:${key}`, JSON.stringify(scrollPos))
}
export const restoreScrollPos = (key, selector) => {
const json = sessionStorage.getItem(`scrollPos:${key}`)
const scrollPos = json ? JSON.parse(json) : { x: 0, y: 0 }
if (selector) {
const element = document.querySelector(selector)
element.scrollTo(scrollPos.x, scrollPos.y)
} else {
window.scrollTo(scrollPos.x, scrollPos.y)
}
}
export const deleteScrollPos = (key) => {
sessionStorage.removeItem(`scrollPos:${key}`)
}
export function useScrollRestoration({ router = null, enabled = true, selector = null, delay = null } = {}) {
const nextRouter = useRouter()
if (!router) {
router = nextRouter
}
const [key, setKey] = useState(null)
useEffect(() => {
setKey(window.history.state.key)
}, [])
useEffect(() => {
if (!enabled) return
const onBeforeUnload = () => {
deleteScrollPos(key)
deleteScrollPos(window.history.state.key)
}
const onRouteChangeStart = () => {
saveScrollPos(key, selector)
}
const onRouteChangeComplete = () => {
setKey(window.history.state.key)
if (delay != null) {
setTimeout(() => {
restoreScrollPos(window.history.state.key, selector, delay)
deleteScrollPos(window.history.state.key)
}, delay)
} else {
restoreScrollPos(window.history.state.key, selector, delay)
deleteScrollPos(window.history.state.key)
}
}
window.addEventListener('beforeunload', onBeforeUnload)
router.events.on('routeChangeStart', onRouteChangeStart)
router.events.on('routeChangeComplete', onRouteChangeComplete)
return () => {
window.removeEventListener('beforeunload', onBeforeUnload)
router.events.off('routeChangeStart', onRouteChangeStart)
router.events.off('routeChangeComplete', onRouteChangeComplete)
}
}, [enabled, key, selector])
}
Default setup:
useScrollRestoration()
Scroll child element <main>
with overflow:
useScrollRestoration(router, { selector: "main" })
Delay Example:
useScrollRestoration(router, { delay: 100 })
@LeakedDave is this supposed to work with app router ?
I've experienced a few issues with the suggested solutions, so I've built this library https://www.npmjs.com/package/next-restore-scroll-position which fixes the following.
useEffect
was called multiple times (my repo uses useRef instead of local variableshouldScrollRestore
to solve the issue.useRef
is also better thanuseState
becauseuseRef
does not rerender components after an update)Example
👍 ✅