StyleSheet is a library to author styles in JavaScript.
It is fast and generates optimized, tiny bundles by compiling rules to atomic CSS that can then be extracted to .css file with a Babel plugin.
Play with StyleSheet on CodeSandbox!
import { StyleSheet, StyleResolver } from 'style-sheet'
const styles = StyleSheet.create({
one: {
color: 'red',
},
another: {
color: 'green'
}
})
const className = StyleResolver.resolve([styles.one, styles.another])
Instead of making use of the Cascade, StyleSheet resolves styles deterministically based on their application order.
StyleResolver.resolve([styles.one, styles.another])
// color is green
StyleResolver.resolve([styles.another, styles.one])
// color is red
StyleResolver.resolve
works like Object.assign
and merges rules right to left.
StyleSheet comes with built-in support for pseudo classes and media queries, i18n, React and customizable css
prop.
The StyleSheet library API is highly inspired to React Native and React Native for Web's and implements a styling solution that is similar to the one used in the new facebook.com website:
This sounds very similar to what we use internally at Facebook for the new version of the site :) "Atomic" CSS via a CSS-in-JS library, that's extracted to static CSS files. Building the New facebook.com touches on it (around 28:40 in the video).
— Daniel Lo Nigro (@Daniel15) Software Engineer at Facebook August 12, 2019
Firstly, install the package:
npm i --save style-sheet
The package exposes a StyleSheet
and StyleResolver
instances that are used to respectively create rulesets and resolve (apply) them to class names.
import { StyleSheet, StyleResolver } from 'style-sheet'
Use StyleSheet.create
to create a style object of rules.
const styles = StyleSheet.create({
one: {
color: 'red',
},
another: {
color: 'green'
}
})
And StyleResolver.resolve
to consume the styles
:
StyleResolver.resolve([styles.one, styles.another])
Remember the order in which you pass rules to StyleResolver.resolve
matters!
StyleSheet supports simple state selectors, media queries and shallow combinator selectors like:
const styles = StyleSheet.create({
root: {
color: 'red',
'&:hover' { // state selector
color: 'green'
},
':focus > &': { // shallow combinator selector
color: 'green'
},
':focus + &': { // shallow combinator selector
color: 'blue'
},
'@media (min-width: 678px)': { // media query
color: 'yellow'
}
},
})
StyleSheet.create
converts rules to arrays of atomic CSS classes. Every atomic CSS class corresponds to a declaration inside of the rule:
const rules = StyleSheet.create({
rule: {
display: 'block', // declaration
color: 'green' // declaration
}
})
StyleResolver.resolve
then, accepts a single rule or an array of rules and it will merge them deterministically in application order (left to right). Finally it inserts the computed styles into the page.
To make sure that styles are resolved deterministically some rules apply:
- Shorthand properties are inserted first.
- Longhand properties override shorthands, always!
- States are sorted as follow:
link
,visited
,hover
,focus-within
,focus-visible
,focus
,active
meaning thatactive
overridesfocus
andhover
for example. - Shorthand and longhand properties used inside of combinator selectors are inserted after their corrispective regular groups.
- Media queries are sorted in a mobile first manner.
For simplicity sake, generally we encourage not use these advanced selectors and simply resolve rules conditionally at runtime based on application state. Note that this won't stop you from extracting styles to .css file!
To render on the server, you can access the underlying style sheet that the library is using at any time with StyleResolver.getStyleSheet()
.
This method returns an ordered StyleSheet that exploses two methods:
-
getTextContent
to get the atomic CSS for the rules that have been resolved. -
flush
togetTextContent
and clear the stylesheet - useful when a server deamon is rendering multiple pages.
import { StyleResolver } from 'style-sheet'
const html = `
<!doctype html>
<html>
<head>
<meta charset=utf-8>
<title>my app</title>
<style id="__style_sheet__">${StyleResolver.getStyleSheet().flush()}</style>
</head>
<body>
<main>${renderedHTML}</main>
</body>
</html>
`
By setting the id
attribute to __style_sheet__
StyleSheet can hydrate styles automatically on the client.
StyleSheet comes with a Babel plugin that can extract static rules. This means that your styles are not computed at runtime or in JavaScript and can be served via link
tag.
Just add style-sheet/babel
to plugins
in your babel configuration:
{
"plugins": [
"style-sheet/babel"
]
}
and compile your JavaScript files with Babel.
Once Babel is done compiling you can import the getCss
function from style-sheet/babel
to get the extracted CSS:
import { writeFileSync } from 'fs'
import { getCss } from 'style-sheet/babel'
const bundleFilePath = './build/bundle.css'
writeFileSync(bundleFilePath, getCss())
In your page then you can reference the bundleFilePath
:
const html = `
<!doctype html>
<html>
<head>
<meta charset=utf-8>
<title>my app</title>
+ <link id="__style_sheet__" rel="stylesheet" href="https://raw.githubusercontent.com/giuseppeg/style-sheet/master/./build/bundle.css">
</head>
<body>
<main>${renderedHTML}</main>
</body>
</html>
`
Note that StyleSheet can also reconcile extracted styles!!! You just need to make sure that the link
tag has the __style_sheet__
set, and keep in mind that CORS apply.
When the Babel plugin can't resolve styles to static, it flags them as dynamic and it leaves them in JavaScript. For this reason it is always a good idea to define dynamic styles in separate rules.
By default the plugin looks for references to StyleSheet
when they are imported from style-sheet
. However both can be configured, via plugin options:
-
importName
- default isStyleSheet
and the plugin looks forStyleSheet.create
. -
packageName
- default isstyle-sheet
but when using advanced features (see below) you can point to your custom setup. -
stylePropName
- default iscss
. In React the plugin looks for inline styles defined via this prop and extracts them. -
stylePropPackageName
- mandatory. When using the style prop you need to set this path to point to where you setup your customcreateElement
(see below). -
rtl
- boolean. When set generates I18n styles and extracts them too.
{
"plugins": [
[
"style-sheet/babel",
{
"importName": "StyleSheet",
"packageName": "./path/to/customInstance",
"stylePropName": "css",
"stylePropPackageName": "./path/to/createElement.js",
"rtl": true
}
]
]
}
This is useful when StyleSheet is useds in custom ways like described in the advanced usage section.
In your webpack configuration create a small plugin to wrap the getCss
function:
const styleSheet = require('style-sheet/babel')
const { RawSource } = require('webpack-sources')
const bundleFilenamePath = 'style-sheet-bundle.css'
class StyleSheetPlugin {
apply(compiler) {
compiler.plugin('emit', (compilation, cb) => {
compilation.assets[bundleFilenamePath] = new RawSource(styleSheet.getCss())
cb()
})
}
}
and register an instance of it in the plugins
section of the webpack configuration:
// class StyleSheetPlugin {
// ...
// }
module.exports = {
// ...
module: {
rules: {
test: /\.jsx?$/,
use: {
loader: 'babel-loader',
options: {
plugins: [styleSheet.default]
}
}
}
},
plugins: [
new StyleSheetPlugin()
],
// ...
}
Remember to also register the Babel plugin if you are using Babel via webpack.
That's it! webpack will write bundleFilenamePath
in your public assets folder.
StyleSheet ships with CommonJS, ESM and UMD bundles respectively available at:
lib/cjs
lib/esm
lib/umd
Throughout the readme we will use lib/esm
in the examples that require you to point to individual modules manually.
StyleSheet comes with a factory to generate an instance of StyleSheet
and StyleResolver
. The factory available at style-sheet/lib/umd/factory
and can be used to have fine control over the style sheets creation and support unusual cases like rendering inside of iframes.
More documentation to come, please refer to the implementation in src/factory.js
and see how it is used in src/index.js
.
StyleSheet is framework agnostic but it works well with React.
import React from 'react'
import { StyleSheet, StyleResolver } from 'style-sheet'
export default ({ children }) => {
const className = StyleResolver.resolve([styles.root, styles.another])
return (
<div className={className}>{children}</div>
)
}
const styles = StyleSheet.create({
root: {
color: 'red',
},
another: {
color: 'green'
}
})
StyleSheet provides an helper to create a custom createElement
function that adds support for a styling prop to React. By default this prop is called css
(but its name can be configured) and it allows you to define "inline styles" that get compiled to real CSS and removed from the element. These are also vendor prefixed and scoped.
Note that when applying styles, className
takes always precedence over the style prop. This allows parent components to pass styles such as overrides to children.
To use this feature you need to create an empty file in your project, name it createElement.js
and add the following code:
import * as StyleSheet from 'style-sheet'
import setup from 'style-sheet/lib/esm/createElement'
const stylePropName = 'css'
export const createElement = setup(StyleSheet, stylePropName)
and then instruct Babel to use this method instead of the default React.createElement
. This can be done in two ways:
- Adding the
/* @jsx createElement */
at the top of every file
/* @jsx createElement */
import React from 'react'
import createElement from './path/to/createElement.js'
export default ({ children }) => (
<div css={{ color: 'red' }}>{children}</div>
)
- In your Babel configuration
{
"plugins": [
["@babel/plugin-transform-react-jsx", {
"pragma": "createElement" // React will use style-sheet's createElement
}],
["style-sheet/babel", {
"stylePropName": "css",
"stylePropPackageName": "./path/to/createElement.js"
}]
]
}
or if you use @babel/preset-react
{
"presets": [
[
"@babel/preset-react",
{
"pragma": "createElement" // React will use style-sheet's createElement
}
]
],
"plugins": [
["style-sheet/babel", {
"stylePropName": "css",
"stylePropPackageName": "./path/to/createElement.js"
}]
]
}
When possible, the Babel plugin will hoist and extract to .css file the style prop!
StyleSheet comes with built-in support for i18n and RTL.
In order for i18n to work StyleSheet requires you to define and set an i18n manager that is an object with two properties:
import { setI18nManager } from 'style-sheet'
const i18nManager = {
isRTL: true, // Boolean
doLeftAndRightSwapInRTL: true // Boolean
}
setI18nManager(i18nManager)
In your app you can then toggle isRTL
and (important) you need to re-render the application yourself i.e. isRTL
is not a reactive property and styles don't resolve automatically when you change direction in the i18n manager. In a React application the i18n manager would likely be kept in state and consumed via context.
I18n works with server side rendering and static extraction too!
Please refer to the contributing guidelines document.
Feel free to contact me if you want to help with any of the following tasks (sorted in terms on priority/dependency):
- [ ] Find a better/smaller deterministic name scheme for classes (right now it is
dss_hashedProperty-hashedValue
) - [ ] Consider adding support for i18n properties like
marginHorizontal
- [ ] Add support for
StyleSheet.createPrimitive
(orcreateRule
) to generate non-atomic rules that can be used for the primitives' base styling (and avoid too many atomic classes on elements)
Thanks to:
- Matt Hamlin for transferring ownership of the npm package to us.
- Callstack.io/linaria for providing the evaluation library to extract styles to static.
MIT