What is styled components
When it comes to styling React components, there are a few different approaches that you can take. One popular approach is using styled-components, a CSS-in-JS library that lets you write CSS rules that are scoped to a particular component. This means that your CSS will only affect the specific component that you’re working on and not any other parts of your app.
Styled-components makes it easy to reuse React components that have their own styles.They’re great for building user interfaces that look consistent and clean, without having to worry about CSS classes and inline styles.
Creating a styled component is easy – just use the styled
function from the styled-components
library, and pass in a React component. For example, to style the <Button>
component:
1 2 3 4 5 6 7 |
<span class="kd">const</span> <span class="nx">Button</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">button</span><span class="s2">` background-color: red; color: white; font-size: 16px; padding: 20px; `</span><span class="p">;</span> |
Exit fullscreen mode
You can then use the styled component like any other React component:
1 2 |
<span class="p"><</span><span class="nc">Button</span><span class="p">></span>Click me!<span class="p"></</span><span class="nc">Button</span><span class="p">></span> |
Exit fullscreen mode
Styled components are easy to customize and extend. In the example above, we could add a new style to the Button
component by using the extend
method:
1 2 3 4 |
<span class="kd">const</span> <span class="nx">ExtendedButton</span> <span class="o">=</span> <span class="nx">Button</span><span class="p">.</span><span class="nx">extend</span><span class="s2">` border: 2px solid black; } </span> |
Exit fullscreen mode
We could also add a new style to all buttons by using the createGlobalStyle
function:
1 2 3 4 5 6 |
<span class="kd">const</span> <span class="nx">GlobalStyle</span> <span class="o">=</span> <span class="nx">createGlobalStyle</span><span class="s2">` button { border: 2px solid black; } }`</span><span class="p">;</span> |
Exit fullscreen mode
Now all buttons will have a black border, regardless of how they’re styled.
Advantages of Styled-components
- It makes your code more readable and easier to understand.
- It solves CSS problem of selector name collisions
- It can help you keep your code DRY (Don’t Repeat Yourself).
- It makes it easy to change the look and feel of your component without having to change the actual implementation of your component.
- It comes with a built-in theme provider, so you can easily change the look and feel of your entire application with a single theme.
- It is easy to unit test because you don’t have to worry about the style rules leaking into your tests.
- It is easy to integrate with other libraries, such as React Router or Redux.
- It is small and fast, so it won’t slow down your application.
- The library has a great community and support.
Disadvantages of Styled-components
- Styled-components creates a lot of extra markup that can bloat your code and make it more difficult to read.
- The library is relatively new, so there is less community support than for other CSS-in-JS libraries.
- Styled-components doesn’t support server-side rendering out of the box. You need to use a library like styled-components-ssr to handle this.
- The library can be difficult to debug because of the way it generates CSS classes.
Responsive Design with styled-components
You can use media queries to style components based on the size of the user’s screen. This is especially useful for responsive design.
CSS @media
To do this, you first need to create a media
object:
1 2 3 4 5 6 7 8 9 10 |
<span class="kd">const</span> <span class="nx">MyComponent</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">div</span><span class="s2">` color: red; @media (min-width: 768px) { color: blue; } @media (min-width: 992px) { color: green; } } </span> |
Exit fullscreen mode
This will make the text color red by default, blue on tablets, and green on desktop devices.
Functions
You can also use the styled-components
media queries helper function to create media objects:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 |
<span class="k">import</span> <span class="p">{</span> <span class="nx">css</span> <span class="p">}</span> <span class="k">from</span> <span class="dl">'</span><span class="s1">styled-components</span><span class="dl">'</span><span class="p">;</span> <span class="kd">const</span> <span class="nx">media</span> <span class="o">=</span> <span class="p">{</span> <span class="na">tablet</span><span class="p">:</span> <span class="p">(</span><span class="nx">styles</span><span class="p">)</span> <span class="o">=></span> <span class="nx">css</span><span class="s2">` @media (min-width: 768px) { </span><span class="p">${</span><span class="nx">styles</span><span class="p">}</span><span class="s2"> } `</span><span class="p">,</span> <span class="na">desktop</span><span class="p">:</span> <span class="p">(</span><span class="nx">styles</span><span class="p">)</span> <span class="o">=></span> <span class="nx">css</span><span class="s2">` @media (min-width: 992px) { </span><span class="p">${</span><span class="nx">styles</span><span class="p">}</span><span class="s2"> } `</span><span class="p">,</span> <span class="p">};</span> |
Exit fullscreen mode
Then, you can use the media object in your styled component:
1 2 3 4 5 6 |
<span class="kd">const</span> <span class="nx">MyComponent</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">div</span><span class="s2">` color: red; </span><span class="p">${</span><span class="nx">media</span><span class="p">.</span><span class="nx">tablet</span><span class="p">(</span><span class="s2">`color: blue;`</span><span class="p">)}</span> <span class="p">${</span><span class="nx">media</span><span class="p">.</span><span class="nx">desktop</span><span class="p">(</span><span class="s2">`color: green;`</span><span class="p">)}</span><span class="s2"> } </span> |
Exit fullscreen mode
This has the same effect as the previous example.
<Media/>
component
You can also use the styled-components
<Media/>
component to style components based on the size of the user’s screen. The Media
component takes a query
prop which is a media query, and children which are the styles to apply when the query matches.
For example, this will make the text color red by default, blue on tablets, and green on desktop devices:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
<span class="k">import</span> <span class="p">{</span> <span class="nx">Media</span> <span class="p">}</span> <span class="k">from</span> <span class="dl">'</span><span class="s1">styled-components</span><span class="dl">'</span><span class="p">;</span> <span class="kd">const</span> <span class="nx">MyComponent</span> <span class="o">=</span> <span class="p">()</span> <span class="o">=></span> <span class="p">(</span> <span class="p"><</span><span class="nt">div</span><span class="p">></span> <span class="p"><</span><span class="nc">Media</span> <span class="na">query</span><span class="p">=</span><span class="s">"(max-width: 767px)"</span><span class="p">></span> <span class="si">{</span><span class="p">(</span><span class="nx">matches</span><span class="p">)</span> <span class="o">=></span> <span class="p">{</span> <span class="k">if</span> <span class="p">(</span><span class="nx">matches</span><span class="p">)</span> <span class="p">{</span> <span class="k">return</span> <span class="p"><</span><span class="nt">p</span><span class="p">></span>This is a mobile phone<span class="p"></</span><span class="nt">p</span><span class="p">>;</span> <span class="p">}</span> <span class="k">else</span> <span class="p">{</span> <span class="k">return</span> <span class="kc">null</span><span class="p">;</span> <span class="p">}</span> <span class="p">}</span><span class="si">}</span> <span class="p"></</span><span class="nc">Media</span><span class="p">></span> <span class="p"><</span><span class="nc">Media</span> <span class="na">query</span><span class="p">=</span><span class="s">"(min-width: 768px) and (max-width: 991px)"</span><span class="p">></span> <span class="si">{</span><span class="p">(</span><span class="nx">matches</span><span class="p">)</span> <span class="o">=></span> <span class="p">{</span> <span class="k">if</span> <span class="p">(</span><span class="nx">matches</span><span class="p">)</span> <span class="p">{</span> <span class="k">return</span> <span class="p"><</span><span class="nt">p</span><span class="p">></span>This is a tablet<span class="p"></</span><span class="nt">p</span><span class="p">>;</span> <span class="p">}</span> <span class="k">else</span> <span class="p">{</span> <span class="k">return</span> <span class="kc">null</span><span class="p">;</span> <span class="p">}</span> <span class="p">}</span><span class="si">}</span> <span class="p"></</span><span class="nc">Media</span><span class="p">></span> <span class="p"><</span><span class="nc">Media</span> <span class="na">query</span><span class="p">=</span><span class="s">"(min-width: 992px)"</span><span class="p">></span> <span class="si">{</span><span class="p">(</span><span class="nx">matches</span><span class="p">)</span> <span class="o">=></span> <span class="p">{</span> <span class="k">if</span> <span class="p">(</span><span class="nx">matches</span><span class="p">)</span> <span class="p">{</span> <span class="k">return</span> <span class="p"><</span><span class="nt">p</span><span class="p">></span>This is a desktop<span class="p"></</span><span class="nt">p</span><span class="p">>;</span> <span class="p">}</span> <span class="k">else</span> <span class="p">{</span> <span class="k">return</span> <span class="kc">null</span><span class="p">;</span> <span class="p">}</span> <span class="p">}</span><span class="si">}</span> <span class="p"></</span><span class="nc">Media</span><span class="p">></span> <span class="p"></</span><span class="nt">div</span><span class="p">></span> <span class="p">);</span> |
Exit fullscreen mode
styled-media
Library.
Another options to implement responsive styles is styled-media
library. It provides a simple API for creating media queries in your component styles.
First, install styled-media
:
1 2 3 |
npm install --save styled-media-query |
Exit fullscreen mode
Then, import it in your component:
1 2 3 |
<span class="k">import</span> <span class="nx">styled</span> <span class="k">from</span> <span class="dl">'</span><span class="s1">styled-components</span><span class="dl">'</span> <span class="k">import</span> <span class="p">{</span> <span class="nx">breakpoint</span> <span class="p">}</span> <span class="k">from</span> <span class="dl">'</span><span class="s1">styled-media-query</span><span class="dl">'</span> |
Exit fullscreen mode
Now you can use media queries in your styles, like this:
1 2 3 4 5 6 7 |
<span class="kd">const</span> <span class="nx">Container</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">div</span><span class="s2">` width: 100%; </span><span class="p">${</span><span class="nx">breakpoint</span><span class="p">(</span><span class="dl">'</span><span class="s1">medium</span><span class="dl">'</span><span class="p">)</span><span class="s2">` width: 50%; `</span><span class="p">}</span><span class="s2"> `</span><span class="p">;</span> |
Exit fullscreen mode
This will make the Container
component have a width
of 100%
on all screen sizes, and a width
of 50%
on medium screens and above.
You can also use the min-width
and max-width
aliases:
1 2 3 4 5 6 |
<span class="kd">const</span> <span class="nx">Container</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">div</span><span class="s2">` </span><span class="p">${</span><span class="nx">breakpoint</span><span class="p">(</span><span class="dl">'</span><span class="s1">medium</span><span class="dl">'</span><span class="p">,</span> <span class="dl">'</span><span class="s1">max-width</span><span class="dl">'</span><span class="p">)</span><span class="s2">` width: 50%; `</span><span class="p">}</span><span class="s2"> `</span><span class="p">;</span> |
Exit fullscreen mode
This will make the Container
component have a width
of 50%
on medium screens and below.
You can also use the orientation
alias to target specific orientations:
1 2 3 4 5 6 |
<span class="kd">const</span> <span class="nx">Container</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">div</span><span class="s2">` </span><span class="p">${</span><span class="nx">breakpoint</span><span class="p">(</span><span class="dl">'</span><span class="s1">medium</span><span class="dl">'</span><span class="p">,</span> <span class="dl">'</span><span class="s1">orientation</span><span class="dl">'</span><span class="p">,</span> <span class="dl">'</span><span class="s1">landscape</span><span class="dl">'</span><span class="p">)</span><span class="s2">` width: 50%; `</span><span class="p">}</span><span class="s2"> `</span><span class="p">;</span> |
Exit fullscreen mode
This will make the Container
component have a width
of 50%
on screens that are both medium
and in landscape
orientation.
Global Style
Styled-components is a great way to create modular and reusable component designs. However, sometimes you need to create global styles that can be applied to all components on the page. This is where createGlobalStyle comes in.
CreateGlobalStyle is a function that accepts CSS and returns a component. That component will inject the CSS into the page when rendered.
Here is an example:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 |
<span class="k">import</span> <span class="nx">React</span> <span class="k">from</span> <span class="dl">"</span><span class="s2">react</span><span class="dl">"</span> <span class="k">import</span> <span class="p">{</span> <span class="nx">createGlobalStyle</span> <span class="p">}</span> <span class="k">from</span> <span class="dl">"</span><span class="s2">styled-components</span><span class="dl">"</span> <span class="kd">const</span> <span class="nx">GlobalStyle</span> <span class="o">=</span> <span class="nx">createGlobalStyle</span><span class="s2">` body { color: </span><span class="p">${</span><span class="nx">props</span> <span class="o">=></span> <span class="p">(</span><span class="nx">props</span><span class="p">.</span><span class="nx">theme</span> <span class="o">===</span> <span class="dl">"</span><span class="s2">grey</span><span class="dl">"</span> <span class="p">?</span> <span class="dl">"</span><span class="s2">grey</span><span class="dl">"</span> <span class="p">:</span> <span class="dl">"</span><span class="s2">white</span><span class="dl">"</span><span class="p">)}</span><span class="s2">; } `</span> <span class="k">export</span> <span class="k">default</span> <span class="kd">function</span> <span class="nx">Layout</span><span class="p">({</span> <span class="nx">children</span> <span class="p">})</span> <span class="p">{</span> <span class="k">return</span> <span class="p">(</span> <span class="p"><</span><span class="nc">React</span><span class="p">.</span><span class="nc">Fragment</span><span class="p">></span> <span class="p"><</span><span class="nc">GlobalStyle</span> <span class="na">theme</span><span class="p">=</span><span class="s">"grey"</span> <span class="p">/></span> <span class="si">{</span><span class="nx">children</span><span class="si">}</span> <span class="p"></</span><span class="nc">React</span><span class="p">.</span><span class="nc">Fragment</span><span class="p">></span> <span class="p">)</span> <span class="p">}</span> |
Exit fullscreen mode
It’s important to note that createGlobalStyle does not support interpolation like the styled component methods. If you need interpolation, you can use a regular CSS template literal.
CreateGlobalStyle is great for setting global variables, such as colors, fonts, or margins. It’s also a good way to reset the browser’s default styles.
If you’re using createGlobalStyle in a React project, you’ll need to make sure to wrap it in a component so that it only renders once. Otherwise, your styles will be injected every time the component renders.
Using className with styled-components
As we saw in the previous section, styled-components allow us to style our components using the template literal syntax. We can also use the className property to add class names to our styled components.
For example, let’s say we have a styled component like this:
1 2 3 4 5 6 7 8 |
<span class="kd">const</span> <span class="nx">Button</span> <span class="o">=</span> <span class="nx">styled</span><span class="p">.</span><span class="nx">button</span><span class="s2">` background-color: blue.; border: none; border-radius: 3px; color: white; padding: 10px 15px; `</span><span class="p">;</span> |
Exit fullscreen mode
If we want to add a class name to this component, we can do so like this:
1 2 3 4 |
<span class="p"><</span><span class="nc">Button</span> <span class="na">className</span><span class="p">=</span><span class="s">"button"</span><span class="p">></span> Click me! <span class="p"></</span><span class="nc">Button</span><span class="p">></span> |
Exit fullscreen mode
Then we can create our own CSS styles using the class name “.button”. The stylesheet used by the end user won’t be impacted if your CSS-in-JS style changes.
1 2 3 4 5 6 |
<span class="nc">.button</span> <span class="p">{</span> <span class="nl">margin</span><span class="p">:</span> <span class="m">5rem</span> <span class="nb">auto</span><span class="p">;</span> <span class="nl">font-size</span><span class="p">:</span> <span class="m">1.3rem</span><span class="p">;</span> <span class="p">}</span> |
Exit fullscreen mode
Now our button will have the class “button” in addition to the styles defined by the styled component. We can use this to apply additional styling to our components if needed.
Conclusion
So far we’ve looked at advantages and disadvantages of styled-components, and how to use it to style our React components. We’ve also seen how to use media queries and other advanced features to make our styles more responsive.
Styled-components library is a great way to style React components. It’s easy to use and has a lot of powerful features. It helps to avoid css selectors name collision and helps to separate UI logic from business logic. If you’re looking for a way to make styles in your React application more readable and maintainable, definitely check out styled-components!