View Transitions Staggering
View Transitions Staggering êŽë š
I love view transitions. When youâre using view transitions to move multiple items, I think staggering them is cool effect and a reasonable ask for doing so succinctly. While I was playing with this recently I learned a lot and a number of different related tech and syntax came up, so I thought Iâd document it. Blogging yâall, itâs cool. You should.
Example
So letâs say we have a menu kinda thing that can open & close. Itâs just an example, feel free to use your imagination to consider two states of any UI with multiple elements. Hereâs ours:


View Transitions is a great way to handle animating this menu open. I wonât beat around the bush with a working example. Hereâs that:
That works in all browsers (see support). It animates (with staggering) in Chrome and Safari, and at this time of this writing, just instantly opens and closes in Firefox (which is fine, just less fancy).
Unique View Transition Names
In order to make the view transition work at all, every single item needs a unique view-transition-name
. Otherwise the items will not animate on their own. If you ever seen a view transition that has a simple fade-out-fade-in, when you were trying to see movement, itâs probably a problem with unique view-transition-name
s.
This brings me to my first point. Generating unique view-transition-name
s is a bit cumbersome. In a âreal worldâ application, itâs probably not that big of a deal as youâll likely be using some kind of templating that could add it. Some variation of this:
<div class="card"
style="view-transition-name: card-<%= card.id %>">
<!-- turns into -->
<div class="card"
style="view-transition-name: card-987adf87aodfasd;">
But⊠you donât always have access to something like that, and even when you do, isnât it a bit weird that the only real practical way to apply these is from the HTML and not the CSS? Donât love it. In my simple example, I use Pug to create a loop to do it.
#grid
- const items = 10;
- for (let i = 0; i < items; i++)
div(style=`view-transition-name: item-${i};`)`
That Pug code turns into:
<div id="grid">
<div style="view-transition-name: item-0;"></div>
<div style="view-transition-name: item-1;"></div>
<div style="view-transition-name: item-2;"></div>
<div style="view-transition-name: item-3;"></div>
<div style="view-transition-name: item-4;"></div>
<div style="view-transition-name: item-5;"></div>
<div style="view-transition-name: item-6;"></div>
<div style="view-transition-name: item-7;"></div>
<div style="view-transition-name: item-8;"></div>
<div style="view-transition-name: item-9;"></div>
</div>
Jen Simmons made the point about how odd this is (w3c/csswg-drafts
).
This is being improved, I hear. The CSSWG has resolved to (w3c/csswg-drafts
)âŠ
Add three keywords, one for ID attribute, one for element identity, and one that does fallback between the two.
Which sounds likely weâll be able to do something like:
#grid {
> div {
view-transition-name: auto;
}
}
This makes me think that it could break in cross-document view transitions, but⊠I donât think it actually will if you use the id
attribute on elements and the view-transition-name
ends up being based on that. Should be sweet.
Customizing the Animation
Weâve got another issue here. It wasnât just a Pug loop need to pull of the view transition staggering, itâs a Sass loop as well. Thatâs because in order to control the animation (applying an animation-delay
which will achieve the staggering), we need to give a pseudo class selector the view-transition-name
, which are all unique. SoâŠ
::view-transition-group(item-0) {
animation-delay: 0s;
}
::view-transition-group(item-1) {
animation-delay: 0.01s;
}
::view-transition-group(item-0) {
animation-delay: 0.02s;
}
/* etc. */
Thatâs just as cumbersome as the HTML part, except maybe even more-so, as itâs less and less common we even have a CSS processor like Sass to help. If we do, we can do it like this:
@for $i from 0 through 9 {
::view-transition-group(item-#{$i}) {
animation-delay: $i * 0.01s;
}
}
Making Our Own Sibling Indexes with Custom Properties
How much do we need to delay each animation in order to stagger it? Well it should be a different timing, probably increasing slightly for each element.
1st element = 0s delay
2nd element = 0.01s delay
3rd element - 0.02s delay
etc
How do we know which element is the 1st, 2nd, 3rd, etc? Well we could use :nth-child(1)
, :nth-child(2)
etc, but that saves us nothing. We still have super repetitive CSS that all but requires a CSS processor to manage.
Since weâre already applying unique view-transition-name
s at the HTML level, we could apply the elementâs âindexâ at that level too, like:
#grid
- const items = 10;
- for (let i = 0; i < items; i++)
div(style=`view-transition-name: item-${i}; --sibling-index: ${i};`) #{icons[i]}`
Which gets us that index as a custom property:
<div id="grid">
<div style="view-transition-name: item-0; --sibling-index: 0;"> </div>
<div style="view-transition-name: item-1; --sibling-index: 1;"> </div>
<div style="view-transition-name: item-2; --sibling-index: 2;"> </div>
<div style="view-transition-name: item-3; --sibling-index: 3;"> </div>
<div style="view-transition-name: item-4; --sibling-index: 4;"> </div>
<div style="view-transition-name: item-5; --sibling-index: 5;"> </div>
<div style="view-transition-name: item-6; --sibling-index: 6;"> </div>
<div style="view-transition-name: item-7; --sibling-index: 7;"> </div>
<div style="view-transition-name: item-8; --sibling-index: 8;"> </div>
<div style="view-transition-name: item-9; --sibling-index: 9;"> </div>
</div>
⊠but does that actually help us?
Not really?
It seems like we should be able to use that value rather than the CSS processor value, likeâŠ
@for $i from 0 through 9 {
::view-transition-group(item-#{$i}) {
animation-delay: calc(var(--sibling-index) * 0.01s);
}
}
But there are two problems with this:
- We need the Sass loop anyway for the view transition names
- It doesnât work
Lolz. There is something about the CSS custom property that doesnât get applied do the ::view-transition-group
like you would expect it to. Or at least *I* would expect it to. đ€·
Enter view-transition-class
There is a way to target and control the CSS animation of a selected bunch of elements at once, without having to apply a ::view-transition-group
to individual elements. Thatâs like this:
#grid {
> div {
view-transition-class: item;
}
}
Notice thatâs class not name in the property name. Now we can use that to select all the elements rather than using a loop.
/* Matches a single element with `view-transition-name: item-5` */
::view-transition-group(item-5) {
animation-delay: 0.05s;
}
/* Matches all elements with `view-transition-class: item` */
::view-transition-group(*.item) {
animation-delay: 0.05s;
}
That *.
syntax is what makes it use the class instead of the name. Thatâs how I understand it at least!
So with this, weâre getting closer to having staggering working without needing a CSS processor:
::view-transition-group(*.item) {
animation-delay: calc(var(--sibling-index) * 0.01s);
}
Except: that doesnât work. It doesnât work because --sibling-index
doesnât seem available to the pseudo class selector weâre using there. I have no idea if that is a bug or not, but it feels like it is to me.
Real Sibling Index in CSS
Weâre kinda âfakingâ sibling index with custom properties here, but we wouldnât have to do that forever. The CSSWG has resolved (w3c/csswg-drafts
):
sibling-count()
andsibling-index()
to css-values-5 ED
Iâm told Chrome is going to throw engineering at it in Q4 2024, so we should see an implementation soon.
So then mayyyyybe weâd see this working:
::view-transition-group(*.item) {
animation-delay: calc(sibling-index() * 0.01s);
}
Now thatâs enabling view transitions staggering beautifully easily, so Iâm going to cross my fingers there.
Random Stagger
And speaking of newfangled CSS, random()
should be coming to native CSS at some point somewhat soon as well as I belive thatâs been given the thumbs up. So rather than perfectly even staggering, we could do likeâŠ
::view-transition-group(*.item) {
animation-delay: calc(random() * 0.01s);
}
Faking that with Sass if fun!
Sibling Count is Useful Too
Sometimes you need to know how many items there are also, so you can control timing and delays such that, for example, the last animation can end when the first one starts again. Hereâs an example from Stephen Shaw with fakes values as Custom Properties showing how that would be used.
One line above could be written removing the need for custom properties:
/* before */
animation-delay: calc(2s * (var(--sibling-index) / var(--sibling-count)));
/* after */
animation-delay: calc(2s * (sibling-index() / sibling-count()));
Overflow is a small bummer
I just noticed while working on this particular demo that during a view transition, the elements that are animating are moved to something like a âtop layerâ in the document, meaning they do not respect the overflow
of parent elements and whatnot. See example:
Donât love that, but Iâm sure there are huge tradeoffs that Iâm just not aware of. Iâve been told this is actually a desirable trait of view transitions đ€·.
p.s. DevTools Can Inspect This Stuff
In Chrome-based browsers, open the Animations tab and slow down the animations way down.

The mid-animation, you can use that Pause icon to literally stop them. Itâs just easier to see everything when itâs stopped. Then youâll see a :view-transition
element at the top of the DOM and you can drill into it an inspect whatâs going on.
