Skip to content

Instantly share code, notes, and snippets.

@mcramm
Last active August 29, 2015 13:55
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save mcramm/8755952 to your computer and use it in GitHub Desktop.
Save mcramm/8755952 to your computer and use it in GitHub Desktop.
Om version of a simple React widget.
(ns om-intro.core
(:require-macros [cljs.core.async.macros :refer [go]])
(:require [om.core :as om :include-macros true]
[om.dom :as dom :include-macros true]
[cljs.core.async :refer [put! chan <!]]))
(enable-console-print!)
(def app-state (atom {:text "Some Text"
:size 15
:colors {:red 0
:green 0
:blue 0}}))
(defn get-value [owner ref]
(-> (om/get-node owner ref)
.-value))
(defn color-slider [colors owner {:keys [label onChange color-key]}]
(reify
om/IRenderState
(render-state [this {:keys [comm]}]
(dom/div nil
(dom/input #js {:type "range"
:min 0
:max 255
:step 1
:ref "color"
:value (color-key colors)
:onChange #(onChange color-key owner)})
(dom/label nil (str label ": " (color-key colors)))))))
(defn my-widget [app owner]
(reify
om/IInitState
(init-state [this]
{:comm {:string (chan)
:size (chan)
:colors (chan})})
om/IWillMount
(will-mount [this]
(let [{:keys [string size colors] :as comm} (om/get-state owner :comm)]
(go (while true
(let [value (<! string)]
(om/transact! app :text (fn [_] value)))))
(go (while true
(let [value (<! size)]
(om/transact! app :size (fn [_] value)))))
(go (while true
(let [[c value] (<! colors)]
(om/update! app assoc-in [:colors c] value))))))
om/IRenderState
(render-state [this {:keys [comm]}]
(dom/div nil
(dom/input #js {:type "text"
:ref "text"
:value (:text app)
:onChange #(put! (:string comm) (get-value owner "text"))})
(dom/div nil
(dom/input #js {:type "range"
:min 10
:max 50
:step 0.2
:ref "size"
:value (:size app)
:onChange #(put! (:size comm) (get-value owner "size"))})
(dom/label nil (str (:size app) "px")))
(let [putfn (fn [k o]
(put! (:colors comm) [k (get-value o "color")]))]
(apply dom/div nil
(map (fn [[label color-key]]
(om/build color-slider
(:colors app)
{:opts {:label label
:color-key color-key
:onChange putfn}}))
[["Red" :red] ["Green" :green] ["Blue" :blue]])))
(let [size (:size app)
text (:text app)
{:keys [red green blue]} (:colors app)]
(dom/div #js {:style #js {:font-size (str size "px")
:color (str "rgb(" red "," green "," blue ")")}}
text))))))
(om/root
app-state
my-widget
(. js/document (getElementById "app")))
<html>
<body>
<div id="app"></div>
<script src="http://fb.me/react-0.8.0.js"></script>
<script src="out/goog/base.js" type="text/javascript"></script>
<script src="om_intro.js" type="text/javascript"></script>
<script type="text/javascript">goog.require("om_intro.core");</script>
</body>
</html>
<html>
<body>
<div id="app"></div>
<script src="om_intro.release.js" type="text/javascript"></script>
</body>
</html>
(defproject om-intro "0.1.0-SNAPSHOT"
:description "FIXME: write this!"
:url "http://example.com/FIXME"
:dependencies [[org.clojure/clojure "1.5.1"]
[org.clojure/clojurescript "0.0-2138"]
[org.clojure/core.async "0.1.267.0-0d7780-alpha"]
[om "0.3.1"]
[com.facebook/react "0.8.0.1"]]
:plugins [[lein-cljsbuild "1.0.1"]]
:source-paths ["src"]
:cljsbuild {
:builds [{:id "dev"
:source-paths ["src"]
:compiler {
:output-to "om_intro.js"
:output-dir "out"
:optimizations :none
:source-map true}}
{:id "release"
:source-paths ["src"]
:compiler {
:output-to "om_intro.release.js"
:pretty-print false
:optimizations :advanced
:preamble ["react/react.min.js"]
:externs ["react/externs/react.js"]}}]})
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment