Styled-Components: CSS-in-JS Library for the Modern Web
While preprocessors like Sass and Less have made CSS development easier, there is still a lot to be desired in this modern age of web development.
In this article, Toptal Freelance JavaScript Developer Jeremy Davis explores styled-components—the niftiest CSS-in-JS library.
While preprocessors like Sass and Less have made CSS development easier, there is still a lot to be desired in this modern age of web development.
In this article, Toptal Freelance JavaScript Developer Jeremy Davis explores styled-components—the niftiest CSS-in-JS library.
With his Master’s in CS, Jeremy specializes in front-end web apps using JavaScript frameworks and also Java and Node.js back ends.
Expertise
CSS was designed for documents, what the “old web” was expected to contain. The emergence of preprocessors like Sass or Less shows that the community needs more than what CSS offers. With web apps getting more and more complex over time, CSS’ limitations have become increasingly visible and difficult to mitigate.
Styled-components leverages the power of a complete programming language—JavaScript—and its scoping capabilities to help structure the code into components. This helps to avoid the common pitfalls of writing and maintaining CSS for large projects. A developer can describe a component’s style with no risk of side effects.
What’s the Problem?
One advantage of using CSS is that the style is completely decoupled from the code. This means that developers and designers can work in parallel without interfering with each other.
On the other hand, styled-components makes it easier to fall in the trap of strongly coupling style and logic. Max Stoiber explains how to avoid this. While the idea of separating logic and presentation is definitely not new, one might be tempted to take shortcuts when developing React components. For example, it’s easy to create a component for a validation button that handles the click action as well as the button’s style. It takes a bit more effort to split it in two components.
The Container/Presentational Architecture
This is a pretty simple principle. Components either define how things look, or they manage data and logic. A very important aspect of presentation components is that they should never have any dependencies. They receive props and render DOM (or children) accordingly. Containers, on the other hand, know about the data architecture (state, redux, flux, etc.) but should never be responsible for display. Dan Abramov’s article is a very good and detailed explanation of this architecture.
Remembering SMACSS
Although the Scalable and Modular Architecture for CSS is a style guide for organizing CSS, the basic concept is one that is followed, for the most part automatically, by styled-components. The idea is to separate CSS into five categories:
- Base contains all the general rules.
- Layout’s purpose it to define the structural properties as well as various sections of content (header, footer, sidebar, content, for instance).
- Module contains subcategories for the various logical blocks of the UI.
- State defines modifier classes to indicate elements’ states, e.g. field in error, disabled button.
- Theme contains color, font, and other cosmetic aspects that may be modifiable or dependent on user preference.
Keeping this separation while using styled-components is easy. Projects usually include some kind of CSS normalization or reset. This typically falls in the Base category. You may also define general font sizing, line sizing, etc. This can be done through normal CSS (or Sass/Less), or through the injectGlobal
function provided by styled-components.
For Layout rules, if you use a UI framework, then it will probably define container classes, or a grid system. You can easily use those classes in conjunction with your own rules in the layout components you write.
Module is automatically followed by the architecture of styled-components, since the styles are attached to components directly, rather than described in external files. Basically, each styled component that you write will be its own module. You can write your styling code without worrying about side effects.
State will be rules you define within your components as variable rules. You simply define a function to interpolate values of your CSS attributes. If using a UI framework, you might have useful classes to add to your components as well. You will probably also have CSS pseudo-selector rules (hover, focus, etc.)
The Theme can simply be interpolated within your components. It is a good idea to define your theme as a set of variables to be used throughout your application. You can even derive colors programmatically (using a library, or manually), for instance to handle contrasts and highlights. Remember that you have the full power of a programming language at your disposal!
Bring Them Together for a Solution
It is important to keep them together, for an easier navigation experience; We don’t want to organize them by type (presentation vs. logic) but rather by functionality.
Thus, we will have a folder for all the generic components (buttons and such). The others should be organized depending on the project and its functionalities. For instance, if we have user management features, we should group all the components specific to that feature.
To apply styled-components’ container/presentation architecture to a SMACSS approach, we need an extra type of component: structural. We end up with three kinds of components; styled, structural, and container. Since styled-components decorates a tag (or component), we need this third type of component to structure the DOM. In some cases, it might be possible to allow a container component to handle the structure of sub-components, but when the DOM structure becomes complex and is required for visual purposes, it’s best to separate them. A good example is a table, where the DOM typically gets quite verbose.
Example Project
Let’s build a small app that displays recipes to illustrate these principles. We can start building a Recipes component. The parent component will be a controller. It will handle the state—in this case, the list of recipes. It will also call an API function to fetch the data.
class Recipes extends Component{
constructor (props) {
super(props);
this.state = {
recipes: []
};
}
componentDidMount () {
this.loadData()
}
loadData () {
getRecipes().then(recipes => {
this.setState({recipes})
})
}
render() {
let {recipes} = this.state
return (
<RecipesContainer recipes={recipes} />
)
}
}
It will render the list of recipes, but it does not (and should not) need to know how. So we render another component that gets the list of recipes and outputs DOM:
class RecipesContainer extends Component{
render() {
let {recipes} = this.props
return (
<TilesContainer>
{recipes.map(recipe => (<Recipe key={recipe.id} {...recipe}/>))}
</TilesContainer>
)
}
}
Here, actually, we want to make a tile grid. It may be a good idea to make the actual tile layout a generic component. So if we extract that, we get a new component that looks like this:
class TilesContainer extends Component {
render () {
let {children} = this.props
return (
<Tiles>
{
React.Children.map(children, (child, i) => (
<Tile key={i}>
{child}
</Tile>
))
}
</Tiles>
)
}
}
TilesStyles.js:
export const Tiles = styled.div`
padding: 20px 10px;
display: flex;
flex-direction: row;
flex-wrap: wrap;
`
export const Tile = styled.div`
flex: 1 1 auto;
...
display: flex;
& > div {
flex: 1 0 auto;
}
`
Notice that this component is purely presentational. It defines its style and wraps whatever children it receives inside another styled DOM element that defines what tiles look like. It’s a good example of what your generic presentational components will look like architecturally.
Then, we need to define what a recipe looks like. We need a container component to describe the relatively complex DOM as well as define the style when necessary. We end up with this:
class RecipeContainer extends Component {
onChangeServings (e) {
let {changeServings} = this.props
changeServings(e.target.value)
}
render () {
let {title, ingredients, instructions, time, servings} = this.props
return (
<Recipe>
<Title>{title}</Title>
<div>{time}</div>
<div>Serving
<input type="number" min="1" max="1000" value={servings} onChange={this.onChangeServings.bind(this)}/>
</div>
<Ingredients>
{ingredients.map((ingredient, i) => (
<Ingredient key={i} servings={servings}>
<span className="name">{ingredient.name}</span>
<span className="quantity">{ingredient.quantity * servings} {ingredient.unit}</span>
</Ingredient>
))}
</Ingredients>
<div>
{instructions.map((instruction, i) => (<p key={i}>{instruction}</p>))}
</div>
</Recipe>
)
}
}
Notice here that the container does some DOM generation, but it’s the only logic it contains. Remember that you can define nested styles, so you don’t need to make a styled element for each tag that requires styling. It’s what we do here for the name and quantity of the ingredient item. Of course, we could split it further and create a new component for an ingredient. That is up to you—depending on the project complexity—to determine the granularity. In this case, it is just a styled component defined along with the rest in the RecipeStyles file:
export const Recipe = styled.div`
background-color: ${theme('colors.background-highlight')};
`;
export const Title = styled.div`
font-weight: bold;
`
export const Ingredients = styled.ul`
margin: 5px 0;
`
export const Ingredient = styled.li`
& .name {
...
}
& .quantity {
...
}
`
For the purpose of this exercise, I have used the ThemeProvider. It injects the theme in the props of styled components. You can simply use it as color: ${props => props.theme.core_color}
, I am just using a small wrapper to protect from missing attributes in the theme:
const theme = (key) => (prop) => _.get(prop.theme, key) || console.warn('missing key', key)
You can also define your own constants in a module and use those instead. For example: color: ${styleConstants.core_color}
Pros
A perk of using styled-components is that you can use it as little as you want. You can use your favorite UI framework and add styled-components on top of it. This also means that you can easily migrate an existing project component by component. You can choose to style most of the layout with standard CSS and only use styled-components for reusable components.
Cons
Designers/style integrators will need to learn very basic JavaScript to handle variables and use them in place of Sass/Less.
They will also have to learn to navigate the project structure, although I would argue that finding the styles for a component in that component’s folder is easier than having to find the right CSS/Sass/Less file that contains the rule you need to modify.
They will also need to change their tools a bit if they want syntax highlighting, linting, etc. A good place to start is with this Atom plugin and this babel plugin.
Understanding the basics
What is a CSS preprocessor?
A CSS preprocessor provides a superset language that allows you to write CSS much more efficiently.
Geneva, Switzerland
Member since November 14, 2017
About the author
With his Master’s in CS, Jeremy specializes in front-end web apps using JavaScript frameworks and also Java and Node.js back ends.