Not to mention
Extra
Manuel Matuzović wrote about the difference between :has(:not())
and :not(:has())
almost two years ago.
Another thing that may trip you up when using :not()
is going the opposite direction: While :has()
is about finding out about child nodes, what about selecting something further up in the DOM? Consider the following example.
:not(.specific) .element {
color: tomato;
}
Here we are changing the color of an .element
that does not have a .specific
ancestor. Well, changes are this is always the case. Unless every ancestor all the way up to <HTML>
has the .specific
class attached, our color will be applied.
For :not()
to make sense, we need to tie to something else. Sadly this requires us to have .something
to work with in the first place.
.something:not(.specific) .element {
color: tomato;
}
Now we target an .element
that is part of .something
, unless said something is .specific
.
So we are done?
Almost. There is a way to make the wildcard version actually useful.
:not(.specific) > .element {
color: tomato;
}
By adding the child combinator, we are not targeting any ancestor, but just the parent. While this obviously a bit limiting, targeting a parent that is not .specific
may at times be exactly what you need.
Showcase
View “Using :not() on ancestors” on CodePen.
Update
Now we are done. Or so I thought. But Christopher Kirk-Nielsen reached out to inform me that by putting the ancestor inside :not()
we can get rid of the limitation. You can check out his article for all the details, I’ll just put the example below.
.element:not(.specific *) {
color: tomato;
}