Skip to content

TehShrike/styled-jsx

 
 

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

17 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

styled-jsx

Build Status XO code style Slack Channel

Full, scoped and component-friendly CSS support for JSX (SSR + browser).

Features

  • Full CSS support, no tradeoffs in power
  • Runtime size of just 500 bytes
  • Complete isolation: Selectors, animations, keyframes
  • Built-in CSS-prefixing
  • Very fast, minimal and efficient transpilation (see below)
  • High-performance runtime-CSS-injection when not server-rendering
  • Future-proof: Equivalent to server-renderable "Shadow CSS"
  • Works like the deprecated <style scoped>, but the styles get injected only once per component

Usage

Firstly, install the package:

$ npm install --save-dev styled-jsx

Next, add styled-jsx/babel to plugins in your babel configuration:

{
  "babel": {
    "plugins": [
      "styled-jsx/babel"
    ]
  }
}

As the last step, simply include <style jsx> in your code:

export default () => (
  <div>
    <p>only this paragraph will get the style :O</p>
    { /* you can include <Component />s here that include
         other <p>s that don't get unexpected styles! */ }
    <style jsx>{`
      p {
        color: red;
      }
    `}</style>
  </div>
)

How It Works

The example above compiles to the following:

import _jsxStyleInject from 'styled-jsx/inject'

export default () => (
  <div>
    <p data-jsx='cn2o3j'>only this paragraph will get the style :O</p>
    { _jsxStyleInject('cn2o3j', `p[data-jsx=cn2o3j] {color: red;}`) }
  </div>
)

Why It Works Like This

Data attributes give us style encapsulation and _jsxStyleInject is heavily optimized for:

  • Injecting styles upon render
  • Only injecting a certain component's style once (even if the component is included multiple times)
  • Keeping track of styles for server-side rendering (discussed in the next section)

Server-Side Rendering

In the server rendering pipeline, you can obtain the entire CSS text of all the combined components by invoking flush:

import flush from 'styled-jsx/flush'

// …
// <render here>
// …

const styles = flush()

for (let id in styles) {
  const css = styles[id]
  console.log(id, css)
}

This API is also available on the client: Instead of returning the CSS text, it returns a reference to the automatically generated <style> tag.

This is useful for performing diffs of elements between top-level render() calls, and ditching style elements that are no longer being used.

Credits

  • Pedram Emrouznejad (rijs) suggested attribute selectors over my initial class prefixing idea.
  • Sunil Pail (glamor) inspired the use of murmurhash2 (minimal and fast hashing) and a efficient style injection logic.
  • Sultan Tarimo built stylis.js, a super fast and tiny CSS parser and compiler.
  • Max Stoiber (styled-components) proved the value of retaining the familiarity of CSS syntax and pointed me to the very efficient stylis compiler (which we forked to very efficiently append attribute selectors to the user's css)
  • Yehuda Katz (ember) convinced me on Twitter to transpile CSS as an alternative to CSS-in-JS.
  • Evan You (vuejs) discussed his Vue.js CSS transformation with me.
  • Henry Zhu (babel) helpfully pointed me to some important areas of the babel plugin API.

Author

Guillermo Rauch (@rauchg) - ▲ZEIT

About

Full CSS support for JSX without compromises

Resources

License

Stars

Watchers

Forks

Packages

No packages published

Languages

  • JavaScript 100.0%