Skip to content
A minimalistic ClojureScript interface to React.js
Clojure CSS JavaScript Shell HTML
Branch: master
Clone or download

Latest commit

Files

Permalink
Type Name Latest commit message Commit time
Failed to load latest commit information.
.clj-kondo Add Kondo config and lint code Feb 5, 2020
.github Update template words May 20, 2016
demo Release alpha2 May 13, 2020
doc Add class vs function benchmark image May 19, 2020
examples Add examples of :r> and :f> shortcuts May 6, 2020
lib Make it easier to use with webpack and node Oct 10, 2015
logo Update README Mar 22, 2020
outsite/public Use a README instead .keep for gh-pages Feb 9, 2015
prerender/sitetools Improve test coverage Mar 22, 2020
site/public Enable advanced build Jun 6, 2017
src Fix key warning with :r> and :f> May 6, 2020
test-environments Set language-out for node module processing builds May 3, 2020
test Fix key warning with :r> and :f> May 6, 2020
.gitignore Enable test coverage reports Mar 22, 2020
CHANGELOG.md Release alpha2 May 13, 2020
CONTRIBUTING.md Fix bad development doc link in CONTRIBUTING.md Jun 30, 2019
LICENSE Rename LICENSE.md to LICENSE Oct 27, 2017
README.md Update README.md Mar 24, 2020
build-docs.sh Deploy site and docs from CircleCI Apr 16, 2018
build-example-site.sh Try fixing docs/master build May 17, 2018
circle.yml Add tests using Cljs bundle output with Karma Apr 16, 2020
codecov.yml Disable codecov patch difference Apr 26, 2020
deps.edn Update cljs deps.edn version, drop transitive clojure dependency May 3, 2020
package-lock.json Update package-lock Apr 16, 2020
package.json Add tests using Cljs bundle output with Karma Apr 16, 2020
prepare-tests.sh Use React 16 and simplify test setup Apr 3, 2018
project.clj Release alpha2 May 13, 2020
run-tests.sh Add tests using Cljs bundle output with Karma Apr 16, 2020
shadow-cljs.edn Add shadow-cljs dev setup Nov 19, 2019

README.md

Reagent

CircleCI Clojars Project codecov cljdoc badge project chat

A simple ClojureScript interface to React.

Reagent provides a way to write efficient React components using (almost) nothing but plain ClojureScript functions.

Usage

To create a new Reagent project using Leiningen template simply run:

lein new reagent myproject

If you wish to only create the assets for ClojureScript without a Clojure backend then do the following instead:

lein new reagent-frontend myproject

This will setup a new Reagent project with some reasonable defaults, see here for more details.

To use Reagent in an existing project you add this to your dependencies in project.clj:

Clojars Project

This is all you need to do if you want the standard version of React. If you want to use your own build of React (or React from a CDN), you have to use :exclusions variant of the dependency, and also provide react and react-dom namespaces (by creating .cljs files with just ns form, or by adding your own :foreign-libs entries).

[reagent "0.x.x" :exclusions [cljsjs/react cljsjs/react-dom]]

Examples

Reagent uses Hiccup-like markup instead of React's sort-of html. It looks like this:

(defn some-component []
  [:div
   [:h3 "I am a component!"]
   [:p.someclass
    "I have " [:strong "bold"]
    [:span {:style {:color "red"}} " and red"]
    " text."]])

Reagent extends standard Hiccup in one way: it is possible to "squeeze" elements together by using a > character.

[:div
  [:p
    [:b "Nested Element"]]]

can be written as:

[:div>p>b "Nested Element"]

Since version 0.8: The :class attribute also supports collections of classes, and nil values are removed:

[:div {:class ["a-class" (when active? "active") "b-class"]}]

You can use one component inside another:

(defn calling-component []
  [:div "Parent component"
   [some-component]])

And pass properties from one component to another:

(defn child [name]
  [:p "Hi, I am " name])

(defn childcaller []
  [child "Foo Bar"])

You mount the component into the DOM like this:

(defn mountit []
  (rd/render [childcaller]
            (.-body js/document)))

assuming we have imported Reagent like this:

(ns example
  (:require [reagent.core :as r]
            [reagent.dom :as rd]))

State is handled using Reagent's version of atom, like this:

(defonce click-count (r/atom 0))

(defn state-ful-with-atom []
  [:div {:on-click #(swap! click-count inc)}
   "I have been clicked " @click-count " times."])

Any component that dereferences a reagent.core/atom will be automatically re-rendered.

If you want do some setting up when the component is first created, the component function can return a new function that will be called to do the actual rendering:

(defn timer-component []
  (let [seconds-elapsed (r/atom 0)]
    (fn []
      (js/setTimeout #(swap! seconds-elapsed inc) 1000)
      [:div
       "Seconds Elapsed: " @seconds-elapsed])))

This way you can avoid using React's lifecycle callbacks like getInitialState and componentWillMount most of the time.

But you can still use them if you want to, either using reagent.core/create-class or by attaching meta-data to a component function:

(defonce my-html (r/atom ""))

(defn plain-component []
  [:p "My html is " @my-html])

(def component-with-callback
  (with-meta plain-component
    {:component-did-mount
     (fn [this]
       (reset! my-html (.-innerHTML (reagent/dom-node this))))}))

See the examples directory for more examples.

Performance

React is pretty darn fast, and so is Reagent. It should even be faster than plain old javascript React a lot of the time, since ClojureScript allows us to skip a lot of unnecessary rendering (through judicious use of React's shouldComponentUpdate).

The ClojureScript overhead is kept down, thanks to lots of caching.

Code size is a little bigger than React.js, but still quite small. The todomvc example clocks in at roughly 79K gzipped, using advanced compilation.

About

The idea and some of the code for making components atom-like comes from pump. The reactive-atom idea (and some code) comes from reflex.

The license is MIT.

You can’t perform that action at this time.