This feature is well established and works across many devices and browser versions. It’s been available across browsers since April 2022.
The scroll-snap-type CSS property is set on a scroll container, opting it into scroll snapping by setting the direction and strictness of snap point enforcement within the snap port.
scroll-snap-type: none;
scroll-snap-type: x mandatory;
scroll-snap-type: x proximity;
<section class="default-example" id="default-example">
<div id="example-element">
<div>1</div>
<div>2</div>
<div>3</div>
</div>
<div class="info">Scroll »</div>
</section>
.default-example {
flex-wrap: wrap;
}
.default-example .info {
width: 100%;
padding: 0.5em 0;
font-size: 90%;
}
#example-element {
text-align: left;
width: 250px;
height: 250px;
overflow-x: scroll;
display: flex;
box-sizing: border-box;
border: 1px solid black;
}
#example-element > div {
flex: 0 0 250px;
width: 250px;
background-color: rebeccapurple;
color: white;
font-size: 30px;
display: flex;
align-items: center;
justify-content: center;
scroll-snap-align: start;
}
#example-element > div:nth-child(even) {
background-color: white;
color: rebeccapurple;
}
If the content in the scroll port changes — for example, if content is added, moved, deleted, or resized — the scroll container will re-snap to the previously snapped content if that content is still present.
If the value of a scroll snap-related property, such as scroll-snap-type or scroll-margin, is changed, the scroll container will re-snap based on the current value of scroll-snap-type.
Specifying any precise animations or physics used to enforce those snap points is not covered by this property but instead left up to the user agent.
/* No snapping */ scroll-snap-type: none; /* Keyword values for snap axes */ scroll-snap-type: x; scroll-snap-type: y; scroll-snap-type: block; scroll-snap-type: inline; scroll-snap-type: both; /* Optional keyword values for snap strictness */ /* mandatory | proximity */ scroll-snap-type: x mandatory; scroll-snap-type: y proximity; scroll-snap-type: both mandatory; /* Global values */ scroll-snap-type: inherit; scroll-snap-type: initial; scroll-snap-type: revert; scroll-snap-type: revert-layer; scroll-snap-type: unset;
noneWhen the visual viewport of this scroll container is scrolled, it must ignore snap points.
xThe scroll container snaps to snap positions in its horizontal axis only.
yThe scroll container snaps to snap positions in its vertical axis only.
blockThe scroll container snaps to snap positions in its block axis only.
inlineThe scroll container snaps to snap positions in its inline axis only.
bothThe scroll container snaps to snap positions in both of its axes independently (potentially snapping to different elements in each axis).
mandatoryThe visual viewport of this scroll container must snap to a snap position if it isn't currently scrolled.
proximityThe visual viewport of this scroll container may snap to a snap position if it isn't currently scrolled. The user agent decides if it snaps or not based on scroll parameters. This is the default snap strictness if any snap axis is specified.
| Initial value | none |
|---|---|
| Applies to | all elements |
| Inherited | no |
| Computed value | as specified |
| Animation type | discrete |
scroll-snap-type =
none |
[ x | y | block | inline | both ] [ mandatory | proximity ]?
<main>
<section class="x mandatory-scroll-snapping" dir="ltr">
<div>X Mand. LTR</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="x proximity-scroll-snapping" dir="ltr">
<div>X Prox. LTR</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="y mandatory-scroll-snapping" dir="ltr">
<div>Y Mand. LTR</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="y proximity-scroll-snapping" dir="ltr">
<div>Y Prox. LTR</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="x mandatory-scroll-snapping" dir="rtl">
<div>X Mand. RTL</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="x proximity-scroll-snapping" dir="rtl">
<div>X Prox. RTL</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="y mandatory-scroll-snapping" dir="rtl">
<div>Y Mand. RTL</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
<section class="y proximity-scroll-snapping" dir="rtl">
<div>Y Prox. RTL</div>
<div>2</div>
<div>3</div>
<div>4</div>
<div>5</div>
</section>
</main>
/* scroll-snap */
.x.mandatory-scroll-snapping {
scroll-snap-type: x mandatory;
}
.x.proximity-scroll-snapping {
scroll-snap-type: x proximity;
}
.y.mandatory-scroll-snapping {
scroll-snap-type: y mandatory;
}
.y.proximity-scroll-snapping {
scroll-snap-type: y proximity;
}
div {
text-align: center;
scroll-snap-align: center;
flex: none;
}
| Specification |
|---|
| CSS Scroll Snap Module Level 1> # scroll-snap-type> |
| Desktop | Mobile | |||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|
| Chrome | Edge | Firefox | Opera | Safari | Chrome Android | Firefox for Android | Opera Android | Safari on IOS | Samsung Internet | WebView Android | WebView on iOS | |
scroll-snap-type |
69 | 7912–79Edge supports an earlier draft of CSS Scroll Snap without axis values. |
9968–99On macOS Monterey, scroll snapping does not complete reliably. See bug 1749352.39–68An earlier draft of CSS Scroll Snap without axis values. |
56 | 119Older Safari versions support an earlier draft of CSS Scroll Snap without axis values. |
69 | 6839–68An earlier draft of CSS Scroll Snap without axis values. |
48 | 119Older Safari on iOS versions support an earlier draft of CSS Scroll Snap without axis values. |
10.0 | 69 | 119Older WebView on iOS versions support an earlier draft of CSS Scroll Snap without axis values. |
block |
69 | 79 | 68 | 56 | 11 | 69 | 68 | 48 | 11 | 10.0 | 69 | 11 |
both |
69 | 79 | 68 | 56 | 11 | 69 | 68 | 48 | 11 | 10.0 | 69 | 11 |
inline |
69 | 79 | 68 | 56 | 11 | 69 | 68 | 48 | 11 | 10.0 | 69 | 11 |
none |
69 | 79 | 39 | 56 | 11 | 69 | 39 | 48 | 11 | 10.0 | 69 | 11 |
x |
69 | 79 | 68 | 56 | 11 | 69 | 68 | 48 | 11 | 10.0 | 69 | 11 |
y |
69 | 79 | 68 | 56 | 11 | 69 | 68 | 48 | 11 | 10.0 | 69 | 11 |
scroll-margin, scroll-padding
scroll-snap-align, scroll-margin, scroll-snap-stop
© 2005–2025 MDN contributors.
Licensed under the Creative Commons Attribution-ShareAlike License v2.5 or later.
https://developer.mozilla.org/en-US/docs/Web/CSS/scroll-snap-type