First, we put the code in charge of talking to the back end and updating the lights atom in a separated component, ApiLightsGateway:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns kata-lights-out.lights-gateway | |
(:require | |
[cljs-http.client :as http] | |
[cljs.core.async :as async] | |
[com.stuartsierra.component :as component])) | |
(defn- extract-lights [response] | |
(->> response | |
:body | |
(.parse js/JSON) | |
.-lights | |
js->clj)) | |
(defn- post [lights-channel uri params] | |
(async/pipeline | |
1 | |
lights-channel | |
(map extract-lights) | |
(http/post uri {:with-credentials? false :form-params params}) | |
false)) | |
(defprotocol LightsGateway | |
(reset-lights! [this m n]) | |
(flip-light! [this pos])) | |
(defrecord ApiLightsGateway [config] | |
component/Lifecycle | |
(start [this] | |
(println ";; Starting ApiLightsGateway component") | |
this) | |
(stop [this] | |
(println ";; Stopping ApiLightsGateway component") | |
this) | |
LightsGateway | |
(reset-lights! [this m n] | |
(post (:lights-channel this) | |
(:reset-lights-url config) | |
{:m m :n n})) | |
(flip-light! [this [x y]] | |
(post (:lights-channel this) | |
(:flip-light-url config) | |
{:x x :y y}))) | |
(defn make-api-gateway [config] | |
(->ApiLightsGateway config)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns kata-lights-out.lights | |
(:require | |
[reagent.core :as r] | |
[cljs.core.async :as async] | |
[com.stuartsierra.component :as component] | |
[kata-lights-out.lights-gateway :as lights-gateway]) | |
(:require-macros | |
[cljs.core.async.macros :refer [go-loop]])) | |
(def ^:private light-off 0) | |
(defn light-off? [light] | |
(= light light-off)) | |
(defn- listen-to-lights-updates! [{:keys [lights-channel lights]}] | |
(go-loop [] | |
(when-let [new-lights (async/<! lights-channel)] | |
(reset! lights new-lights) | |
(recur)))) | |
(defprotocol LightsOperations | |
(reset-lights! [this m n]) | |
(flip-light! [this pos])) | |
(defrecord Lights [lights-gateway] | |
component/Lifecycle | |
(start [this] | |
(println ";; Starting lights component") | |
(let [this (assoc this | |
:lights-channel (async/chan) | |
:lights (r/atom [])) | |
lights-channel (:lights-channel this) | |
lights-gateway (assoc lights-gateway | |
:lights-channel lights-channel) | |
this (assoc this :lights-gateway lights-gateway)] | |
(listen-to-lights-updates! this) | |
this)) | |
(stop [this] | |
(println ";; Stopping lights component") | |
(async/close! (:lights-channel this)) | |
this) | |
LightsOperations | |
(reset-lights! [this m n] | |
(lights-gateway/reset-lights! (:lights-gateway this) m n)) | |
(flip-light! [this pos] | |
(lights-gateway/flip-light! (:lights-gateway this) pos))) | |
(defn all-lights-off? [lights] | |
(every? light-off? (flatten lights))) | |
(defn make-lights [] | |
(map->Lights {})) |
Next, we used the Lights component from the view code:
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns kata-lights-out.lights-view | |
(:require | |
[reagent.core :as r] | |
[kata-lights-out.lights :as lights])) | |
(def ^:private light-on "1") | |
(def ^:private light-off "0") | |
(defn- all-lights-off-message-content [lights] | |
(if (lights/all-lights-off? lights) | |
"Lights out, Yay!" | |
{:style {:display :none}})) | |
(defn- all-lights-off-message-component [lights] | |
[:div#all-off-msg | |
(all-lights-off-message-content lights)]) | |
(defn- on-light-click [lights-component pos] | |
(lights/flip-light! lights-component pos)) | |
(defn- render-light [light] | |
(if (lights/light-off? light) | |
light-off | |
light-on)) | |
(defn- light-component [lights-component i j light] | |
^{:key (+ i j)} | |
[:button | |
{:on-click #(on-light-click lights-component [i j])} (render-light light)]) | |
(defn- row-lights-component [lights-component i row-lights] | |
^{:key i} | |
[:div (map-indexed (partial light-component lights-component i) row-lights)]) | |
(defn- home-page [lights-component] | |
(fn [] | |
[:div [:h2 "Kata Lights Out"] | |
(map-indexed (partial row-lights-component lights-component) @(:lights lights-component)) | |
[all-lights-off-message-component @(:lights lights-component)]])) | |
(defn mount [lights-component] | |
(r/render | |
[home-page lights-component] | |
(.getElementById js/document "app"))) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
(ns kata-lights-out.core | |
(:require | |
[kata-lights-out.lights-view :as lights-view] | |
[com.stuartsierra.component :as component] | |
[kata-lights-out.lights :as lights] | |
[kata-lights-out.lights-gateway :as lights-gateway])) | |
(enable-console-print!) | |
;; ------------------------- | |
;; Initialize app | |
(defrecord MainComponent [lights-component m n] | |
component/Lifecycle | |
(start [this] | |
(println ";; Starting main component") | |
(lights/reset-lights! lights-component m n) | |
(lights-view/mount lights-component) | |
this) | |
(stop [this] | |
(println ";; Stopping lights component") | |
this)) | |
(defn main-component [m n] | |
(map->MainComponent {:n n :m m})) | |
(defn init! [m n] | |
(component/start | |
(component/system-map | |
:lights-gateway (lights-gateway/make-api-gateway | |
{:reset-lights-url "http://localhost:3000/reset-lights" | |
:flip-light-url "http://localhost:3000/flip-light"}) | |
:lights-component (component/using | |
(lights/make-lights) | |
[:lights-gateway]) | |
:main (component/using | |
(main-component m n) | |
[:lights-component])))) | |
(init! 3 3) |
You can find the code we produced in these two GitHub repositories: the server and the client (see the componentization branch).
You can check the changes we made to componentize the code here (see the commits made on Aug 30, 2016).
As usual it was a great pleasure to do mob programming and learn with the members of Clojure Developers Barcelona.
No comments:
Post a Comment