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>
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