CSS Selectors Redux
• by • in categories: development, speaking, web development
Last month in Sydney and Melbourne I was honoured to speak at Respond 2016, the third Web Directions conference at which I’ve had the privilege of speaking after Web Essentials 2006 and Web Directions South 2010. John Allsopp assigned me a great topic that would never have occurred to me otherwise: CSS selectors.
Here’s what I came up with:
CSS Selectors Redux
Permalink to CSS Selectors Reduxdata:image/s3,"s3://crabby-images/2025c/2025cf76d7a4ee1cd1efd6ce1a3ce127c758bdd1" alt="title slide: CSS Selectors Redux. Kevin Yank, @sentience, kevin@cultureamp.com"
I’m Kevin Yank. I work at Culture Amp, and today I’m here to talk about CSS selectors.
data:image/s3,"s3://crabby-images/eaa22/eaa2253a7c20d8db3c08b85b570df6cf32025729" alt="slide: CSS Selectors. A code listing showing a number of simple CSS declarations, with the part before the curly braces highlighted."
They’re the things in pink. Selectors are the part of CSS that lets us point at particular bits of a web page and say “I want to style you.”
data:image/s3,"s3://crabby-images/1a2ff/1a2ff575d03bae17efe97650100d6d62024bc9fe" alt="slide: what’s new; why there’s nothing new; how to be MacGyver"
In this talk, I’ll be covering what’s new in CSS selectors. I’ll then move swiftly along to why there’s nothing new in CSS selectors. And then we’ll finish strong with how to be MacGyver.
What’s New
Permalink to What’s Newdata:image/s3,"s3://crabby-images/5b624/5b624ed72303715bee2e94b00653fc4a279a832f" alt="slide: what’s new (a non-exhaustive list)"
Mostly, there are a bunch of new pseudo-classes in CSS:
data:image/s3,"s3://crabby-images/e8110/e8110154886c0cebcc287115753fed8b4dbc72fd" alt="slide: :enabled, :disabled, :checked"
data:image/s3,"s3://crabby-images/587d2/587d2246cabf73698749df66020a3720f139146a" alt="slide: :valid, :invalid, :required, :optional, :read-only, :read-write, :in-range, :out-of-range and :user-error (unsupported)"
Because most form fields start off empty and therefore invalid, most of these are of limited usefulness, unless you’re happy with your forms starting mostly red. The :user-error
pseudo-class has been proposed to match fields that are invalid as a result of user input, but it doesn’t have any support yet.
data:image/s3,"s3://crabby-images/41cae/41caed15785cd19f77745d080c413d16e8345e71" alt="slide: :first-child, :last-child, :only-child, :first-of-type, :last-of-type, :only-of-type, and highlighted: :nth-child(), :nth-last-child(), :nth-of-type() and :nth-last-of-type()"
The last four are especially interesting, because they let us build really flexible selections of children:
data:image/s3,"s3://crabby-images/ccdf7/ccdf7adcb1237b1003df3751e79e50b33363be6e" alt="slide: :nth-child(4) selects the 4th child"
data:image/s3,"s3://crabby-images/13539/13539d31e420e85942dfefe9e6b5db92cd601c8f" alt="slide: :nth-child(even) selects the 2nd, 4th, 6th child and so on"
data:image/s3,"s3://crabby-images/c1740/c174007b4ee398a2f3185b470cdb4b9c2c4447fe" alt="slide: :nth-child(n+4) selects the 4th, 5th, 6th child and so on"
data:image/s3,"s3://crabby-images/4394e/4394e22ddeac1a69ab51a5ac898bc8e469122de8" alt="slide: :nth-child(-n+4) selects the 1st, 2nd, 3rd and 4th child"
data:image/s3,"s3://crabby-images/b1ae0/b1ae06772ac3aa4f5eab5607298c5aa5d3a6fea8" alt="slide: :nth-child(3n+1) selects the 1st, 4th, 7th child and so on"
data:image/s3,"s3://crabby-images/79134/79134a8c19b13f4c378958081f16ca172f43ac00" alt="slide: :nth-last-child(3) selects the 5th child out of 7"
data:image/s3,"s3://crabby-images/dfc06/dfc0670c2019c2a20a86b449b8730acb4cb3a2c3" alt="slide: :nth-last-child(-n+3) selects the 5th, 6th and 7th child out of 7"
These child-selecting pseudo-classes will come in handy in some surprising ways a bit later.
data:image/s3,"s3://crabby-images/644e6/644e68b6d546b6536f5026672b3190d40eda9a12" alt="slide: :empty Example: .errors { border: 1px solid red; } .errors:empty { display: none; }"
:empty
can be handy if, say, your page contains a div
into which your JavaScript might insert form validation errors, but if there are no errors, then you probably don’t want to display that div
at all.
data:image/s3,"s3://crabby-images/4bbee/4bbeefeeeff04f430dc98a0d0ab100d200116e64" alt="slide: :empty doesn’t apply to a div with a line break between the opening and closing tags"
Unfortuntely, :empty
won’t match an element if it contains so much as a just line break, so you really have to control the whitespace in your markup.
data:image/s3,"s3://crabby-images/92ba3/92ba3431cbc8f14db10172ca3de269f5da604bdc" alt="slide: :blank also matches elements that contain only whitespace"
There’s basically no support for :blank
yet, but it’ll be handy if it catches on.
data:image/s3,"s3://crabby-images/a9b5f/a9b5f6b80ec01edc594f08cc128c574ac077ae84" alt="slide: :target doesn’t apply to a lightbox that is otherwise styled with display: none"
The :target
pseudo-class lets you style the element whose ID appears in the fragment identifier in the URL. You can use this to do surprising things like create lightboxes without any JavaScript! Here we have a hyperlink that points to the message
element in the page, which is a lightbox that’s styled hidden by default.
data:image/s3,"s3://crabby-images/8b912/8b912c772a65817a3323577dfb5d4e968168a3b2" alt="slide: the lightbox appears after following the link, and contains a close button that links back to an empty fragment identifier"
When you click the link, the URL points to the lightbox’s ID, so the :target
selector matches it, and the new styles make the lightbox appear. The lightbox can also contain a link back to the empty fragment identifier, so that when you click it…
data:image/s3,"s3://crabby-images/513c6/513c65656c4c989e4d841a25d049a1d79e784987" alt="slide: the URL has no fragment identifier, so the lightbox is hidden"
…the lightbox is hidden again. All without a shred of JavaScript!
data:image/s3,"s3://crabby-images/30ba5/30ba5e0e709dfd3c2618ae5a303bad9fafa1b90c" alt="slide: :not Example: article:not(:last-child) can be used to apply a border to the bottom of all but the last in a series of articles"
Say you want to put a border on the bottom of all but the last in a series of article
s. You might think to apply the border to all article
s, and then with a second rule remove the border from the last article
. Instead, with a single selector using :not
, you can apply the border efficiently by selecting all but the last child.
![slide: a[href]:not([class]) matches standard links only, .my_link_class can then match links with a class](/assets/images/blog/css-selectors-redux/slide.029.png)
In this example, the :not
pseudo-class lets you match only links that have no class
assigned. These might be hyperlinks in your body content, the ones that come out of your site’s content management system, for example. You can then use simple class selectors to style links with particular classes, and you don’t have to roll back your basic link styles because those classed links aren’t matched by the first selector.
data:image/s3,"s3://crabby-images/80ee5/80ee545564f7f92b554af9a40c085c305bdb1499" alt="slide: .trigger + * matches the sibling element that follows the one with the trigger class applied"
Here’s our first new selector that isn’t a pseudo-class!
Most of you are probably familiar with +
, the adjacent sibling selector. It lets you write a selector to match the element immediately following another element, that shares the same parent element.
data:image/s3,"s3://crabby-images/3811f/3811f7810525ff2cafd3f9090340ec84726b9b8e" alt="slide: .trigger ~ * matches all of the sibling elements following the one with the trigger class applied"
Well now we have ~
, the general sibling selector. It lets you look for matches among all of an element’s following siblings.
This is another selector that has some surprising uses, as we’ll see in a bit.
data:image/s3,"s3://crabby-images/d7224/d72245175aabd8f3c41a643df649635be0d01c93" alt="slide: “that’s old news!”"
Now you might be thinking all these new selectors have been around for years, and you’d be right; they have. So let’s look at some really new selectors.
![slide: [attribute=value] [attribute~=value] [attribute|=value] [attribute^=value] [attribute$=value] [attribute*=value]](/assets/images/blog/css-selectors-redux/slide.033.png)
![slide: [attribute=value i] [attribute~=value i] [attribute|=value i] [attribute^=value i] [attribute$=value i] [attribute*=value i]](/assets/images/blog/css-selectors-redux/slide.034.png)
This is boring. I honestly don’t know why you would want this, so let’s move on.
data:image/s3,"s3://crabby-images/c1b57/c1b5766f34599c8f79f20527d35fe96fbff42381" alt="slide: :matches() Example: ol > li :link, ol > li :visited, ul > li :link, ul > li :visited can be rewritten as :matches(ol, ul) > li :matches(:link, :visited)"
The selectors above match visited or unvisited links inside of list items, which are children of an ordered or unordered list. The first version is quite repetitive because of the alternatives. If you’re familiar with Sass, you can eliminate this repetition using nesting. But now, you can do the same thing in pure CSS using :matches
.
But this is really just syntactic sugar. It doesn’t let you select any elements that you couldn’t select before. So, boring.
data:image/s3,"s3://crabby-images/33b2d/33b2d987e4906273d37c49778b78e49035e6b8aa" alt="slide: :placeholder-shown"
Boring!
Why There’s Nothing New
Permalink to Why There’s Nothing Newdata:image/s3,"s3://crabby-images/12b78/12b787d44a14b8631fb3ec417a29e0b13d5ede64" alt="slide: why there’s nothing new"
So, why are there no new CSS Selectors worth talking about? I think it’s because we’re not asking for them, because we’re all doing things like this:
data:image/s3,"s3://crabby-images/5e8a5/5e8a55346294f4480b2b3d469cf6f41da70d09de" alt="slide: Bootstrap A search form is styled entirely by adding class attributes that apply ready-made styles to HTML elements"
But even if you’re writing your own styles, chances are you’ve recently adopted a methodology like this one:
data:image/s3,"s3://crabby-images/1fe30/1fe30c4f4e06344f0dd977000d82d5d4c7c7c92c" alt="slide: BEM .everything__is-a--class"
Even CSS Modules is just an easy way to generate unique class names:
data:image/s3,"s3://crabby-images/5c74a/5c74a2744a68767d8fc4450e62e211de9c49a8cf" alt="slide: CSS Modules A set of generic class selectors (.menu, .item, .active) in a stylesheet are compiled to unique class names (.site_nav__menu--8675e09, .site_nav__item--8675e09, .site_nav__active--8675e09"
Let me be clear, we’ve adopted CSS Modules at Culture Amp and they’ve changed our game. We love them. But they really just make it easier to use class selectors.
data:image/s3,"s3://crabby-images/ed6ef/ed6ef263af3b832af47189de9d00ca7e9791781a" alt="slide: We are abandoning HTML semantics"
In other words, we are no longer styling elements because of what they actually are, but because of what we have demanded they look like, using classes.
data:image/s3,"s3://crabby-images/e683b/e683bbf821e315f081b3af8678255fc502354c69" alt="slide: Not a button: <span class="button"> Click me! </span>"
Whether or not you think this is a bad thing probably depends on whether you’re building something with the same semantics as HTML—text documents! If you’re building a GUI application that runs in the browser, it’s harder to justify sacrificing the control of classes in support of HTML semantics. You’re probably more concerned about ARIA attributes than element types.
But for better or worse, we’re turning into these people:
data:image/s3,"s3://crabby-images/5b47e/5b47e8e9ff11a1105bb093b39c5f8a01df1fc28b" alt="slide: Bryce Shivers and Lisa Eversman stand in an open doorway with their bird-painting kits in hand"
We show up, we want to make our page “all pretty”, so we put classes on things until we’re done. What kind of craftspeople are we if we always use the same tool for every job?
data:image/s3,"s3://crabby-images/bde53/bde53888a617692d89f574d468afecfc39988784" alt="slide: prescriptive vs axiomatic, unintelligent vs intelligent"
Heydon Pickering, to whom I’m indebted for several of the techniques in this talk, has proposed that we write selectors that respond to things like the relationships between elements, rather than prescribing the styles of individual elements, one at a time. These ‘axiomatic’ or intelligent styles can give rise to useful, emergent behaviour. I don’t have a lot of time to go into this, but if you’re interested please do check out his talk, Effortless Style. The video is online for free and it’s worth your time.
data:image/s3,"s3://crabby-images/193f3/193f3fb456583960acf662149c10fc964866d48e" alt="slide: CSS Selectors Level 4 Working Draft"
Let’s take a look at a couple of features in the current CSS Selectors Level 4 Working Draft:
data:image/s3,"s3://crabby-images/0ccaa/0ccaadfbea92c15a3e32fcbdecde36562db30412" alt="slide: :local-link"
Now this would be really useful; I would totally use this! But there’s no support for it in browsers and no sign of it coming.
data:image/s3,"s3://crabby-images/14b0f/14b0feb0c2c04d8fe5282fdd867f4363f88811db" alt="slide: Reference Combinator Example: label:hover /for/ input"
If you’ve heard of the CSS checkbox hack, well this creates all sorts possibilities for interactive styles. This would be amazing. But there’s no support for it in browsers and no sign of it coming.
data:image/s3,"s3://crabby-images/c47ba/c47bac1053c4c57ed42122eb3118fbe14b0f8a80" alt="slide: Selectors Level 4 Editor’s Draft, 22 March 2016. “Removed the ‘:local-link’ and reference combinator for lack of interest.”"
That’s the ‘lack of interest’ of you, me and everyone in this room reading this.
Here’s a feature that’s still in the Editor’s Draft:
![slide: :has() Example 1: a[href]:has(> img), Example 2: section:not(:has(h1, h2, h3))](/assets/images/blog/css-selectors-redux/slide.057.png)
I’ve been wanting this for years. In the first example, a[href]:has(> img)
matches any link that contains an image as its immediate child. Right now you could match that image, but if you wanted to style the link, you’d need to add a class to it.
The second example, section:not(:has(h1, h2, h3))
matches any section
that does not contain any headings. It’d take a gnarly bit of template logic to get your CMS to add a class for this to the right section
s automatically.
But as with many things in life, you need to read the fine print:
![slide: Dynamic profile: “…appropriate in any context, including dynamic browser CSS selector matching. Includes every selector … except for: the :has() pseudo-class” Static profile: “…appropriate for contexts which aren’t extremely performance sensitive [such as] a static document tree. For example, the querySelector() method.”](/assets/images/blog/css-selectors-redux/slide.059.png)
Elsewhere in the CSS Selectors Level 4 draft, the concept of the ‘dynamic’ and ‘static’ profiles is introduced, and unlike every other selector in the spec, the :has()
pseudo-class is restricted to the static profile, which means you can only use it in JavaScript, not in your CSS stylesheets.
Browser vendors say it has to be this way, because it’s impossible to apply selectors that contain :has()
dynamically, as a page is progressively rendered and then modified.
Because we’re all just settling for class selectors, there’s a voice missing from this debate: the voice of developers like us who have wanted this kind of power and flexibility for as long as we’ve been writing CSS.
In my nearly 20 years as a professional developer, I’ve lost count of the number of times I’ve told my boss something was impossible, but once I was given a push I found a way to make it happen. Browser vendors may well be right; it could be impossible to implement full support for :has()
in a performant way, but I think we should give them that push.
data:image/s3,"s3://crabby-images/a210d/a210d39d57bf953e4fe3f50291b58bc723e01b94" alt="slide: a power drill with a gleaming assortment of sharp bits"
data:image/s3,"s3://crabby-images/0639b/0639b7068a4f01dc6fb22ea875b373f59ee35fd6" alt="slide: two bulldog clips and a rubber band"
Incidentally, this is the most depressing piece of stock photography I’ve ever paid money for, and I blame all of you for it.
Thankfully, I know someone who’s pretty good with paper clips and rubber bands.
How to be MacGyver
Permalink to How to be MacGyverdata:image/s3,"s3://crabby-images/0ed7a/0ed7ad6684ddf58d12425b91009c2271dd15456b" alt=""
MacGyver is a TV action star from the 80s, who used science to— Well, it’s probably easier just to show you:
That’s MacGyver rescuing a hostage from a terrorist camp in a bamboo airplane built from parts he found around the tent in which they were being held.
So if all we have are paper clips and rubber bands, let’s see if we can make a couple of bamboo airplanes.
data:image/s3,"s3://crabby-images/d68eb/d68eb40e7d58d39fc527a8ec87154c1b236eeb9e" alt="slide: * + *"
data:image/s3,"s3://crabby-images/dd7ca/dd7ca85df0d0d0353965d2cc8a5eb3a454d25093" alt="slide: Lobotomized Owl * + * { margin-top: 1.5em; }"
Invented by Heydon Pickering, this selector combines two universal selectors (*
) with the adjacent sibling selector (+
). It matches the second of any two adjacent elements, but by applying a top margin, which only affects block elements, you’re only styling adjacent blocks.
data:image/s3,"s3://crabby-images/575e1/575e103957eb22aec8958ebc5bf468db7e935902" alt=""
data:image/s3,"s3://crabby-images/12094/120944f3a470ac3dc4a71075ce852e6c8add385c" alt="slide: :not(:first-child)"
If you know your selectors, you might be thinking that * + *
could be more clearly written as :not(:first-child)
. Well, I’d tell you that you were almost right. First, it would need to be :not(:first-child):not(:root)
so that it didn’t match the body
element.
data:image/s3,"s3://crabby-images/03b6c/03b6c928c59e1c7ae31c8ee8c0b7802f72aafe83" alt="slide: :not(:first-child):not(:root) IE9+, high specificity"
With two pseudo-class selectors in this alternative form, you’d have a selector with relatively high specificty, that was difficult to override with other selectors. * + *
, on the other hand, has a specificity of zero. Any selector will override it!
For example:
data:image/s3,"s3://crabby-images/919e0/919e06efa49686f091cf879dec3f1fdc5259b1c5" alt="slide: p + p { margin-top: 0; text-indent: 2em; }"
Isn’t that cool?
Here’s another bamboo airplane, also thanks to Heydon Pickering.
data:image/s3,"s3://crabby-images/34c9e/34c9e0955e05b928d0fe2418f826fd8382ae8ccc" alt="slide: quantity selectors"
Since I’m presenting this at Respond, I don’t need to tell you what media queries are. They let you apply different styles depending on the size of the device’s viewport.
data:image/s3,"s3://crabby-images/a45a4/a45a47b0596a716a342a4bd8f84d43d0f5436e6e" alt="slide: media queries display: table-cell;"
data:image/s3,"s3://crabby-images/35a80/35a807d668e520f31bcc49e03ccb4fd3ac400e82" alt="display: inline-block;"
data:image/s3,"s3://crabby-images/cda11/cda11fa01fa3bac3bc37f6b84353189b44613dac" alt="display: block;"
But what if we could write styles that responded to the content of the page the same way media queries let us respond to the size of the browser window?
Let’s say we have the same nav bar.
data:image/s3,"s3://crabby-images/5af69/5af6918eafb5e1577e2b5cc7b28d0642a0da8bf4" alt="slide: quantity selectors display: table-cell;"
data:image/s3,"s3://crabby-images/56fe8/56fe846e87942692e90e5cf4e45b1fe114a2519a" alt=""
data:image/s3,"s3://crabby-images/ab786/ab78604be0c7479d505e83d8061c24b2026ed04c" alt=""
data:image/s3,"s3://crabby-images/45b9d/45b9de54d568b03506163c8b314308e603a79137" alt="display: inline-block;"
data:image/s3,"s3://crabby-images/758ad/758ad142d0bffb3cfc87d19155f57ab349fcc690" alt=""
So, what do you think? Can we do this?
Let’s start simple:
data:image/s3,"s3://crabby-images/470fb/470fb8e88605581bdaecab3902161f60eca2f525" alt="slide: one element li:only-child"
data:image/s3,"s3://crabby-images/3bb5b/3bb5b0fa2722d438dd7ce2ce4cf2908c5a66d7f2" alt="slide: li:not(:only-child)"
Okay, that was too easy. Let’s try a bigger challenge:
data:image/s3,"s3://crabby-images/370ab/370ab7312bccadf512b96d96e1c9e6a2d967e6b2" alt="slide: n elements"
data:image/s3,"s3://crabby-images/670d6/670d65afc61944c2931c6cd9d9e75b3d1e576eef" alt="slide: li:nth-last-child(6):first-child"
:nth-last-child(6)
matches the sixth-to-last child, and :first-child
matches the first child. For one element to be both the sixth-to-last child and the first child, there must be exactly six children!
data:image/s3,"s3://crabby-images/b3535/b3535b258c886de3bebebfde09d86f9a117379b1" alt="slide: li:nth-last-child(6):first-child ~ li"
Earlier we saw ~
, the general sibling selector, which matches all of the following siblings of a given element. Since we’ve already got a selector to match the first child, we can use this to write this second selector that matches the remaining five children, but again, only if there are exactly six children.
data:image/s3,"s3://crabby-images/4a36c/4a36cbc556cc140919c12d63525c6b4b7158409e" alt="slide: li:nth-last-child(6):first-child, li:nth-last-child(6):first-child ~ li"
Not bad! But the challenge before us is to write a selector that matches n or more children.
data:image/s3,"s3://crabby-images/22f04/22f045f7770b03ee0304e229e3b2969211b204ae" alt="slide: n or more elements"
data:image/s3,"s3://crabby-images/caa48/caa488cf110adde5c2bada42552ad49759d971df" alt="slide: li:nth-last-child(n+6)"
If there are fewer than six children, this selector won’t match anything. So now that we’ve matched the first several, all we need is a selector to match the remaining five children. And I just showed you how to do that!
data:image/s3,"s3://crabby-images/b4790/b4790dda4f96bb96dda46894adbde3f717714c31" alt="slide: li:nth-last-child(n+6) ~ li"
And there you have it:
data:image/s3,"s3://crabby-images/ef7ce/ef7cebe4b475d0350741c6794a995375ba7accf3" alt="slide: li:nth-last-child(n+6), li:nth-last-child(n+6) ~ li"
These two selectors combined will match all of the children, but only if there are six or more of them!
We can also invert the logic, and write a selector that matches all of the children, but only if there are n or fewer of them:
data:image/s3,"s3://crabby-images/4c1cb/4c1cb56ce0108eb0f6717e10bf25b8849f065634" alt="slide: n or fewer elements li:nth-last-child(-n+5):first-child, li:nth-last-child(-n+5):first-child ~ li"
There we go: the Lobotomized Owl and Quantity Selectors. Two bamboo airplanes assembled from parts we found lying around.
I hope I’ve rekindled your interest in CSS Selectors, and encouraged you to get out there and demand the powerful selectors we need.
data:image/s3,"s3://crabby-images/778f9/778f948d0962434badb9770325b7611de7b346d6" alt="slide: CSS Selectors Redux, Kevin Yank, @sentience, kevin@cultureamp.com, Culture Amp is hiring in Melbourne!"