Lorem

Everything's a box

In this post iā€™m going to introduce one of the concepts you can use when developing web pages/apps using Theme UI

If youā€™re not familiar with Theme UI take a moment to have a read of the docsā€¦ and if youā€™re not familiar with CSS-in-Js perhaps have a read up on that first.

Iā€™m going to talk about one quite complicated but in my opinion unnecessary part of Theme UI and itā€™s something that took me a while to work out since the docs donā€™t really mention this in a clear and concise way.

Itā€™s my hope that this post might clarify a couple of things and help lower the barrier to entry when using Theme UI. I wonā€™t go into detail about how the CSS properties link to the their respective scales or keysā€¦ thatā€™s a post for another time.

In the docs youā€™ll see a code snippet like this.

/** @jsx jsx */
import { jsx } from 'theme-ui';

export default (props) => (
  <div
    sx={{
      fontWeight: 'bold',
      fontSize: 4,
      color: 'primary',
    }}
  >
    Hello
  </div>
);

When i first saw this i was like ā€œWhat the flip is @jsxā€, and ā€œwhereā€™s the import for Reactā€ ā€¦ and then after a while i read on and started to understand what jsx pragma actually is.

The TLDR version is as follows.

When you include /** @jsx jsx */ in your React component you donā€™t need to import React. This because the jsx pragma kind of includes the functionality to transform JSX for us. The jsx pragma also allows a new ā€œtypeā€ of HTML attribute, itā€™s called the sx prop which can be applied to any normal HTML element.

With the sx prop you can now style you UI using Theme UIā€™s super powers.

If youā€™d like to know about more, have a read of this: What is JSX Pragma

No more naming of CSS classes, or importing global variables šŸ¤¢ and in no way will we need to worry about order of specificity šŸ™Œā€¦ nice ay!

Butā€¦

Importing jsx and having to explain the jsx pragma is all a bit unnecessary so iā€™m proposing we try this another way.

Instead of doing what the docs say, try this approach instead.

import React from 'react';
import { Box } from 'theme-ui';

export default (props) => (
  <Box
    sx={{
      fontWeight: 'bold',
      fontSize: 4,
      color: 'primary',
    }}
  >
    Hello
  </Box>
);

You can see from the above that we donā€™t need to import the jsx pragma and instead we can import React as we normally would and then use one of the components that ships with Theme UIā€¦ The <Box />

The Box is technically just a div but if you inspect a Theme UI element, thereā€™s a few things Theme UI does for us which will save us time later.

JavaScript for Box šŸ‘‡

<Box>I'm a Box</Box>

JavaScript for div šŸ‘‡

<div>I'm a div</div>
Iā€™m a div

CSS for Box šŸ‘‡

display: block;
box-sizing: border-box;
margin: 0;
min-width: 0;

CSS for div šŸ‘‡

display: block;

You can see from the above that the CSS for the <Box /> includes some resets for us, eg. box-sizing,margin and min-width and whilst in the past we could have used a global CSS file that handles the resets, we donā€™t want to use global CSS because this is where problems arise.

Global styles and the dreaded !important are escape hatches. These are work arounds weā€™ve developed over the years to compensate for some of the shortcomings of native CSS. But in the new world of CSS-in-Js these native shortcomings have been removed which allows us to spend more time focussing on actually building something.

But wait, thereā€™s more, hereā€™s the <Flex /> component that ships with Theme UI

JavaScript for Flex šŸ‘‡

<Flex>I'm a Flex</Flex>

CSS for Flex šŸ‘‡

display: flex;
box-sizing: border-box;
margin: 0;
min-width: 0;

ā€¦and you can see from the CSS snippet that the defaults that come with this component save us the time of adding them ourselves. e.g <Flex /> already has display: flex; Cool ay! šŸ˜Ž

One last thing. Theme UI also allows the Polymorphic as prop. Poly meaning many, and morphic meaning forms so you can do things like this.

JavaScript for aside šŸ‘‡

<Box as='aside'>I'm an aside</Box>

CSS for aside šŸ‘‡

display: block;
box-sizing: border-box;
margin: 0;
min-width: 0;

Youā€™ll see if you inspect the ā€œIā€™m man asideā€ element that the actual HTML element is in fact an <aside /> and it still inherits all the resets from <Box />

The as props works with all manner of HTML elementsā€¦ even inputs:

<Box as='input' placeholder="I'm an input" />

ā€¦ but donā€™t do as above, use the <Input /> component instead because it maps to the correct key in the theme object: forms.input

<Input placeholder="I'm an input" />

So now you know.

Everythingā€™s a <Box />ā€¦ sort of