autoscroll
Module emulates native browser scroll behavior on SPA-navigations:
- scroll to top of the page if there is no hash in the URL
- scroll to the element with id that equals to the hash in the URL if it exists
- restore scroll position on back/forward navigations
The behaviour is similar to the react-router
Installation
First install @tramvai/module-autoscroll:
- npm
- Yarn
yarn add @tramvai/module-autoscroll
yarn add @tramvai/module-autoscroll
And add ScrollRestorationModule to the modules list:
import { createApp } from '@tramvai/core';
import { ScrollRestorationModule } from '@tramvai/module-autoscroll';
createApp({
name: 'tincoin',
modules: [ScrollRestorationModule],
});
Explanation
Behavior
behavior: smooth is not supported by every browser (e.g. doesn't work in Safari). In this case you can use polyfill smoothscroll-polyfill that you should add to your app.
Scroll restoration
Native browser scroll restoration is disabled when autoscroll is enabled (window.history.scrollRestoration = 'manual', depending on the navigateState.disableAutoscroll parameter or AUTOSCROLL_DISABLED_TOKEN token).
Module saves scroll position on every navigation and restores it on back/forward navigations. Scroll position is saved in sessionStorage by navigation index (history.state.index).
How to
Disable autoscroll for page
If you need to disable autoscroll on the specific pages you can specify parameter navigateState.disableAutoscroll = true to the navigate call:
import { useNavigate } from '@tramvai/module-router';
function Component() {
const navigateToWithoutScroll = useNavigate({
url: '/url/',
navigateState: { disableAutoscroll: true },
});
return <Button onClick={navigateToWithoutScroll} />;
}
Disable autoscroll for custom conditions
By default, autoscroll is disabled, when navigateState.disableAutoscroll is true.
For example, if you want to disable it for all updateCurrentRoute calls, you can provide AUTOSCROLL_DISABLED_TOKEN token:
import { AUTOSCROLL_DISABLED_TOKEN } from '@tramvai/module-autoscroll';
import { provide } from '@tramvai/core';
const providers = [
// ...,
provide({
provide: AUTOSCROLL_DISABLED_TOKEN,
useFactory:
({ router }) =>
() => {
// disable autoscroll for navigation with query `?autoscroll_disabled=true`
if (router.getCurrentUrl().query.autoscroll_disabled === 'true') {
return true;
}
// return nothing to use default behavior
},
deps: {
router: ROUTER_TOKEN,
},
}),
];
Use autoscroll with View Transitions
To avoid inconsistent animations when using View Transition API (e.g. X and Y axis movements in the same time), recommended to set autoscroll behavior to instant:
import { AUTOSCROLL_BEHAVIOR_MODE_TOKEN } from '@tramvai/module-autoscroll';
import { provide } from '@tramvai/core';
const providers = [
// ...,
provide({
provide: AUTOSCROLL_BEHAVIOR_MODE_TOKEN,
useValue: (defaultBehavior) => 'instant', // default is 'smooth' for autoscroll in new pages or anchors and 'instant' for scroll restoration
}),
];
Scroll behavior change
Global
import { AUTOSCROLL_BEHAVIOR_MODE_TOKEN } from '@tramvai/module-autoscroll';
import { provide } from '@tramvai/core';
const providers = [
// ...,
provide({
provide: AUTOSCROLL_BEHAVIOR_MODE_TOKEN,
useValue: (defaultBehavior) => 'auto', // default is 'smooth' for autoscroll in new pages or anchors and 'instant' for scroll restoration
}),
];
Local
import { useNavigate } from '@tramvai/module-router';
function Component() {
const navigateToWithAutoBehavior = useNavigate({
url: '/url/',
navigateState: { autoscrollBehavior: 'auto' },
});
return <Button onClick={navigateToWithAutoBehavior} />;
}
Scroll top change
Global
import { AUTOSCROLL_SCROLL_TOP_TOKEN } from '@tramvai/module-autoscroll';
import { provide } from '@tramvai/core';
const providers = [
// ...,
provide({
provide: AUTOSCROLL_SCROLL_TOP_TOKEN,
useValue: -1, // default is 0
}),
];
Local
You can also provide a function that will return scroll top value, for example if you save page scroll position in sessionStorage and then want to restore it:
import { AUTOSCROLL_SCROLL_TOP_TOKEN } from '@tramvai/module-autoscroll';
import { provide } from '@tramvai/core';
const providers = [
// ...,
provide({
provide: AUTOSCROLL_SCROLL_TOP_TOKEN,
// if `isRestoredValue` is `true`, it means that `defaultScrollTop` is resolved previous position for auto scroll restoration
useValue: (defaultScrollTop, isRestoredValue) => {
try {
const savedScrollTop = JSON.parse(sessionStorage.get('scrollTop'));
// for example, if you save scroll position by navigation index
return savedScrollTop[history.state.index] ?? 0;
} catch (e) {
return 0;
}
},
}),
];