Scroll-Driven⊠Sections
Scroll-Driven⊠Sections êŽë š
I was checking out a very cool art-directed article the other day, full of scrollytelling, and, like us web devs will be forever cursed to do, wondering what they used to build it. Spoiler: itâs GSAP and ScrollTrigger.
No shame in those tech choices, they are great. But with scroll-driven animations now being a web standard with growing support, it begs the question whether we could do this with native technologies.
My brain focused on one particular need of the scrollytelling style:
- While the page scrolls through a particular section
- Have a child element appear in a fixed position and be animated
- ⊠but before and after this section is being scrolled through, the element is hidden
Perhaps a diagram can help drive that home:

But I was immediately confused when thinking about how to do this with scroll-driven animations. The problem is that that âsectionâ itself is the thing we need to apply the animation-timeline: view();
to, such that we have the proper moment to react to (âthe section is currently in view!â). But in my diagram above, itâs actually a <blockquote>
that we need to apply special conditional styling to, not the section. In a @keyframe
animation, all we can do is change declarations, we canât select other elements. Apologies if that confusing, but the root of is that we need to transfer styles from the section to the blockquote without using selectors â and itâs weird.
The good news is that what we can do is update CSS custom properties on the section, and those values will cascade to all the children of the section, and we can use those to style the blockquote.
First, in order to make a custom property animatable, we need to declare itâs type. Letâs do a fade in first, thus we need opacity:
@property --blockquoteOpacity {
syntax: "<percentage>";
inherits: true;
initial-value: 0%;
}
Now the section itself has the animation timeline:
section.has-pullquote {
animation: reveal linear both;
animation-timeline: view();
animation-range: cover 0% cover 100%;
}
And that animation weâve named reveal
above can now update the custom property:
@keyframes reveal {
from {
--blockquoteOpacity: 0%;
}
to% {
--blockquoteOpacity: 100%;
}
}
Now as the animation runs, based on itâs visibility in the viewport, it will update the custom property and thus fade/in out the blockquote:
blockquote {
opacity: var(--blockquoteOpacity);
position: sticky;
top: 50%;
transform: translateY(-50%);
}
Note Iâm using position: sticky
in there too, which will keep our blockquote in the middle of the viewport while weâre cruising through that section.
Try it out (Chrome ânâ friends have stable browser support):
Hereâs a video of it working in case youâre in a non-supporting browser:
Because we instantiated the opacity custom property for the opacity at 100%, even in a non-supporting browser like Safari, the blockquote will be visible and itâs a fine experience.
I found this all a little fiddly, but Iâm not even sure Iâm doing this âcorrectlyâ. Maybe there is a way to tap into another elements view timeline Iâm not aware of? If Iâm doing it the intended way, I could see this getting pretty cumbersome with lots of elements and lots of different values needing updated. But after all, thatâs the job sometimes. This is intricate stuff and weâre using the CSS primitives directly. The control we have is quite fine-grained, and thatâs a good thing!