Check comments below, this is only my opinion and a choice specific to my use case
We've architected a SPA to be universal-ready. It grew a lot, when we introduced code splitting we realised converting to CSS in JS was unavoidable in order to have pre-rendering and not load all our CSS up front. We've procrastinated on looking at CSS in JS properly, prioritasing immediate business needs, but kept an eye on industry evolutions.
Our CSS solution for components was a CSS companion file per component, imported with style / postCSS loader (with webpack) but no CSS modules: we were missing a tighter coupling between component rendering and styles. We were at the bottom of the CSS and componentization ladder, the following links have influenced us in choosing the right solution for us:
- Christopher Chedeaux, 2015, "CSS in JS" https://speakerdeck.com/vjeux/react-css-in-js
- Mark Dagleish, 2015, "The case for CSS modules" https://www.youtube.com/watch?v=zR1lOuyQEt8
- Cristiano Rastelli, 2017, Cristiano Rastelli, https://www.youtube.com/watch?v=bb_kb6Q2Kdc & https://speakerdeck.com/didoo/let-there-be-peace-on-css
- Daniel Steigerwald, 2017, "CSS in JS: The Argument Refined" https://medium.com/@steida/css-in-js-the-argument-refined-471c7eb83955
- Ludovico Fisher, 2017, "Comparing CSS in JS solutions" https://www.ludovf.net/blog/comparing-css-in-js-solutions/
- Mark Dagleish, 2017, "A unified styling language" https://medium.com/seek-blog/a-unified-styling-language-d0c208de2660
There is a vast number of libraries to choose from: https://github.com/MicheleBertoli/css-in-js.
CSS modules would have worked for us, and would have been a minimal effort (in term of refactoring and changing habits). However, we wanted to go further in our componentization approach. We discarded inline styles: they are restrictive, the trend is about dynamic style sheets, to enjoy CSS full power. We also discarded solutions which simply consist of embedding CSS in JS, and not embedding CSS in components.
We chose to highlight four libraries (non-exhaustive list):
- styled-components
- glamorous
- styletron
- fela
Other libraries we didn't evaluate but we could have:
- JSS
All libraries mentioned are excellent. Our experimentation and choice making was time-boxed, and at the end we feel we made what we think is the best choice for us and our circumstances, maybe not for everyone.
They are all inline CSS libraries (and not inline styles), so they support pseudo-elements, pseudo-selectors, media queries, prefixing and value fallbacks.
They also all support server-side rendering. Support for React Native was a nice to have (styled-components, fela, glamorous). We regarded atomic CSS (styletron, fela) as an implementation detail.
Syled-components uses raw CSS, other libraries use object literals.
We weren't going to base our choice on popularity: they all have safe levels of monthly downloads and activity in their repositories.
Fela was discovered when doing our research, we hadn't heard about it before, and it won the argument for mainly two reasons: flexibility and extensibility.
- Flexible API: you can abstract away class names by wrapping elements with a styling HoC (like with styled-components, glamorous or styletron). Or you can use a styling HoC on a composite component, manually distributing class names. I view this as very important for componentisation (passing class names as props, avoiding to wrap components with elements) and for integration with 3rd party libraries.
- Extensible: fela has plugins, reminding me of postCSS plugins. It adds to fela's flexibility. Existing plugins for pseudo selectors and media queries make working with object literals much better (
onHover
instead of:hover
, etc...). It gives you power to create your own API / way of writing style objects. It also has enhancers which are basically renderer decorators. Fela uses atomic CSS which is reknown its bad developer experience. Fela offers a monolithic enhancer (with the option to include rule names) for a better DX.
For styled-components I thought I'd add some comments, in case someone would like to use this to make an informed choice π
Vendor Prefixing
styled-components doesn't actually use
inline-style-prefixer
. It usesstylis
for CSS flattening and prefixing (which can be summarised as CSS parsing and processing).Atomic classes
Despite claims that it is vastly more performant compared to other factors this operates on a micro level. CSS injection and processing are hugely more important. CSS injection actually being the most important one by an entire order of magnitude compared to CSS processing.
The biggest factor for a decision
All the listed libraries put forward very different good practices and ways of working with them.
I don't want to list all of them, but in general fela and styletron are more flexible and traditional libraries for CSS-in-JS to which for example aphrodite and glamor count as well. They don't restrict any usage, but that means that DX is going to be worse when starting a project from scratch initially.
Glamorous and styled-components are part of the new component-based CSS-in-JS libraries, and both focus on enforcing best practices. This has the advantage that most concerns are abstracted away, such as performance, CSS injection, component creation and structure, and so on. On the other hand, if someone doesn't prefer a BEM-like approach to CSS, then this won't feel perfect while using it. But a lot of users agree with us in that it makes them feel more efficient, and a project stays more maintainable without effort. (Yes, I know. Careful statement haha)
Preprocessing
One thing that isn't being taken into consideration here is preprocessing. styled-components (and possibly Glamorous in a potential joint-effort) are moving towards preprocessing CSS for inter-compatibility and performance. This will allow customisation of the CSS pipeline (which will then be part of babel or webpack) besides other things.