apps/mantine.dev/src/pages/hooks/use-scroll-spy.mdx
import { UseScrollSpyDemos } from '@docs/demos'; import { Layout } from '@/layout'; import { MDX_DATA } from '@/mdx';
export default Layout(MDX_DATA.useScrollSpy);
The use-scroll-spy hook tracks the scroll position and returns the index of the
element that is currently in the viewport. It is useful for creating
table of contents components (like in the mantine.dev sidebar on the right side)
and similar features.
The use-scroll-spy hook accepts an object with options:
selector - selector to get headings; by default it is 'h1, h2, h3, h4, h5, h6'getDepth - a function to retrieve the depth of a heading; by default depth is calculated based on the tag namegetValue - a function to retrieve the heading value; by default element.textContent is usedscrollHost - host element to attach scroll event listener, if not provided, window is usedoffset - offset from the top of the viewport to use when determining the active heading, by default 0 is usedExample of using custom options to get headings with the data-heading attribute:
By default, use-scroll-spy does not track changes in the DOM. If you want
to update the headings data after the parent component has mounted, you can use
the reinitialize function:
import { useEffect } from 'react';
import { useScrollSpy } from '@mantine/hooks';
function Demo({ dependency }) {
const { reinitialize } = useScrollSpy();
useEffect(() => {
reinitialize();
}, [dependency]);
return null;
}
All types used in the definition are exported from @mantine/hooks package.
interface UseScrollSpyHeadingData {
/** Heading depth, 1-6 */
depth: number;
/** Heading text content value */
value: string;
/** Heading id */
id: string;
/** Function to get heading node */
getNode: () => HTMLElement;
}
interface UseScrollSpyOptions {
/** Selector to get headings, `'h1, h2, h3, h4, h5, h6'` by default */
selector?: string;
/** A function to retrieve depth of heading, by default depth is calculated based on tag name */
getDepth?: (element: HTMLElement) => number;
/** A function to retrieve heading value, by default `element.textContent` is used */
getValue?: (element: HTMLElement) => string;
/** Host element to attach scroll event listener, if not provided, `window` is used */
scrollHost?: HTMLElement;
/** Offset from the top of the viewport to use when determining the active heading, `0` by default */
offset?: number;
}
interface UseScrollSpyReturnValue {
/** Index of the active heading in the `data` array */
active: number;
/** Headings data. If not initialize, data is represented by an empty array. */
data: UseScrollSpyHeadingData[];
/** True if headings value have been retrieved from the DOM. */
initialized: boolean;
/** Function to update headings values after the parent component has mounted. */
reinitialize: () => void;
}
function useScrollSpy(options?: UseScrollSpyOptions): UseScrollSpyReturnValue
UseScrollSpyOptions and UseScrollSpyReturnValue types are exported from the @mantine/hooks package;
you can import them in your application:
import type { UseScrollSpyOptions, UseScrollSpyReturnValue } from '@mantine/hooks';