DEV Community

Ben McMahen
Ben McMahen

Posted on • Originally published at benmcmahen.com on

Writing your CSS with Emotion

[Edit] Styled-components also supports the css prop! So consider this more of an article about comparing two apis - the styled api vs the css api.

I have written css for the web in just about every way imaginable. I’ve used css modules, preprocessors including Sass, Less, and Stylus, BEM, functional css (using Tachyons), and I’ve used numerous CSS-in-JS libraries including Styled-components, and react-native stylesheets. I’ve tried all of this, and using Emotion’s css prop is easily my favourite. I’ll try to explain why.

Emotion vs Styled-components

Let’s assume that you’re already on the CSS-in-JS bandwagon. If you’re not, it’s worth perusing the many great articles and videos about the benefits of writing css in Javascript.

My main task here is to argue that using Emotion is the best library with which to style your apps. In truth, Emotion shares much with the other behomoth in the CSS-in-JS realm, Styled-components. It even exposes a styled api that basically mimics how styled-components works. But in comparing the two libraries, I’m really comparing two distinct options for styling your components.

One is the styled-components way:

import styled from 'styled-components'

const Button = styled.button` 
  background: #08e; 
  color: white; 
  padding: 6px 10px; 
  border: none;
`

function Example() {
  return (
    <div>
      <Button>Hello world</Button>
    </div>
  )
}

Styled-components uses the styled.button style API combined with template literals to create buttons with styles. You can see why it’s popular. With examples like this, it’s a beautiful API.

In contrast, with Emotion we can add styles to an element using the css prop much as you’d use the style prop.

/** @jsx jsx */
import { css, jsx } from '@emotion/core'

const Button = props => (
  <button
    css={{ 
      background: '#08e', 
      color: 'white', 
      padding: '6px 10px', 
      border: 'none', 
    }}
    {...props}
  />
)

function Example() {
  return (
    <div>
      <Button>Hello world</Button>
    </div>
  )
}

On first glance, the styled-component example seems more elegant. But I’ve found that over the long run, I generally prefer the css style api for the following reasons:

You’re still writing regular React components.

Especially when working with typescript, I consider this beneficial. You type a Button component just as you would a regular React component, allowing you to clearly specify which props are accepted. As a result, you’re less likely to pollute your dom with odd attributes—a problem I found common when passing custom attributes to styled components.

Object styles are easier to work with.

When working with typescript, I love that all of my css is typechecked and provides robust autocompletion. And I generally find it easier to insert theme variables into objects instead of using the ${theme.color.red} style notation. The small downside to objects is that they are slightly more cumbersome to write and aren’t easily copied from browsers.

Naming things is hard. And I’m lazy.

When working with styled-components you generally need to create names for components with distinct styles. This results in many components which lack obvious semantic importance which nevertheless require distinct names. Naming these components with a meaningful descriptor can be tough. Furthermore, the boilerplate often feels burdensome when applying small custom styles, such as altering margins or padding. So, because I’m lazy, I find myself often resorting to using the style prop for quick adjustments.

But using the css prop avoids these pitfalls, while still providing the opportunity to wrap styles into a component if it’s worth reusing and semantically meaningful.

function Example() {
  return (
    <div
      css={{
        margin: theme.spacing.sm,
        padding: theme.spacing.sm,
      }}
    >
      <Button variant="primary">Hi there</Button>
    </div>
  )
}

You colocate styles with elements.

With the css prop, what you see is what you get. It’s a small point, but not having to scroll away from an element to find your style definition really improves my workflow. It feels more efficient and keeps me in the flow when writing my components. Need to delete an element? There’s no need to hunt down the orphaned style definition.

Composition is dead easy.

Consider our Button component. What if we want to provide a margin to it?

function Example() {
  return (
    <>
      <Button css={{ marginRight: '1rem' }}>Cancel</Button> 
      <Button variant="primary">Save</Button>
    </>
  )
}

This passes the styles defined in our example to the Button element and composes the styles for us, applying the custom margin.

Both are great, but I choose Emotion

Styled-components is a wonderful tool and combined with something like styled-system you can get the benefits of functional style css which can alleviate some of the naming issues. But I’ve found that using the css prop, especially with typescript, reduces the need for something like styled-system and generally provides the most flexible means of writing your styles in Javascript.

Suggested links :

Emotion.sh

(This is an article posted to my blog at benmcmahen.com. You can read it online by clicking here.)

Top comments (4)

Collapse
 
dance2die profile image
Sung M. Kim • Edited

Thanks, Ben.

I've been wondering why Kent has been excited about Emotion 😉

This article explains why anyone would like to use it over Styled Components


Collapse
 
bmcmahen profile image
Ben McMahen

Kent knows his stuff!

Collapse
 
ahmadawais profile image
Ahmad Awais ⚡️

Latest beta of styled-components is faster than emotion.

I used both. Both are compatible with each other css and styled methods.

Thread Thread
 
bmcmahen profile image
Ben McMahen

Yeah, the difference between the two is quite negligible these days. This article should be considered more of a css prop vs styled debate. Luckily you can use css on both.