CSS specificity
CSS specificity is not necessarily something I learned today, but this specific usage of it certainly is.
In CSS, `input[type=text]` is more specific than a class name (eg. `.fun-input`). What do you do when you want to override default input styles?
— Josh W. Comeau (@JoshWComeau) August 5, 2019
Turns out, you can double-up class selectors, and it increases their specificity. `.fun-input.fun-input` overrides `input[type=text]`!
What is CSS specificity
Let’s start from the beginning to give some context. CSS specificity describes how browsers decide which styles to apply when they have multiple style definitions to pick from. For that, they give different selectors different weights, the rules with higher weight will override rules with lower weight.
While this sounds pretty straight forwards, it’s an important lesson to learn since it builds the foundation of the cascading part of CSS. Very simplified the hierarchie is as follows:
- element type selector (e.g.
a
,div
,body
etc.) - class selectors (e.g.
.example
), attributes selectors (e.g.[type="radio"]
) and pseudo selectors (e.g.:hover
) - id selectors (e.g.
#example
) - inline styles
!important
rules
The weights of all selectors for a rule get summed up, so .classA .classB
has a higher specificity than .classB
. All of this is important, because this should be the way you control which styles get applied for specific elements in the tree. For rules with the same specificity, the one that was defined last wins.
Enter CSS-in-JS
To see why this is important, let’s take a look at what many people (including me) use today for styling e.g. in React applicactions. CSS-in-JS, e.g. through libraries like styled-components
or emotion
makes styling on a component level with frameworks like React super convenient. You can write your styles in JS, and they generates CSS for you and adds the rules to the document. They will create class names on the fly, using hashes to avoid unintended style conflicts. However, that means all rules will end up with the same specificity. This is usually not a problem, since you have control over the order in which the rules will be defined, but it can become an issue when you deal with a white branded component library that defines the base styles and another styling layer in your app that defines the brand styling or specific overrides for the component in special use cases.
CSS-in-JS libraries deal with this in different ways. emotion
e.g. has a util method, cx
, which is an extended version of the classnames
library by Jed Watson. It automatically detects emotion styles and instead of just concatinting the given class names, it will create a new class with the styles in the order of the class names passed in, so you don’t run into issues with the order the classes are being appended to the document head.
At rexlabs we have our own abstraction for styling, specifically around the functionality of applying brand styles on app level to a component library that is built white branded. And we used to struggle a bit with CSS specificity in the past. Utils like cx
help, but you can also solve these by knowing how specificity works, e.g. with the trick expained in the tweet above, i.e. by simply repeating the hashed class name if you want to make a rule more important.
I’m not 100% sure if I’m going to need or want to use this knowledge, but understanding the cascade in CSS definitely doesn’t hurt 😅