Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/janosh/svelte-toc
Sticky responsive table of contents component
https://github.com/janosh/svelte-toc
component responsive-web-design sticky svelte sveltekit table-of-contents toc
Last synced: 6 days ago
JSON representation
Sticky responsive table of contents component
- Host: GitHub
- URL: https://github.com/janosh/svelte-toc
- Owner: janosh
- License: mit
- Created: 2021-05-19T16:08:15.000Z (over 3 years ago)
- Default Branch: main
- Last Pushed: 2024-10-07T16:34:23.000Z (4 months ago)
- Last Synced: 2024-10-12T15:56:52.948Z (3 months ago)
- Topics: component, responsive-web-design, sticky, svelte, sveltekit, table-of-contents, toc
- Language: Svelte
- Homepage: https://janosh.github.io/svelte-toc
- Size: 465 KB
- Stars: 112
- Watchers: 4
- Forks: 6
- Open Issues: 4
-
Metadata Files:
- Readme: readme.md
- Changelog: changelog.md
- Contributing: contributing.md
- License: license
Awesome Lists containing this project
README
βSvelte ToC
[![Tests](https://github.com/janosh/svelte-toc/actions/workflows/test.yml/badge.svg)](https://github.com/janosh/svelte-toc/actions/workflows/test.yml)
[![GitHub Pages](https://github.com/janosh/svelte-toc/actions/workflows/gh-pages.yml/badge.svg)](https://github.com/janosh/svelte-toc/actions/workflows/gh-pages.yml)
[![pre-commit.ci status](https://results.pre-commit.ci/badge/github/janosh/svelte-toc/main.svg)](https://results.pre-commit.ci/latest/github/janosh/svelte-toc/main)
[![NPM version](https://img.shields.io/npm/v/svelte-toc?color=blue&logo=NPM)](https://npmjs.com/package/svelte-toc)
[![Open in StackBlitz](https://img.shields.io/badge/Open%20in-StackBlitz-darkblue?logo=stackblitz)](https://stackblitz.com/github/janosh/svelte-toc)
[![REPL](https://img.shields.io/badge/Svelte-REPL-blue?label=Try%20it!)](https://svelte.dev/repl/e292ff8935dc4f5d97e5373f9f611c1b)Sticky responsive table of contents component. Live Demo
## π¨ Β Installation
```sh
npm install --dev svelte-toc
```## π Β Usage
```svelte
import Toc from 'svelte-toc'
Page Title
Section
Subsection
Next Section
Another Subsection
```
## π£ Β Props
Full list of props and bindable variables for this component (all of them optional):
1. ```ts
activeHeading: HTMLHeadingElement | null = null
```The DOM node of the currently active (highlighted) heading (based on user's scroll position on the page).
1. ```ts
activeHeadingScrollOffset: number = 100
```Distance in pixels to top edge of screen at which a heading jumps from inactive to active. Increase this value if you have a header that makes headings disappear earlier than the viewport's top edge.
1. ```ts
activeTocLi: HTMLLIElement | null = null
```The DOM node of the currently active (highlighted) ToC item (based on user's scroll position on the page).
1. ```ts
aside: HTMLElement | undefined = undefined
```The DOM node of the outer-most `aside` element. This is the element that gets the `toc` class. Cannot be passed in as a prop, only for external access!
1. ```ts
blurParams: BlurParams | undefined = { duration: 200 }
```Parameters to pass to `transition:blur` from `svelte/transition`. Set to `null` or `{ duration: 0 }` to disable blurring.
1. ```ts
breakpoint: number = 1000
```At what screen width in pixels to break from mobile to desktop styles.
1. ```ts
desktop: boolean = true
````true` if current window width > `breakpoint` else `false`.
1. ```ts
flashClickedHeadingsFor: number = 1500
```How long (in milliseconds) a heading clicked in the ToC should receive a class of `.toc-clicked` in the main document. This can be used to help users immediately spot the heading they clicked on after the ToC scrolled it into view. Flash duration is in milliseconds. Set to 0 to disable this behavior. Style `.toc-clicked` however you like, though less is usually more. For example, the demo site uses
```css
:is(h2, h3, h4) {
transition: 0.3s;
}
.toc-clicked {
color: cornflowerblue;
}
```1. ```ts
getHeadingIds = (node: HTMLHeadingElement): string => node.id
```Function that receives each DOM node matching `headingSelector` and returns the string to set the URL hash to when clicking the associated ToC entry. Set to `null` to prevent updating the URL hash on ToC clicks if e.g. your headings don't have IDs.
1. ```ts
getHeadingLevels = (node: HTMLHeadingElement): number =>
Number(node.nodeName[1]) // get the number from H1, H2, ...
```Function that receives each DOM node matching `headingSelector` and returns an integer from 1 to 6 for the ToC depth (determines indentation and font-size).
1. ```ts
getHeadingTitles = (node: HTMLHeadingElement): string =>
node.textContent ?? ``
```Function that receives each DOM node matching `headingSelector` and returns the string to display in the TOC.
1. ```ts
headings: HTMLHeadingElement[] = []
```Array of DOM heading nodes currently listed and tracked by the ToC. Is bindable but mostly meant for reading, not writing. Deciding which headings to list should be left to the ToC and controlled via `headingSelector`.
1. ```ts
headingSelector: string = `:is(h2, h3, h4):not(.toc-exclude)`
```CSS selector that matches all headings to list in the ToC. You can try out selectors in the dev console of your live page to make sure they return what you want by passing it into `[...document.querySelectorAll(headingSelector)]`. The default selector `:is(h2, h3, h4):not(.toc-exclude)` excludes `h5` and `h6` headings as well as any node with a class of `toc-exclude`. For example `
Section Title
` will not be listed.1. ```ts
hide: boolean = false
```Whether to render the ToC. The reason you would use this and not wrap the component as a whole with Svelte's `{#if}` block is so that the script part of this component can still operate and keep track of the headings on the page, allowing conditional rendering based on the number or kinds of headings present (see [PR#14](https://github.com/janosh/svelte-toc/pull/14)). To access the headings `` is currently tracking, use ``.
1. ```ts
autoHide: boolean = true
```Whether to automatically hide the ToC when it's empty, i.e. when no headings match `headingSelector`. If true, ToC also automatically un-hides itself when re-querying for headings (e.g. on scroll) and finding some.
1. ```ts
keepActiveTocItemInView: boolean = true
```Whether to keep the active ToC item in view when scrolling the page. Only applies to long ToCs that are too high to fit on screen. If true, the ToC container will scroll itself to keep the active item in view and centered (if possible).
Requires [`scrollend` event](https://developer.mozilla.org/en-US/docs/Web/API/Document/scrollend_event) browser support ([71% as of 2024-01-22](https://caniuse.com/mdn-api_element_scrollend_event)), with Safari the only major browser lacking support.1. ```ts
minItems: number = 0
```Completely prevent the ToC from rendering if it doesn't find at least `minItems` matching headings on the page. The default of 0 means the ToC will always render, even if it's empty.
1. ```ts
nav: HTMLElement | undefined = undefined
```The DOM node of the `nav` element. Cannot be passed in as a prop, only for external access!
1. ```ts
open: boolean = false
```Whether the ToC is currently in an open state on mobile screens. Can be used to externally control the `open` state through 2-way binding. This value is ignored on desktop.
1. ```ts
openButtonLabel: string = `Open table of contents`
```What to use as ARIA label for the button shown on mobile screens to open the ToC. Not used on desktop screens.
1. ```ts
pageBody: string | HTMLElement = `body`
```Which DOM node to use as the `MutationObserver` root node. This is usually the page's `` tag or `` element. All headings to list in the ToC should be children of this root node. Use the closest parent node containing all headings for efficiency, especially if you have a lot of elements on the page that are on a separate branch of the DOM tree from the headings you want to list.
1. ```ts
reactToKeys: string[] = [`ArrowDown`, `ArrowUp`, ` `, `Enter`, `Escape`, `Tab`]
```Which keyboard events to listen for. The default set of keys closes the ToC on `Escape` and `Tab` out, navigates the ToC list with `ArrowDown`, `ArrowUp`, and scrolls to the active ToC item on `Space`, and `Enter`. Set `reactToKeys = false` or `[]` to disable keyboard support entirely. Remove individual keys from the array to disable specific behaviors.
1. ```ts
scrollBehavior: 'auto' | 'smooth' = `smooth`
```Whether to scroll the page smoothly or instantly when clicking on a ToC item. Set to `'auto'` to use the browser's default behavior.
1. ```ts
title: string = `On this page`
```ToC title to display above the list of headings. Set `title=''` to hide.
1. ```ts
titleTag: string = `h2`
```Change the HTML tag to be used for the ToC title. For example, to get `{title}`, set `titleTag='strong'`.
1. ```ts
tocItems: HTMLLIElement[] = []
```Array of rendered Toc list items DOM nodes. Essentially the result of `document.querySelectorAll(headingSelector)`. Can be useful for binding.
1. ```ts
warnOnEmpty: boolean = true
```Whether to issue a console warning if the ToC is empty.
To control how far from the viewport top headings come to rest when scrolled into view from clicking on them in the ToC, use
```css
/* replace next line with appropriate CSS selector for all your headings */
:where(h1, h2, h3, h4) {
scroll-margin-top: 50px;
}
```## π° Β Slots
`Toc.svelte` has 3 named slots:
- `slot="toc-item"` to customize how individual headings are rendered inside the ToC. It has access to the DOM node it represents via `let:heading` as well as the list index `let:idx` (counting from 0) at which it appears in the ToC.
```svelte
{idx + 1}. {heading.innerText}
```- `slot="title"`: Title shown above the list of ToC entries. Props `title` and `titleTag` have no effect when filling this slot.
- `slot="open-toc-icon"`: Icon shown on mobile screens which opens the ToC on clicks.## β¨ Β Styling
The HTML structure of this component is
```html
open/close (only present on mobile)
{title}
- {heading1}
- {heading2}
...
```
`Toc.svelte` offers the following CSS variables which can be [passed in directly as props](https://github.com/sveltejs/rfcs/pull/13):
- `aside.toc`
- `z-index: var(--toc-z-index, 1)`: Applies on both mobile and desktop.
- `aside.toc > nav`
- `overflow: var(--toc-overflow, auto)`
- `min-width: var(--toc-min-width)`
- `max-width: var(--toc-desktop-max-width)`
- `width: var(--toc-width)`
- `max-height: var(--toc-max-height, 90vh)`: Height beyond which ToC will use scrolling instead of growing vertically.
- `padding: var(--toc-padding, 1em 1em 0)`
- `font-size: var(--toc-font-size)`
- `aside.toc > nav > ol > .toc-title`
- `padding: var(--toc-title-padding)`
- `margin: var(--toc-title-margin)`
- `aside.toc > nav > ol`
- `list-style: var(--toc-ol-list-style, none)`
- `padding: var(--toc-ol-padding, 0)`
- `margin: var(--toc-ol-margin)`
- `aside.toc > nav > ol > li`
- `border-radius: var(--toc-li-border-radius)`
- `padding: var(--toc-li-padding, 2pt 4pt)`
- `margin: var(--toc-li-margin)`
- `border: var(--toc-li-border)`
- `color: var(--toc-li-color)`
- `aside.toc > nav > ol > li:hover`
- `color: var(--toc-li-hover-color, cornflowerblue)`: Text color of hovered headings.
- `background: var(--toc-li-hover-bg)`
- `aside.toc > nav > ol > li.active`
- `color: var(--toc-active-color, white)`: Text color of the currently active heading (the one nearest but above top side of current viewport scroll position).
- `background: var(--toc-active-bg, cornflowerblue)`
- `font-weight: var(--toc-active-font-weight)`
- `border: var(--toc-active-border)`
- `border-width: var(--toc-active-border-width)`: Allows setting top, right, bottom, left border widths separately.
- `border-radius: var(--toc-active-border-radius, 2pt)`
- `aside.toc > button`
- `color: var(--toc-mobile-btn-color, black)`: Menu icon color of button used as ToC opener on mobile.
- `background: var(--toc-mobile-btn-bg, rgba(255, 255, 255, 0.2))`: Background of padding area around the menu icon button.
- `padding: var(--toc-mobile-btn-padding, 2pt 3pt)`
- `border-radius: var(--toc-mobile-btn-border-radius, 4pt)`
- `aside.toc.mobile`
- `bottom: var(--toc-mobile-bottom, 1em)`
- `right: var(--toc-mobile-right, 1em)`
- `aside.toc.mobile > nav`
- `width: var(--toc-mobile-width, 18em)`
- `background: var(--toc-mobile-bg, white)`: Background color of the `nav` element hovering in the lower-left screen corner when the ToC was opened on mobile screens.
- `box-shadow: var(--toc-mobile-shadow)`
- `border: var(--toc-mobile-border)`
- `aside.toc.desktop`
- `margin: var(--toc-desktop-aside-margin)`: Margin of the outer-most `aside.toc` element on desktops.
- `aside.toc.desktop > nav`
- `margin: var(--toc-desktop-nav-margin)`
- `top: var(--toc-desktop-sticky-top, 2em)`: How far below the screen's top edge the ToC starts being sticky.
- `background: var(--toc-desktop-bg)`Example:
```svelte
```
## π§ͺ β Coverage
| Statements | Branches | Lines |
| ---------------------------------------------------------------------------------- | ------------------------------------------------------------------------------------ | ------------------------------------------------------------------------ |
| ![Statements](https://img.shields.io/badge/statements-78.57%25-red.svg?style=flat) | ![Branches](https://img.shields.io/badge/branches-100%25-brightgreen.svg?style=flat) | ![Lines](https://img.shields.io/badge/lines-78.57%25-red.svg?style=flat) |## π Β Changelog
[View the changelog](changelog.md).
## π Β Contributing
Here are some steps to [get you started](contributing.md) if you'd like to contribute to this project!