CSS-in-JS for ClojureScript
Ask questions on #cljss chat at Clojuarians Slack
- Why write CSS in ClojureScript?
- Features
- How it works
- Installation
- Usage
- Production build
- Roadmap
- Contributing
- Supporters
- License
Writing styles this way has the same benefits as writing components that keep together view logic and presentation. It all comes to developer efficiency and maintainability.
Thease are some resources that can give you more context:
- “A Unified Styling Language” by Mark Dalgleish
- “A Unified Styling Language” (talk) by Mark Dalgleish
- “The road to styled components: CSS in component-based systems” by Glen Maddern
- Automatic scoped styles by generating unique names
- CSS pseudo-classes and pseudo-elements
- CSS animations via
@keyframes
at-rule - Injects styles into
<style>
tag at run-time - Debuggable styles in development (set via
goog.DEBUG
) - Fast, 10000 insertions in ~200ms
defstyles
macro expands into a function which accepts arbitrary number of arguments and returns a string of auto-generated class names that references both static and dynamic styles.
(defstyles button [bg]
{:font-size "14px"
:background-color bg})
(button "#000")
;; "-css-43696 -vars-43696"
Dynamic styles are updated via CSS Variables (see browser support).
defstyled
macro accepts var name, HTML element tag name as a keyword and a hash of styles.
The macro expands into a function which accepts optional hash of attributes and child components, and returns React element. It is available for Om, Rum and Reagent libraries. Each of them are in corresponding namespaces: cljss.om/defstyled
, cljss.rum/defstyled
and cljss.reagent/defstyled
.
A hash of attributes with dynamic CSS values as well as normal HTML attributes can be passed into underlying React element. Reading from attributes hash map can be done via anything that satisfies cljs.core/ifn?
predicate (Fn
and IFn
protocols, and normal functions).
NOTE: Dynamic props that used only to compute styles are also passed onto React element and thus result in adding an unknown attribute on a DOM node. To prevent this it is recommended to use keyword or a function marked with with-meta
so the library can remove those props (see example below).
- keyword value — reads the value from props map and removes matching attribute
with-meta
with a single keyword — passes a value of a specified attribute from props map into a function and removes matching attributewith-meta
with a collection of keywords — passes values of specified attributes from props map into a function in the order of these attributes and removes matching attributes
(defstyled h1 :h1
{:font-family "sans-serif"
:font-size :size ;; reads the value and removes custom `:size` attribute
:color (with-meta #(get {:light "#fff" :dark "#000"} %) :color)} ;; gets `:color` value and removes this attribute
:padding (with-meta #(str %1 " " %2) [:padding-v :padding-h])) ;; gets values of specified attrs as arguments and remove those attrs
(h1 {:size "32px" ;; custom attr
:color :dark ;; custom attr
:padding-v "8px" ;; custom attr
:padding-h "4px" ;; custom attr
:margin "8px 16px"} ;; normal CSS rule
"Hello, world!")
;; (js/React.createElement "h1" #js {:className "css-43697 vars-43697"} "Hello, world!")
Sometimes you want toggle between two values. In this example a menu item can switch between active and non-active styles using :active?
attribute.
(defstyled MenuItem :li
{:color (with-meta #(if % "black" "grey") :active?)})
(MenuItem {:active? true})
Because this pattern is so common there's a special treatment for predicate attributes (keywords ending with ?
) in styles definition.
(defstyled MenuItem :li
{:color "grey"
:active? {:color "black"}})
(MenuItem {:active? true})
:css
attribute allows to define styles inline and still benefit from CSS-in-JS approach.
NOTE: This feature is supported only for Rum/Sablono elements
(def color "#000")
[:button {:css {:color color}} "Button"]
;; (js/React.createElement "button" #js {:className "css-43697 vars-43697"} "Button")
defkeyframes
macro expands into a function which accepts arbitrary number of arguments, injects @keyframes declaration and returns a string that is an animation name.
(defkeyframes spin [from to]
{:from {:transform (str "rotate(" from "deg)")
:to {:transform (str "rotate(" to "deg)")}})
[:div {:style {:animation (str (spin 0 180) "500ms ease infinite")}}]
;; (js/React.createElement "div" #js {:style #js {:animation "animation-43697 500ms ease infinite"}})
font-face
macro allows to define custom fonts via @font-face
CSS at-rule. The macro generates CSS string and injects it at runtime. The syntax is defined in example below.
(require '[cljss.core :refer [font-face]])
(def path "https://fonts.gstatic.com/s/patrickhandsc/v4/OYFWCgfCR-7uHIovjUZXsZ71Uis0Qeb9Gqo8IZV7ckE")
(font-face
{:font-family "Patrick Hand SC"
:font-style "normal"
:font-weight 400
:src [{:local "Patrick Hand SC"}
{:local "PatrickHandSC-Regular"}
{:url (str path ".woff2")
:format "woff2"}
{:url (str path ".otf")
:format "opentype"}]
:unicode-range ["U+0100-024F" "U+1E00-1EFF"]})
inject-global
macro allows to defined global styles, such as to reset user agent default styles. The macro generates CSS string and injects it at runtime. The syntax is defined in example below.
(require '[cljss.core :refer [inject-global]])
(def v-margin 4)
(inject-global
{:body {:margin 0}
:ul {:list-style "none"}
"ul > li" {:margin (str v-margin "px 0")}})
Add to project.clj: [org.roman01la/cljss "1.5.13"]
(defstyles name [args] styles)
name
name of a var[args]
argumentsstyles
a hash map of styles definition
(ns example.core
(:require [cljss.core :refer [defstyles]]))
(defstyles button [bg]
{:font-size "14px"
:background-color bg})
[:div {:class (button "#fafafa")}]
(defstyled name tag-name styles)
name
name of a vartag-name
HTML tag name as a keywordstyles
a hash map of styles definition
Using Sablono templating for React
(ns example.core
(:require [sablono.core :refer [html]]
[cljss.rum :refer [defstyled]]))
(defstyled Button :button
{:padding "16px"
:margin-top :v-margin
:margin-bottom :v-margin})
(html
(Button {:v-margin "8px"
:on-click #(console.log "Click!")}))
Dynamically injected CSS:
.css-43697 {
padding: 16px;
margin-top: var(--css-43697-0);
margin-bottom: var(--css-43697-1);
}
.vars-43697 {
--css-43697-0: 8px;
--css-43697-1: 8px;
}
Set goog.DEBUG
to false
to enable fast path styles injection.
{:compiler
{:closure-defines {"goog.DEBUG" false}}}
- Media Queries syntax
- Server-side rendering
- Pick an issue with
help wanted
label (make sure no one is working on it) - Stick to project's code style as much as possible
- Make small commits with descriptive commit messages
- Submit a PR with detailed description of what was done
A repl for the example project is provided via lein-figwheel.
$ cd example
$ lein figwheel
If using emacs cider - you can also launch the repl using M-x cider-jack-in-clojurescript
.
cljss uses a combination of Clojure and ClojureScript tests. Clojure tests are run via lein test
and ClojureScript tests are run via doo. ClojureScript tests require a valid environment in order to run - PhantomJS being the easiest to install.
Once a valid environment is setup, ClojureScript tests can be run like so:
$ lein doo phantom test once
Or with file watching:
$ lein doo phantom test
To run Clojure and ClojureScript tests at once use the test-all
task:
$ lein test-all
Copyright © 2017 Roman Liutikov
Distributed under the Eclipse Public License either version 1.0 or (at your option) any later version.