Build web UIs in Clojure/Script, the easy way.
-
Zero components are Web Components, which means they're easy to use from anywhere, including:
- Raw HTML
- JavaScript DOM API
- Any frontend framework
Besides their reusability, Web Components have many useful features that are lacking from traditional React-style components, for example:
- Slots are great for boosting reconciliation performance
- Stylesheet encapsulation means components can include their own stylesheets, without affecting anything else on the page
- An enclosing DOM node to which custom styling and event handlers can be attached
-
Components can be (partially or fully) rendered to raw HTML.
- Component markup is rendered into a declarative shadow DOM.
- Structured data can be rendered to element attributes in a built-in format designed for that purpose.
-
A robust data-oriented state management system makes building complex components a breeze.
- Component view functions can be pure, and generate pure data; easy to test and reason about
- State management constructs are themselves data, they can be serialized/deserialized, compared semantically, etc.
(ns zero.demos.bullet-list "
A simple bullet list.
"
(:require
[zero.core :refer [<<ctx << act css] :as z]
[zero.dom :as zd]
[zero.config :as zc]))
(def ^:private style
(css "
.option {
font-size: 1rem;
display: flex;
align-items: center;
text-align: left;
padding: 0 0.5rem;
border: none;
background: none;
margin: 0.5rem 0;
}
.option:hover .bullet {
background-color: #e5e5e5;
}
.bullet {
border-radius: 1000px;
width: 1rem;
height: 1rem;
border: 1px solid #d4d4d4;
background-color: #fafafa;
display: flex;
align-items: center;
justify-content: center;
margin-right: 0.75rem;
}
.bullet.selected {
border: 1px solid #3182ce;
background-color: transparent;
}
.bullet .iris {
border-radius: 1000px;
background-color: #3182ce;
}
"))
(defn view [{:keys [options value]}]
[:root>
:#css css
:#style {:display "block"}
(map
(fn [option]
[:button.option
:#on {:click (act [::zd/dispatch :value :data (:value option)])}
[:div.bullet
:#class (when (= (:value option) value) :selected)
(when (= (:value option) value)
[:div.iris
:#style {:height "0.75rem" :width "0.75rem"}])]
(:view option)])
options)])
(zc/reg-components
:bullet-list
{:props #{:options :value}
:view view})
Add something akin to the following somewhere in your boot up logic:
(zero.config/install! zero.config/!default-db)
;; only for browsers, sets up the web component registry
(zero.wcconfig/install! zero.config/!default-db)
;; only for browsers, and optional, adds some DOM utilities and convenient components
(zero.dom/install! zero.config/!default-db)
Register components with zero.config/reg-components
.
On the browser side, registered components will be added to the browser's web component registry. So any time the browser attaches a matching DOM element to the page, your registered component will be used.
When rendering to HTML, component views are rendered into declarative shadow DOMs. Attributes for registered Zero components are serialized as CDF (Concise Data Format), which allows for seamless transfer of structured data to browser-side component implementations.
Here are a few resources to help learn the basics:
And some (only one for now) examples:
You can also browse c0 (a library of Zero components) for examples.
Feel free to reach out in the #zero-lib channel on the Clojurians slack for any help, questions, feedback, etc.