Say you have a z-index
bug. Something is being covered up by something else. In my experience, a typical solution is to put position: relative
on the thing so z-index
works in the first place, and maybe rejigger the z-index
value until the right thing is on top.
The danger here is that this sets off a little z-index
war. Raising a z-index
value over here fixes one bug, and then causes another by covering up some other element somewhere else when you didn’t want to. Hopefully, you can reason about the situation and fix the bugs, even if it means refactoring more z-index
values than you thought you might need to.
If the “stack” of z-index
values is complicated enough, you might consider an abstraction. One of the problems is that your z-index
values are probably sprinkled rather randomly throughout your CSS. So, instead of simply letting that be, you can give yourself a central location for z-index
values to reference later.
I’ve covered doing that as a Sass map in the past:
$zindex: (
modal : 9000,
overlay : 8000,
dropdown : 7000,
header : 6000,
footer : 5000
);
.header {
z-index: map-get($zindex, header);
}
Now you’ve got a central place to manage those values.
Rafi Strauss has taken that a step further with OZMap. It’s also a Sass map situation, but it’s configured like this:
$globalZIndexes: (
a: null,
b: null,
c: 2000,
d: null,
);
The idea here is that most values are auto-generated by the tool (the null
values), but you can specify them if you want. That way, if you have a third-party component with a z-index
value you can’t change, you plug that into the map, and then the auto-generated numbers will factor that in when you make layers on top. It also means it’s very easy to slip layers in between others.
I think all that is clever and useful stuff — but I also think it doesn’t help with another common z-index
bug: stacking contexts. It’s always the stacking context, as I’ve written. If some element is in a stacking context that is under some other stacking context, there is no z-index
value possible that will bring it on top. That’s a much harder bug to fix.
Andy Blum wrote about this recently.
One of the great frustrations of front-end development is the unexpected interaction and overlapping of those same elements. Struggling to arrange elements along the z-axis, which extends perpendicularly through the computer screen towards and away from the viewer, is such a shared front-end experience that an element’s z-index can sometimes be used as a frustrate-o-meter gauging the developer’s mood.
The key to maintainable z-index values is understanding that z-index values can’t always be directly compared. They’re not an absolute measurement along an imaginary ruler extending out of the viewport; rather, they are a relative order between elements within the same stacking context.
Turns out there is a nice little debugging tool for stacking contexts in the form of a browser extension (Chrome and Firefox.) Andy gets into a very tricky situation where an RTL version of a website has a rule that uses a transform
to move a video on the page. That transform
triggers a new stacking context and hence a bug with an unclickable button. Gnarly.
Kinda makes me think that there could be some kinda built-in DevTools way of seeing/understanding stacking contexts. I don’t know if this is the right answer, but I’ve been seeing a leaning into “badges” in DevTools more and more. These things:
In fact, Chrome 94 recently dropped a new one for container queries, so they are being used more and more.
Maybe there could be a badge for stacking contexts? Typically what happens with badges is that you click it on and it shows a special UI. For example, for flexbox or grid it will show the overlay for all the grid lines. The special UI for stacking contexts could be color/texture coded and labelled areas showing all the stacking contexts and how they are layered.
I wrote a solution in Stylus, based on contexts, a couple of years ago and I still use it today.
https://github.com/acauamontiel/mantis-layers
This doesn’t exactly help in situations with stacking contexts, but I tweeted a few months ago about how z-index accepts calc() and custom property values:
https://www.joshwcomeau.com/css/stacking-contexts is a good read on this subject.
I’ve been using
isolation: isolate
to create new stacking contexts (with no side-effects) ever since reading that post.Well, Chrome has had the Layers tab in devtools for a loooong time, it even has a 3d representation of the page (which is not super usable).
But I agree it should be brought over into the new Layout tab and spruced up a bit.
To quote myself, I once fixed a
z-index
bug by moving elements in DOM. It’s worth pointing it out since my first instict was to mess with stacking context.