An Introduction\u00a0To\u00a0CSS\u00a0Scroll-Driven Animations: Scroll And View Progress Timelines<\/h1>\nMariana Beldi<\/address>\n 2024-12-11T15:00:00+00:00
\n 2025-03-06T17:04:34+00:00
\n <\/header>\n
You can safely use scroll-driven animations in Chrome as of December 2024. Firefox supports them, too, though you\u2019ll need to enable a flag. Safari? Not yet, but don\u2019t worry — you can still offer a seamless experience across all browsers with a polyfill<\/a>. Just keep in mind that adding a polyfill involves a JavaScript library, so you won\u2019t get the same performance boost.<\/p>\nThere are plenty of valuable resources to dive into scroll-driven animations, which I\u2019ll be linking throughout the article. My starting point was Bramus\u2019 video tutorial<\/a>, which pairs nicely with Geoff\u2019s in-depth notes<\/a> Graham<\/a> that build on the tutorial.<\/p>\nIn this article, we\u2019ll walk through the latest published version by the W3C<\/a> and explore the two types of scroll-driven timelines — scroll progress timelines<\/strong> and view progress timelines<\/strong>. By the end, I hope that you are familiar with both timelines, not only being able to tell them apart but also feeling confident enough to use them in your work.<\/p>\nNote<\/strong>: All demos in this article only work in Chrome 116 or later at the time of writing.<\/em><\/p>\nScroll Progress Timelines<\/h2>\n
The scroll progress timeline links an animation\u2019s timeline to the scroll position of a scroll container along a specific axis. So, the animation is tied directly to scrolling. As you scroll forward, so does the animation. You\u2019ll see me refer to them as scroll-timeline<\/code> animations in addition to calling them scroll progress timelines.<\/p>\nJust as we have two types of scroll-driven animations, we have two types of scroll-timeline<\/code> animations: anonymous timelines<\/strong> and named timelines<\/strong>.<\/p>\nAnonymous scroll-timeline<\/code><\/h3>\nLet\u2019s start with a classic example: creating a scroll progress bar at the top of a blog post to track your reading progress.<\/p>\n\nSee the Pen [Scroll Progress Timeline example – before animation-timeline scroll() [forked]](https:\/\/codepen.io\/smashingmag\/pen\/RNbRqoj) by Mariana Beldi<\/a>.<\/p>See the Pen Scroll Progress Timeline example – before animation-timeline scroll() [forked]<\/a> by Mariana Beldi<\/a>.<\/figcaption><\/figure>\nIn this example, there\u2019s a <div><\/code> with the ID \u201cprogress.\u201d At the end of the CSS file, you\u2019ll see it has a background color, a defined width and height, and it\u2019s fixed at the top of the page. There\u2019s also an animation that scales it from 0<\/code> to 1<\/code> along the x-axis — pretty standard if you\u2019re familiar with CSS animations!<\/p>\nHere\u2019s the relevant part of the styles:<\/p>\n
#progress {\n \/* ... *\/\n animation: progressBar 1s linear;\n}\n\n\n@keyframes progressBar {\n from { transform: scaleX(0); }\n}\n<\/code><\/pre>\nThe progressBar<\/code> animation runs once and lasts one second with a linear timing function. Linking this animation scrolling is just a single line in CSS:<\/p>\nanimation-timeline: scroll();\n<\/code><\/pre>\nNo need to specify seconds for the duration\u200a—\u200athe scrolling behavior itself will dictate the timing. And that\u2019s it! You\u2019ve just created your first scroll-driven animation! Notice how the animation\u2019s direction is directly tied to the scrolling direction — scroll down, and the progress indicator grows wider; scroll up, and it becomes narrower.<\/p>\n\nSee the Pen [Scroll Progress Timeline example – animation-timeline scroll() [forked]](https:\/\/codepen.io\/smashingmag\/pen\/ByBzGpO) by Mariana Beldi<\/a>.<\/p>See the Pen Scroll Progress Timeline example – animation-timeline scroll() [forked]<\/a> by Mariana Beldi<\/a>.<\/figcaption><\/figure>\nscroll-timeline<\/code> Property Parameters<\/h3>\nIn a scroll-timeline<\/code> animation, the scroll()<\/code> function is used inside the animation-timeline<\/code> property. It only takes two parameters: <scroller><\/code> and <axis><\/code>.<\/p>\n\n<scroller><\/code><\/strong> refers to the scroll container, which can be set as nearest<\/code> (the default), root<\/code>, or self<\/code>.<\/li>\n<axis><\/code><\/strong> refers to the scroll axis, which can be block<\/code> (the default), inline<\/code>, x<\/code>, or y<\/code>.<\/li>\n<\/ul>\nIn the reading progress example above, we didn\u2019t declare any of these because we used the defaults. But we could achieve the same result with:<\/p>\n
animation-timeline: scroll(nearest block);\n<\/code><\/pre>\nHere, the nearest<\/code> scroll container is the root scroll of the HTML element. So, we could also write it this way instead:<\/p>\nanimation-timeline: scroll(root block);\n<\/code><\/pre>\nThe block<\/code> axis confirms that the scroll moves top to bottom in a left-to-right writing mode. If the page has a wide horizontal scroll, and we want to animate along that axis, we could use the inline<\/code> or x<\/code> values (depending on whether we want the scrolling direction to always be left-to-right or adapt based on the writing mode).<\/p>\nWe\u2019ll dive into self<\/code> and inline<\/code> in more examples later, but the best way to learn is to play around with all the combinations, and this tool by Bramus<\/a> lets you do exactly that. Spend a few minutes before we jump into the next property associated with scroll timelines.<\/p>\n\n
\n 2025-03-06T17:04:34+00:00
\n <\/header>\n
There are plenty of valuable resources to dive into scroll-driven animations, which I\u2019ll be linking throughout the article. My starting point was Bramus\u2019 video tutorial<\/a>, which pairs nicely with Geoff\u2019s in-depth notes<\/a> Graham<\/a> that build on the tutorial.<\/p>\n In this article, we\u2019ll walk through the latest published version by the W3C<\/a> and explore the two types of scroll-driven timelines — scroll progress timelines<\/strong> and view progress timelines<\/strong>. By the end, I hope that you are familiar with both timelines, not only being able to tell them apart but also feeling confident enough to use them in your work.<\/p>\n Note<\/strong>: All demos in this article only work in Chrome 116 or later at the time of writing.<\/em><\/p>\n The scroll progress timeline links an animation\u2019s timeline to the scroll position of a scroll container along a specific axis. So, the animation is tied directly to scrolling. As you scroll forward, so does the animation. You\u2019ll see me refer to them as Just as we have two types of scroll-driven animations, we have two types of Let\u2019s start with a classic example: creating a scroll progress bar at the top of a blog post to track your reading progress.<\/p>\n See the Pen [Scroll Progress Timeline example – before animation-timeline scroll() [forked]](https:\/\/codepen.io\/smashingmag\/pen\/RNbRqoj) by Mariana Beldi<\/a>.<\/p> In this example, there\u2019s a Here\u2019s the relevant part of the styles:<\/p>\n The No need to specify seconds for the duration\u200a—\u200athe scrolling behavior itself will dictate the timing. And that\u2019s it! You\u2019ve just created your first scroll-driven animation! Notice how the animation\u2019s direction is directly tied to the scrolling direction — scroll down, and the progress indicator grows wider; scroll up, and it becomes narrower.<\/p>\n See the Pen [Scroll Progress Timeline example – animation-timeline scroll() [forked]](https:\/\/codepen.io\/smashingmag\/pen\/ByBzGpO) by Mariana Beldi<\/a>.<\/p> In a In the reading progress example above, we didn\u2019t declare any of these because we used the defaults. But we could achieve the same result with:<\/p>\n Here, the The We\u2019ll dive into Scroll Progress Timelines<\/h2>\n
scroll-timeline<\/code> animations in addition to calling them scroll progress timelines.<\/p>\n
scroll-timeline<\/code> animations: anonymous timelines<\/strong> and named timelines<\/strong>.<\/p>\n
Anonymous
scroll-timeline<\/code><\/h3>\n
<div><\/code> with the ID \u201cprogress.\u201d At the end of the CSS file, you\u2019ll see it has a background color, a defined width and height, and it\u2019s fixed at the top of the page. There\u2019s also an animation that scales it from
0<\/code> to
1<\/code> along the x-axis — pretty standard if you\u2019re familiar with CSS animations!<\/p>\n
#progress {\n \/* ... *\/\n animation: progressBar 1s linear;\n}\n\n\n@keyframes progressBar {\n from { transform: scaleX(0); }\n}\n<\/code><\/pre>\n
progressBar<\/code> animation runs once and lasts one second with a linear timing function. Linking this animation scrolling is just a single line in CSS:<\/p>\n
animation-timeline: scroll();\n<\/code><\/pre>\n
scroll-timeline<\/code> Property Parameters<\/h3>\n
scroll-timeline<\/code> animation, the
scroll()<\/code> function is used inside the
animation-timeline<\/code> property. It only takes two parameters:
<scroller><\/code> and
<axis><\/code>.<\/p>\n
\n
<scroller><\/code><\/strong> refers to the scroll container, which can be set as
nearest<\/code> (the default),
root<\/code>, or
self<\/code>.<\/li>\n
<axis><\/code><\/strong> refers to the scroll axis, which can be
block<\/code> (the default),
inline<\/code>,
x<\/code>, or
y<\/code>.<\/li>\n<\/ul>\n
animation-timeline: scroll(nearest block);\n<\/code><\/pre>\n
nearest<\/code> scroll container is the root scroll of the HTML element. So, we could also write it this way instead:<\/p>\n
animation-timeline: scroll(root block);\n<\/code><\/pre>\n
block<\/code> axis confirms that the scroll moves top to bottom in a left-to-right writing mode. If the page has a wide horizontal scroll, and we want to animate along that axis, we could use the
inline<\/code> or
x<\/code> values (depending on whether we want the scrolling direction to always be left-to-right or adapt based on the writing mode).<\/p>\n
self<\/code> and
inline<\/code> in more examples later, but the best way to learn is to play around with all the combinations, and this tool by Bramus<\/a> lets you do exactly that. Spend a few minutes before we jump into the next property associated with scroll timelines.<\/p>\n