Styling React Components
with glamorous 💄
Kent C. Dodds
Utah
1 wife, 3 kids
PayPal, Inc.
@kentcdodds
data:image/s3,"s3://crabby-images/f727d/f727db4c6f5abd2a41042f154fed7b2ec0f19f51" alt=""
Please Stand...
if you are able
data:image/s3,"s3://crabby-images/dba7d/dba7d1dacbff29ad90a279e71b22f833ca6a75a4" alt=""
What this talk is
- Brief intro to CSS-in-JS
- Story of why glamorous is a thing
- Elicit feedback from you all
data:image/s3,"s3://crabby-images/acf40/acf408bdc1d8904cda12798862fda00dfa4025b7" alt=""
What this talk is not
- An attempt to get you to use glamorous
data:image/s3,"s3://crabby-images/fbd92/fbd92c5497fd0628d0f6cc5b1446158d23c322ad" alt=""
Let's
Get
STARTED!
data:image/s3,"s3://crabby-images/b56b1/b56b15c491577b7a3922f572e4ff7af0eb15f27d" alt=""
data:image/s3,"s3://crabby-images/c089c/c089c55905e1ef0ed3e85b234ac9a3d9fc9f4090" alt=""
Our Component:
data:image/s3,"s3://crabby-images/a06e7/a06e7b9e46c56545033081c17d56a0ac3290ae51" alt=""
JavaScript + HTML = JSX
class Toggle extends Component {
state = { toggledOn: false };
handleToggleClick = () => {
this.setState(
({ toggledOn }) => ({ toggledOn: !toggledOn }),
(...args) => {
this.props.onToggle(this.state.toggledOn);
},
);
};
render() {
const { children } = this.props;
const { toggledOn } = this.state;
const active = toggledOn ? 'active' : '';
return (
<button
className={`btn btn-primary ${active}`}
onClick={this.handleToggleClick}
>
{children}
</button>
);
}
}
export default Toggle;
Components!
import { PrimaryButton } from './css-buttons';
class Toggle extends Component {
state = { toggledOn: false };
handleToggleClick = () => {
this.setState(
({ toggledOn }) => ({ toggledOn: !toggledOn }),
(...args) => {
this.props.onToggle(this.state.toggledOn);
},
);
};
render() {
const { children } = this.props;
const { toggledOn } = this.state;
const active = toggledOn ? 'active' : '';
return (
<PrimaryButton
active={active}
onClick={this.handleToggleClick}
>
{children}
</PrimaryButton>
);
}
}
export default Toggle;
Components!
function AppButton({ className = '', active, ...props }) {
return (
<button
className={`btn ${className} ${active ? 'active' : ''}`}
{...props}
/>
);
}
function PrimaryButton({ className = '', ...props }) {
return <AppButton className={`btn-primary ${className}`} {...props} />;
}
export default AppButton;
export { PrimaryButton };
CSS? Where are the styles?
CSS? 🤔
.btn {
display: inline-block;
font-weight: 400;
line-height: 1.25;
text-align: center;
white-space: nowrap;
vertical-align: middle;
user-select: none;
border: 1px solid transparent;
padding: .5rem 1rem;
font-size: 1rem;
border-radius: .25rem;
transition: all .2s ease-in-out;
}
.btn.btn-primary {
color: #fff;
background-color: #0275d8;
border-color: #0275d8;
}
.btn-primary:hover, .btn-primary.active {
background-color: #025aa5;
border-color: #01549b;
}
- Conventions?
- Who's using these styles?
- Can I delete these styles?
- Can I change these styles?
CSS?
The P2P Funnel Page
data:image/s3,"s3://crabby-images/8ee2a/8ee2a0cded9342328e3f6144dad2d36cca1b90b8" alt=""
data:image/s3,"s3://crabby-images/6f2fc/6f2fc58f65909e165fd51a9bf03e0372e102fec6" alt=""
data:image/s3,"s3://crabby-images/25c94/25c940f263af601a84351e8fa63a17c31eaee30e" alt=""
data:image/s3,"s3://crabby-images/b4764/b4764ffa34ef5313fe3fb509fcf8fb18a5f02e41" alt=""
data:image/s3,"s3://crabby-images/3e563/3e56310af06a380751a1b34c3a130e60bd74103f" alt=""
data:image/s3,"s3://crabby-images/4976a/4976aa1a9d434a4af68074cd2705c59e79773222" alt=""
Our CSS...
// import styles for funnel pages control, T2, T3, T4 and T5
.funnel_control,
.funnel_2A,
.funnel_2B,
.funnel_3,
.funnel_4,
.funnel_5_2dot5,
.funnel_5_xb,
.funnel_overseas {
@import "./funnel";
}
// import styles for funnel page T6 and variants
.funnel_6_2dot5,
.funnel_6_xb,
.funnel_7_xb,
.funnel_8_xb,
.funnel_9_xb,
.funnel_10_xb,
.funnel_11_xb,
.funnel_12_xb,
.funnel_7_gb {
@import "./funnelT6";
}
// import styles for ppme entry on funnel pages T5, T6, T7 and T8
.funnel_5_2dot5,
.funnel_5_xb,
.funnel_6_2dot5,
.funnel_6_xb,
.funnel_7_xb,
.funnel_8_xb,
.funnel_9_xb,
.funnel_10_xb,
.funnel_11_xb,
.funnel_12_xb,
.funnel_7_gb {
@import './block-group-entry';
@import './ppme-entry/ppme-entry';
@import './ppme-entry/ppme-entry-buttons';
@import './ppme-entry/ppme-entry-interaction';
}
// ...etc
What impact will my change have elsewhere?
Components
data:image/s3,"s3://crabby-images/1b3c6/1b3c6c4949576520d551eb1afc82e3fcc7e4c91c" alt=""
data:image/s3,"s3://crabby-images/e531b/e531b0e40965ea656fbec18e7aa04ba92d5c294d" alt=""
Default
Styled
Unstyled
data:image/s3,"s3://crabby-images/f4968/f49687a5accc63bd1eade201a3d931f558b08411" alt=""
Focus
Toggled
data:image/s3,"s3://crabby-images/3a125/3a12567f0d37192d10b9a44d170c9f73e31485b8" alt=""
data:image/s3,"s3://crabby-images/26d3a/26d3a2e9800989fa3eb283d78d3d98d3b67b88da" alt=""
data:image/s3,"s3://crabby-images/3a125/3a12567f0d37192d10b9a44d170c9f73e31485b8" alt=""
Components
What is a UI component made of?
HTML
CSS
JS
Component
Enter CSS-in-JS
data:image/s3,"s3://crabby-images/f4b42/f4b42d3a7840426acf539173aa4eb1f3339446e9" alt=""
Remember our CSS?
<style>
.btn {
display: inline-block;
font-weight: 400;
line-height: 1.25;
text-align: center;
white-space: nowrap;
vertical-align: middle;
user-select: none;
border: 1px solid transparent;
padding: .5rem 1rem;
font-size: 1rem;
border-radius: .25rem;
transition: all .2s ease-in-out;
}
.btn.btn-primary {
color: #fff;
background-color: #0275d8;
border-color: #0275d8;
}
.btn-primary:hover, .btn-primary.active {
background-color: #025aa5;
border-color: #01549b;
}
</style>
<script>
function AppButton({ className = '', active, ...props }) {
return (
<button
className={`btn ${className} ${active ? 'active' : ''}`}
{...props}
/>
);
}
function PrimaryButton({ className = '', ...props }) {
return <AppButton className={`btn-primary ${className}`} {...props} />;
}
export default AppButton;
export { PrimaryButton };
</script>
import { css } from 'glamor';
const appButtonClassName = css({
display: 'inline-block',
fontWeight: '400',
lineHeight: '1.25',
textAlign: 'center',
whiteSpace: 'nowrap',
verticalAlign: 'middle',
userSelect: 'none',
border: '1px solid transparent',
padding: '.5rem 1rem',
fontSize: '1rem',
borderRadius: '.25rem',
transition: 'all .2s ease-in-out',
});
const highlightStyles = {
backgroundColor: '#025aa5',
borderColor: '#01549b',
};
const primaryButtonClassName = css({
color: '#fff',
backgroundColor: '#0275d8',
borderColor: '#0275d8',
':hover': highlightStyles,
});
const activeClassName = css(highlightStyles);
function AppButton({ className = '', active, ...props }) {
return (
<button
className={`${appButtonClassName} ${className} ${active ? activeClassName : ''}`}
{...props}
/>
);
}
function PrimaryButton({ className = '', ...props }) {
return (
<AppButton
className={`${primaryButtonClassName} ${className}`}
{...props}
/>
);
}
export default AppButton;
export { PrimaryButton };
Enter glamorous 💄
import glamorous from 'glamorous';
const AppButton = glamorous.button({
display: 'inline-block',
fontWeight: '400',
lineHeight: '1.25',
textAlign: 'center',
whiteSpace: 'nowrap',
verticalAlign: 'middle',
userSelect: 'none',
border: '1px solid transparent',
padding: '.5rem 1rem',
fontSize: '1rem',
borderRadius: '.25rem',
transition: 'all .2s ease-in-out',
});
const highlightStyles = {
backgroundColor: '#025aa5',
borderColor: '#01549b',
};
const PrimaryButton = glamorous(AppButton)(
{
color: '#fff',
backgroundColor: '#0275d8',
borderColor: '#0275d8',
':hover': highlightStyles,
},
({ active }) => (active ? highlightStyles : null),
);
export default AppButton;
export { PrimaryButton };
Solving CSS problems
- Global Namespace ✅
- Dependencies ✅
- Dead Code Elimination ✅
- Minification ✅
- Sharing Constants ✅
- Non-deterministic Resolution ✅
- Isolation ✅
jest-glamor-react
📸
data:image/s3,"s3://crabby-images/b2fc1/b2fc1153cb553493124efef9bf55a5494c10cd4f" alt=""
+ Our codebase now
function Consumer(props) {
return (
<FunnelPage>
<FunnelLinkGroup title={i18n('flexible.group.send')}>
<ConsumerFunnelLink {...funnelProps.buyLink(props)} />
<ConsumerFunnelLink {...funnelProps.sendLink(props)} />
<ConsumerFunnelLink {...funnelProps.xbLink(props)} />
<ConsumerFunnelLink {...funnelProps.giftLink(props)} />
<ConsumerFunnelLink {...funnelProps.massPaymentLink(props)} />
</FunnelLinkGroup>
<FunnelLinkGroup {...funnelProps.requestGroup(props)}>
<ConsumerFunnelLink {...funnelProps.requestLink(props)} />
<ConsumerFunnelLink {...funnelProps.invoiceLink(props)} />
<ConsumerFunnelLink {...funnelProps.poolsLink(props)} />
<ConsumerFunnelLink {...funnelProps.ppmeLink(props)} />
</FunnelLinkGroup>
</FunnelPage>
)
}
// funnelProps are pretty neat-o, here's the buy.js one
import { canSendMoney } from './utils'
const staticProps = {
id: 'nemo_buyLink',
icon: 'register',
descriptionKey: 'flexible.buy.description',
badgeHTMLKey: 'flexible.buy.protectionBadge',
}
export default buy
function buy(props: FunnelProps): {} {
const { country, paymentTypes, redirectToClassicInfo: { shouldRedirectToClassic, classicSendUrl } } = props
const shouldRender = canSendMoney(props) && paymentTypes.includes('PURCHASE')
if (!shouldRender) {
return { shouldRender }
}
const titleKey = country === 'DE' ? 'iWantTo.payDE' : 'flexible.buy.title'
const dynamicProps = shouldRedirectToClassic ? ({
titleKey,
href: classicSendUrl,
name: '',
'data-pagename': '',
'data-pagename2': '',
}) : ({
titleKey,
to: '/myaccount/transfer/buy',
name: 'goodsStart',
'data-pagename': 'main:walletweb:transfer:buy:start',
'data-pagename2': 'main:walletweb:transfer:buy:start:::',
})
return { ...staticProps, ...dynamicProps }
}
// and tests are pretty cool:
import { propsTester } from './helpers/utils'
import buyLink from '../buy'
const defaults = {
input: {
businessAccountInfo: { isSecondaryUser: false, privileges: [] },
paymentTypes: ['PURCHASE'],
redirectToClassicInfo: {
shouldRedirectToClassic: false,
},
},
result: {
id: 'nemo_buyLink',
icon: 'register',
descriptionKey: 'flexible.buy.description',
badgeHTMLKey: 'flexible.buy.protectionBadge',
},
}
propsTester({
fn: buyLink,
fnTests: [
{
input: { ...defaults.input },
result: {
...defaults.result,
titleKey: 'flexible.buy.title',
to: '/myaccount/transfer/buy',
name: 'goodsStart',
'data-pagename': 'main:walletweb:transfer:buy:start',
'data-pagename2': 'main:walletweb:transfer:buy:start:::',
},
},
{
input: {
...defaults.input,
businessAccountInfo: {
isSecondaryUser: true,
privileges: ['send_money'],
},
},
result: {
...defaults.result,
titleKey: 'flexible.buy.title',
to: '/myaccount/transfer/buy',
name: 'goodsStart',
'data-pagename': 'main:walletweb:transfer:buy:start',
'data-pagename2': 'main:walletweb:transfer:buy:start:::',
},
},
]
})
// glossing over some details...
function ConsumerFunnelLink() {
return (
<FunnelLink shouldRender={shouldRender}>
<Anchor verticalSpacing={verticalSpacing} {...anchorProps}>
<Icon icon={icon} svg={svg} />
<Title>{title}</Title>
<ConditionalDescription shouldRender={renderDescription}>
{description}
</ConditionalDescription>
</Anchor>
<DesktopBadge filler={!subtext}>{badge}</DesktopBadge>
<ConsumerSubtext>{subtext}</ConsumerSubtext>
</FunnelLink>
)
}
// and here's the Anchor component:
import glamorousLink from '../../component-factories/glamorous-link'
import { mediaQueries } from '../../../../styles'
import {
spaceChildrenVertically,
spaceChildrenHorizontally,
} from '../../css-utils'
const { phoneLandscapeMin: desktop, phoneLandscapeMax: mobile } = mediaQueries
const onAttention = '&:hover, &:focus, &:active'
const Anchor = glamorousLink(
{
display: 'flex',
'& > *:last-child': {
flex: '0 1 auto',
},
minHeight: 0, // firefox weirdness
[onAttention]: {
textDecoration: 'none',
'& .funnel-description': {
color: '#333333',
},
},
[mobile]: {
padding: '22px 12px 20px 12px',
flexDirection: 'row',
'& > *:last-child': {
flex: 1,
},
...spaceChildrenHorizontally(30),
},
[desktop]: {
marginTop: 12,
flex: 1,
flexDirection: 'column',
justifyContent: 'flex-start',
[onAttention]: {
'& .icon, & .icon-svg': {
color: '#ffffff',
backgroundColor: '#0070ba',
},
},
},
},
({ verticalSpacing = 10 }) => ({
[desktop]: {
...spaceChildrenVertically(verticalSpacing),
},
}),
)
export default Anchor
Now it's a funnel
data:image/s3,"s3://crabby-images/285bd/285bd47387e8bc432cd77c00bbd42870236695ec" alt=""
data:image/s3,"s3://crabby-images/48216/4821685376ca3849786d410722363fdb8a0d2962" alt=""
And code reuse is 👌
const dataPagenameOverrides = {
'data-pagename': 'main:p2p:send::funnel',
'data-pagename2': 'main:p2p:send::funnel:node::',
}
function SendMoney(props) {
return (
<NewUserFunnelPage>
<BackLink />
<glamorous.Div display="flex" justifyContent="space-between">
<div>
<PageTitle>{i18n('newUser.sendMoney.title')}</PageTitle>
<PageSubtitle />
</div>
<SendSVG />
</glamorous.Div>
<LinkContainer>
<LinkCard
{...funnelProps.buyLink(props)}
descriptionHTMLKey="newUser.sendMoney.goodsAndServices.description"
{...dataPagenameOverrides}
/>
<LinkCard {...funnelProps.sendLink(props)} {...dataPagenameOverrides} />
<LinkCard {...funnelProps.xbLink(props)} {...dataPagenameOverrides} />
<LinkCard {...funnelProps.giftLink(props)} {...dataPagenameOverrides} />
<Protection>
<div
dangerouslySetInnerHTML={{
__html: i18n('newUser.sendMoney.purchaseProtection'),
}}
/>
</Protection>
</LinkContainer>
</NewUserFunnelPage>
)
}
Resources
- CSS in JS - Vjeux - "The original"
- glamor - Sunil Pai - css in your javascript
- glamorous - PayPal - React component Styling Solved 💄
- jest-glamor-react - Kent C. Dodds - Jest utilities for Glamor and React
- Code Sandbox - the examples from these slides in an interactive code editor in your browser.
data:image/s3,"s3://crabby-images/23bdf/23bdf30571eeb86b48b9758a2cf26afdcc53291b" alt=""
Thank you!
data:image/s3,"s3://crabby-images/4fa24/4fa24bfd16af0f6fd147036027e57b82e19613e2" alt=""
Styling React Components
By Kent C. Dodds
Styling React Components
The CSS-in-JS movement is strong in the community and we're still figuring things out. But one thing is clear: keeping your UI logic, markup, and styles together to make a single component is a great way to build applications.
- 5,611