Fanout with Grid and View Transitions
Fanout with Grid and View Transitions êŽë š
I got a little nerdsniped by Preethiâs post CSS Fan Out with Grid and @property
the other day. I like the idea of a opening a menu of items where the layout is powered by CSS grid. Then it collapses back into just one cell of the grid. You can even animate the grid columns/rows themselves to pull this off, as Preethi demonstrated. If you know how many columns/rows you want, you can animate that number up and down.
I found the animations just a bit less smooth than Iâd like to see, generally. The smoothness depends on lots of factors, like how many columns/rows there are, how long the duration is, and how big those columns/rows are. But imagine 3 rows collapsing to 1 over a full second. Since what is being animated is an integer. The best that can do is have two keyframes (3 to 2, 2 to 1) at 500ms each. It will not feel smooth. Preethi smoothed it over by animating the heights of items too, but the column/rows changes canât be smoothed over (there can never be 1.5 rows, for example).
My mind went right to View Transitions. Particularly the âsame pageâ style of View Transitions you can call with the JavaScript API document.startViewTransition
. With it, we actually donât even need CSS transitions/animations at all. Weird right?! We just alter the DOM (by changing a class and letting CSS do itâs thing) inside a startViewTransition
function, and the browser will automatically tween any elements with unique view-transition-name
values.
Hereâs me re-creating a similar layout to the fan out navigation in Preethiâs article:
Above, the #grid
uses CSS grid to make an 1-column 7-row grid. By default all items are placed in the 4th grid row, making a âclosedâ state. When the open
class is applied to the grid, the grid-row
is replaced with auto
letting them fall to where they normally would in the grid (the âfan outâ). The item in the middle position is just styled differently to look and act as a toggle.
Hereâs a video if youâre on a device that doesnât support View Transitions
In that above example, the space the grid occupies is the same in both states, but that wouldnât need to be the case. If you want to alter the grid columns/rows, thus changing the dimensions and nature of the grid, then view transition between those states, you can absolutely do that too.
There really is no limit to what you want to do with the grid in the open and closed state. You donât even have to think of the âstatesâ in that way, although I do find it satisfying myself. Hereâs many more items laid out on a grid with both columns and rows:
When I was poking around with that demo, it felt it was just begging for âstaggered transitionsâ, that is, animations that occur with a slight time delay between each element. Iâm eyeing up future CSS that looks like itâs going to help with this, but we can actually do it now even using the view transitions we already have.
I used Pug to create the HTML because itâs so repetitive and a processor can help abstract that and make it easier to update, but ultimately the HTML is like this:
<div id="grid">
<div class="item" style="view-transition-name: item-0"><a>0</a>
</div>
<div class="item" style="view-transition-name: item-1"><a>1</a>
</div>
<div class="item" style="view-transition-name: item-2"><a>2</a>
</div>
<!-- ... -->
We can target each one of those items with unique view-transition-specific CSS and apply the animation-delay
there. I used a Sass loop for the same reason as above, but ultimately the CSS looks like:
::view-transition-group(item-0) {
animation-delay: 0s;
}
::view-transition-group(item-1) {
animation-delay: 0.01s;
}
::view-transition-group(item-2) {
animation-delay: 0.02s;
}
/* ... */
That ends up like this:
I find that terribly pleasing.
Again Iâm leaving the entire grid in place here rather than changing the amount or size of any of the columns/rows. But you could, and it wouldnât be terribly different. It might actually be smart so the âclosedâ state isnât taking up as much space in the flow.
Again if a browser doesnât support this kind of view transition (Firefox, at the time of writing), it doesnât matter, it will still toggle open and closed just fine, just without animation.