mirror of
https://github.com/twbs/bootstrap.git
synced 2025-01-10 03:46:13 +01:00
92e6856235
* Docs(Scrollspy): minor changes to be aligned with new version of javascript code * first example: add proper root-margin * 'nested nav' example: add more content to sections & enable smooth-scroll * fix typo * Changes after feedback * Update examples and docs to streamline some things and improve how it works section Co-authored-by: Mark Otto <markdotto@gmail.com>
394 lines
21 KiB
Markdown
394 lines
21 KiB
Markdown
---
|
|
layout: docs
|
|
title: Scrollspy
|
|
description: Automatically update Bootstrap navigation or list group components based on scroll position to indicate which link is currently active in the viewport.
|
|
group: components
|
|
toc: true
|
|
---
|
|
|
|
## How it works
|
|
|
|
Scrollspy toggles the `.active` class on anchor (`<a>`) elements when the element with the `id` referenced by the anchor's `href` is scrolled into view. Scrollspy is best used in conjunction with a Bootstrap [nav component]({{< docsref "/components/navs-tabs" >}}) or [list group]({{< docsref "/components/list-group" >}}), but it will also work with any anchor elements in the current page. Here's how it works.
|
|
|
|
- To start, scrollspy requires two things: a navigation, list group, or a simple set of links, plus a scrollable container. The scrollable container can be the `<body>` or a custom element with a set `height` and `overflow-y: scroll`.
|
|
|
|
- On the scrollable container, add `data-bs-spy="scroll"` and `data-bs-target="#navId"` where `navId` is the unique `id` of the associated navigation. Be sure to also include a `tabindex="0"` to ensure keyboard access.
|
|
|
|
- As you scroll the "spied" container, an `.active` class is added and removed from anchor links within the associated navigation. Links must have resolvable `id` targets, otherwise they're ignored. For example, a `<a href="#home">home</a>` must correspond to something in the DOM like `<div id="home"></div>`
|
|
|
|
- Target elements that aren't visible will be ignored and their corresponding nav items will never receive an `.active` class.
|
|
|
|
Checkout the examples below to see it in action.
|
|
|
|
## Examples
|
|
|
|
### Navbar
|
|
|
|
Scroll the area below the navbar and watch the active class change. Open the dropdown menu and watch the dropdown items be highlighted as well.
|
|
|
|
<div class="bd-example">
|
|
<nav id="navbar-example2" class="navbar bg-light px-3 mb-3">
|
|
<a class="navbar-brand" href="#">Navbar</a>
|
|
<ul class="nav nav-pills">
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="#scrollspyHeading1">First</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="#scrollspyHeading2">Second</a>
|
|
</li>
|
|
<li class="nav-item dropdown">
|
|
<a class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" role="button" aria-expanded="false">Dropdown</a>
|
|
<ul class="dropdown-menu">
|
|
<li><a class="dropdown-item" href="#scrollspyHeading3">Third</a></li>
|
|
<li><a class="dropdown-item" href="#scrollspyHeading4">Fourth</a></li>
|
|
<li><hr class="dropdown-divider"></li>
|
|
<li><a class="dropdown-item" href="#scrollspyHeading5">Fifth</a></li>
|
|
</ul>
|
|
</li>
|
|
</ul>
|
|
</nav>
|
|
<div class="scrollspy-example bg-light p-3 rounded-2" data-bs-spy="scroll" data-bs-target="#navbar-example2" data-bs-root-margin="0px 0px -40%" data-bs-smooth-scroll="true" tabindex="0">
|
|
<h4 id="scrollspyHeading1">First heading</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="scrollspyHeading2">Second heading</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="scrollspyHeading3">Third heading</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="scrollspyHeading4">Fourth heading</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="scrollspyHeading5">Fifth heading</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
</div>
|
|
</div>
|
|
|
|
```html
|
|
<nav id="navbar-example2" class="navbar bg-light px-3">
|
|
<a class="navbar-brand" href="#">Navbar</a>
|
|
<ul class="nav nav-pills">
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="#scrollspyHeading1">First</a>
|
|
</li>
|
|
<li class="nav-item">
|
|
<a class="nav-link" href="#scrollspyHeading2">Second</a>
|
|
</li>
|
|
<li class="nav-item dropdown">
|
|
<a class="nav-link dropdown-toggle" data-bs-toggle="dropdown" href="#" role="button" aria-expanded="false">Dropdown</a>
|
|
<ul class="dropdown-menu">
|
|
<li><a class="dropdown-item" href="#scrollspyHeading3">Third</a></li>
|
|
<li><a class="dropdown-item" href="#scrollspyHeading4">Fourth</a></li>
|
|
<li><hr class="dropdown-divider"></li>
|
|
<li><a class="dropdown-item" href="#scrollspyHeading5">Fifth</a></li>
|
|
</ul>
|
|
</li>
|
|
</ul>
|
|
</nav>
|
|
<div data-bs-spy="scroll" data-bs-target="#navbar-example2" data-bs-root-margin="0px 0px -40%" data-bs-smooth-scroll="true" class="scrollspy-example" tabindex="0">
|
|
<h4 id="scrollspyHeading1">First heading</h4>
|
|
<p>...</p>
|
|
<h4 id="scrollspyHeading2">Second heading</h4>
|
|
<p>...</p>
|
|
<h4 id="scrollspyHeading3">Third heading</h4>
|
|
<p>...</p>
|
|
<h4 id="scrollspyHeading4">Fourth heading</h4>
|
|
<p>...</p>
|
|
<h4 id="scrollspyHeading5">Fifth heading</h4>
|
|
<p>...</p>
|
|
</div>
|
|
```
|
|
|
|
### Nested nav
|
|
|
|
Scrollspy also works with nested `.nav`s. If a nested `.nav` is `.active`, its parents will also be `.active`. Scroll the area next to the navbar and watch the active class change.
|
|
|
|
<div class="bd-example">
|
|
<div class="row">
|
|
<div class="col-4">
|
|
<nav id="navbar-example3" class="h-100 flex-column align-items-stretch pe-4 border-end">
|
|
<nav class="nav nav-pills flex-column">
|
|
<a class="nav-link" href="#item-1">Item 1</a>
|
|
<nav class="nav nav-pills flex-column">
|
|
<a class="nav-link ms-3 my-1" href="#item-1-1">Item 1-1</a>
|
|
<a class="nav-link ms-3 my-1" href="#item-1-2">Item 1-2</a>
|
|
</nav>
|
|
<a class="nav-link" href="#item-2">Item 2</a>
|
|
<a class="nav-link" href="#item-3">Item 3</a>
|
|
<nav class="nav nav-pills flex-column">
|
|
<a class="nav-link ms-3 my-1" href="#item-3-1">Item 3-1</a>
|
|
<a class="nav-link ms-3 my-1" href="#item-3-2">Item 3-2</a>
|
|
</nav>
|
|
</nav>
|
|
</nav>
|
|
</div>
|
|
<div class="col-8">
|
|
<div data-bs-spy="scroll" data-bs-target="#navbar-example3" data-bs-smooth-scroll="true" class="scrollspy-example-2" tabindex="0">
|
|
<div id="item-1">
|
|
<h4>Item 1</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
<div id="item-1-1">
|
|
<h5>Item 1-1</h5>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
<div id="item-1-2">
|
|
<h5>Item 1-2</h5>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
<div id="item-2">
|
|
<h4>Item 2</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
<div id="item-3">
|
|
<h4>Item 3</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
<div id="item-3-1">
|
|
<h5>Item 3-1</h5>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
<div id="item-3-2">
|
|
<h5>Item 3-2</h5>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<p>Keep in mind that the JavaScript plugin tries to pick the right element among all that may be visible. Multiple visible scrollspy targets at the same time may cause some issues.</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
```html
|
|
<nav id="navbar-example3" class="navbar bg-light flex-column align-items-stretch p-3">
|
|
<a class="navbar-brand" href="#">Navbar</a>
|
|
<nav class="nav nav-pills flex-column">
|
|
<a class="nav-link" href="#item-1">Item 1</a>
|
|
<nav class="nav nav-pills flex-column">
|
|
<a class="nav-link ms-3 my-1" href="#item-1-1">Item 1-1</a>
|
|
<a class="nav-link ms-3 my-1" href="#item-1-2">Item 1-2</a>
|
|
</nav>
|
|
<a class="nav-link" href="#item-2">Item 2</a>
|
|
<a class="nav-link" href="#item-3">Item 3</a>
|
|
<nav class="nav nav-pills flex-column">
|
|
<a class="nav-link ms-3 my-1" href="#item-3-1">Item 3-1</a>
|
|
<a class="nav-link ms-3 my-1" href="#item-3-2">Item 3-2</a>
|
|
</nav>
|
|
</nav>
|
|
</nav>
|
|
|
|
<div data-bs-spy="scroll" data-bs-target="#navbar-example3" data-bs-smooth-scroll="true" tabindex="0">
|
|
<div id="item-1">
|
|
<h4>Item 1</h4>
|
|
<p>...</p>
|
|
</div>
|
|
<div id="item-1-1">
|
|
<h5>Item 1-1</h5>
|
|
<p>...</p>
|
|
</div>
|
|
<div id="item-1-2">
|
|
<h5>Item 1-2</h5>
|
|
<p>...</p>
|
|
</div>
|
|
<div id="item-2">
|
|
<h4>Item 2</h4>
|
|
<p>...</p>
|
|
</div>
|
|
<div id="item-3">
|
|
<h4>Item 3</h4>
|
|
<p>...</p>
|
|
</div>
|
|
<div id="item-3-1">
|
|
<h5>Item 3-1</h5>
|
|
<p>...</p>
|
|
</div>
|
|
<div id="item-3-2">
|
|
<h5>Item 3-2</h5>
|
|
<p>...</p>
|
|
</div>
|
|
</div>
|
|
```
|
|
|
|
### List group
|
|
|
|
Scrollspy also works with `.list-group`s. Scroll the area next to the list group and watch the active class change.
|
|
|
|
<div class="bd-example">
|
|
<div class="row">
|
|
<div class="col-4">
|
|
<div id="list-example" class="list-group">
|
|
<a class="list-group-item list-group-item-action" href="#list-item-1">Item 1</a>
|
|
<a class="list-group-item list-group-item-action" href="#list-item-2">Item 2</a>
|
|
<a class="list-group-item list-group-item-action" href="#list-item-3">Item 3</a>
|
|
<a class="list-group-item list-group-item-action" href="#list-item-4">Item 4</a>
|
|
</div>
|
|
</div>
|
|
<div class="col-8">
|
|
<div data-bs-spy="scroll" data-bs-target="#list-example" data-bs-smooth-scroll="true" class="scrollspy-example" tabindex="0">
|
|
<h4 id="list-item-1">Item 1</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="list-item-2">Item 2</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="list-item-3">Item 3</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="list-item-4">Item 4</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
```html
|
|
<div id="list-example" class="list-group">
|
|
<a class="list-group-item list-group-item-action" href="#list-item-1">Item 1</a>
|
|
<a class="list-group-item list-group-item-action" href="#list-item-2">Item 2</a>
|
|
<a class="list-group-item list-group-item-action" href="#list-item-3">Item 3</a>
|
|
<a class="list-group-item list-group-item-action" href="#list-item-4">Item 4</a>
|
|
</div>
|
|
<div data-bs-spy="scroll" data-bs-target="#list-example" data-bs-smooth-scroll="true" class="scrollspy-example" tabindex="0">
|
|
<h4 id="list-item-1">Item 1</h4>
|
|
<p>...</p>
|
|
<h4 id="list-item-2">Item 2</h4>
|
|
<p>...</p>
|
|
<h4 id="list-item-3">Item 3</h4>
|
|
<p>...</p>
|
|
<h4 id="list-item-4">Item 4</h4>
|
|
<p>...</p>
|
|
</div>
|
|
```
|
|
|
|
### Simple anchors
|
|
|
|
Scrollspy is not limited to nav components and list groups, so it will work on any `<a>` anchor elements in the current document. Scroll the area and watch the `.active` class change.
|
|
|
|
<div class="bd-example">
|
|
<div class="row">
|
|
<div class="col-4">
|
|
<div id="simple-list-example" class="d-flex flex-column gap-2 simple-list-example-scrollspy text-center">
|
|
<a class="p-1 rounded" href="#simple-list-item-1">Item 1</a>
|
|
<a class="p-1 rounded" href="#simple-list-item-2">Item 2</a>
|
|
<a class="p-1 rounded" href="#simple-list-item-3">Item 3</a>
|
|
<a class="p-1 rounded" href="#simple-list-item-4">Item 4</a>
|
|
<a class="p-1 rounded" href="#simple-list-item-5">Item 5</a>
|
|
</div>
|
|
</div>
|
|
<div class="col-8">
|
|
<div data-bs-spy="scroll" data-bs-target="#simple-list-example" data-bs-offset="0" data-bs-smooth-scroll="true" class="scrollspy-example" tabindex="0">
|
|
<h4 id="simple-list-item-1">Item 1</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="simple-list-item-2">Item 2</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="simple-list-item-3">Item 3</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="simple-list-item-4">Item 4</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
<h4 id="simple-list-item-5">Item 5</h4>
|
|
<p>This is some placeholder content for the scrollspy page. Note that as you scroll down the page, the appropriate navigation link is highlighted. It's repeated throughout the component example. We keep adding some more example copy here to emphasize the scrolling and highlighting.</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
```html
|
|
<div id="list-example" class="d-flex flex-column">
|
|
<a href="#item-1">Item 1</a>
|
|
<a href="#item-2">Item 2</a>
|
|
<a href="#item-3">Item 3</a>
|
|
<a href="#item-4">Item 4</a>
|
|
</div>
|
|
<div data-bs-spy="scroll" data-bs-target="#list-example" data-bs-offset="0" data-bs-smooth-scroll="true" class="scrollspy-example" tabindex="0">
|
|
<h4 id="item-1">Item 1</h4>
|
|
<p>...</p>
|
|
<h4 id="item-2">Item 2</h4>
|
|
<p>...</p>
|
|
<h4 id="item-3">Item 3</h4>
|
|
<p>...</p>
|
|
<h4 id="item-4">Item 4</h4>
|
|
<p>...</p>
|
|
</div>
|
|
```
|
|
|
|
## Usage
|
|
|
|
### Via data attributes
|
|
|
|
To easily add scrollspy behavior to your topbar navigation, add `data-bs-spy="scroll"` to the element you want to spy on (most typically this would be the `<body>`). Then add the `data-bs-target` attribute with the `id` or class name of the parent element of any Bootstrap `.nav` component.
|
|
|
|
```html
|
|
<body data-bs-spy="scroll" data-bs-target="#navbar-example">
|
|
...
|
|
<div id="navbar-example">
|
|
<ul class="nav nav-tabs" role="tablist">
|
|
...
|
|
</ul>
|
|
</div>
|
|
...
|
|
</body>
|
|
```
|
|
|
|
### Via JavaScript
|
|
|
|
```js
|
|
const scrollSpy = new bootstrap.ScrollSpy(document.body, {
|
|
target: '#navbar-example'
|
|
})
|
|
```
|
|
|
|
### Options
|
|
|
|
{{< markdown >}}
|
|
{{< partial "js-data-attributes.md" >}}
|
|
{{< /markdown >}}
|
|
|
|
{{< bs-table "table" >}}
|
|
| Name | Type | Default | Description |
|
|
| --- | --- | --- | --- |
|
|
| `rootMargin` | string | `0px 0px -40%` | Intersection Observer [rootMargin](https://developer.mozilla.org/en-US/docs/Web/API/IntersectionObserver/rootMargin) valid units, when calculating scroll position. |
|
|
| `smoothScroll` | boolean | `false` | Enables smooth scrolling when a user clicks on a link that refers to ScrollSpy observables. |
|
|
| `target` | string \| jQuery object \| DOM element | | Specifies element to apply Scrollspy plugin. |
|
|
{{< /bs-table >}}
|
|
|
|
{{< callout warning >}}
|
|
**Deprecated Options**
|
|
|
|
Up until v5.1.3 we were using `offset` & `method` options, which are now deprecated and replaced by `rootMargin`.
|
|
To keep backwards compatibility, we will continue to parse a given `offset` to `rootMargin`, but this feature will be removed in **v6**.
|
|
{{< /callout >}}
|
|
|
|
### Methods
|
|
|
|
{{< bs-table "table" >}}
|
|
| Method | Description |
|
|
| --- | --- |
|
|
| `refresh` | When adding or removing elements in the DOM, you'll need to call the refresh method. |
|
|
| `dispose` | Destroys an element's scrollspy. (Removes stored data on the DOM element) |
|
|
| `getInstance` | *Static* method to get the scrollspy instance associated with a DOM element |
|
|
| `getOrCreateInstance` | *Static* method to get the scrollspy instance associated with a DOM element, or to create a new one in case it wasn't initialized. |
|
|
{{< /bs-table >}}
|
|
|
|
Here's an example using the refresh method:
|
|
|
|
```js
|
|
const dataSpyList = document.querySelectorAll('[data-bs-spy="scroll"]')
|
|
dataSpyList.forEach(dataSpyEl => {
|
|
bootstrap.ScrollSpy.getInstance(dataSpyEl).refresh()
|
|
})
|
|
```
|
|
|
|
### Events
|
|
|
|
{{< bs-table "table" >}}
|
|
| Event | Description |
|
|
| --- | --- |
|
|
| `activate.bs.scrollspy` | This event fires on the scroll element whenever an anchor is activated by the scrollspy. |
|
|
{{< /bs-table >}}
|
|
|
|
```js
|
|
const firstScrollSpyEl = document.querySelector('[data-bs-spy="scroll"]')
|
|
firstScrollSpyEl.addEventListener('activate.bs.scrollspy', () => {
|
|
// do something...
|
|
})
|
|
```
|