Recipes can be added
This commit is contained in:
parent
e14fb90e0f
commit
d4fbae9464
15 changed files with 432 additions and 318 deletions
1
.gitignore
vendored
1
.gitignore
vendored
|
@ -1,4 +1,5 @@
|
|||
/target
|
||||
/out
|
||||
/lib
|
||||
/classes
|
||||
/checkouts
|
||||
|
|
23
project.clj
23
project.clj
|
@ -4,34 +4,29 @@
|
|||
:license {:name "Eclipse Public License"
|
||||
:url "http://www.eclipse.org/legal/epl-v10.html"}
|
||||
:dependencies [[org.clojure/clojure "1.5.1"]
|
||||
[org.clojure/clojurescript "0.0-1909"]
|
||||
[core.async "0.1.0-SNAPSHOT"]
|
||||
[http-kit "2.1.8"]
|
||||
[compojure "1.1.5"]
|
||||
[ring/ring-devel "1.2.0"]
|
||||
[ring/ring-core "1.2.0"]
|
||||
[hiccup "1.0.4"]
|
||||
[prismatic/dommy "0.1.1"]
|
||||
[org.clojure/core.async "0.1.0-SNAPSHOT"]
|
||||
[com.novemberain/monger "1.5.0"]]
|
||||
:repositories {"sonatype-oss-public" "https://oss.sonatype.org/content/groups/public/"}
|
||||
:profiles {:dev {:dependencies [[speclj "2.5.0"]
|
||||
[specljs "2.7.4"]
|
||||
[clj-webdriver "0.6.0"]]}}
|
||||
:plugins [[lein-cljsbuild "0.3.2"]
|
||||
[lein-ring "0.8.6"]
|
||||
[speclj "2.5.0"]
|
||||
[specljs "2.7.4"]]
|
||||
:cljsbuild ~(let [run-specs ["phantomjs"
|
||||
"bin/specljs_runner.js"
|
||||
"public/js/grub_dev.js"]]
|
||||
{:builds {:dev {:source-paths ["src/cljs" "spec/cljs"]
|
||||
:compiler {:output-to "public/js/grub_dev.js"
|
||||
[speclj "2.5.0"]]
|
||||
:cljsbuild {:builds {:dev {:source-paths ["src/cljs"]
|
||||
:compiler {:output-dir "out"
|
||||
:output-to "public/js/grub_dev.js"
|
||||
:source-map "public/js/grub_dev.js.map"
|
||||
:optimizations :whitespace
|
||||
:pretty-print true}
|
||||
:notify-command run-specs}
|
||||
:pretty-print false}}
|
||||
:prod {:source-paths ["src/cljs"]
|
||||
:compiler {:output-to "public/js/grub.js"
|
||||
:optimizations :simple}}
|
||||
:test-commands {"test" run-specs}}})
|
||||
:optimizations :simple}}}}
|
||||
:source-paths ["src/clj" "integration"]
|
||||
:test-paths ["spec/clj"]
|
||||
:ring {:handler grub.core/app}
|
||||
|
|
1
public/.gitignore
vendored
1
public/.gitignore
vendored
|
@ -1,2 +1,3 @@
|
|||
js/grub.js
|
||||
js/grub_dev.js
|
||||
js/grub_dev.js.map
|
||||
|
|
|
@ -79,3 +79,52 @@ tr:hover .grub-close {
|
|||
#clear-all-btn {
|
||||
float: right;
|
||||
}
|
||||
|
||||
.recipe-panel {
|
||||
padding: 0px;
|
||||
}
|
||||
|
||||
.recipe-header {
|
||||
margin: 0px;
|
||||
padding: 0px;
|
||||
background-color: #ffffff;
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.recipe-header-input {
|
||||
border: none;
|
||||
box-shadow: none;
|
||||
transition: none;
|
||||
border-bottom: none;
|
||||
}
|
||||
|
||||
.recipe-header-input:focus {
|
||||
border: none;
|
||||
box-shadow: none;
|
||||
transition: none;
|
||||
outline: none;
|
||||
}
|
||||
|
||||
.recipe-done-btn {
|
||||
margin-left: 290px;
|
||||
margin-top: 10px;
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
.recipe-steps {
|
||||
}
|
||||
|
||||
.recipe-steps-input {
|
||||
border: none;
|
||||
box-shadow: none;
|
||||
transition: none;
|
||||
border-bottom: none;
|
||||
resize: none;
|
||||
}
|
||||
|
||||
.recipe-steps-input:focus {
|
||||
border: none;
|
||||
box-shadow: none;
|
||||
transition: none;
|
||||
border-bottom: none;
|
||||
}
|
||||
|
|
|
@ -1,59 +0,0 @@
|
|||
(ns grub.state-spec
|
||||
(:require [specljs.core]
|
||||
[grub.state :as state])
|
||||
(:require-macros [specljs.core :refer [describe it before
|
||||
should= should-contain
|
||||
should-not-be-nil]]
|
||||
[grub.macros :refer [log logs]]))
|
||||
|
||||
(describe
|
||||
"State"
|
||||
|
||||
(describe
|
||||
"event handling:"
|
||||
(before (reset! state/grubs []))
|
||||
|
||||
(describe "Add"
|
||||
(it "should add a grub to the state when an add event comes"
|
||||
(let [test-grub {:_id 12345 :grub "testgrub" :completed true}
|
||||
add-event (assoc test-grub :event :add)]
|
||||
(state/handle-event add-event)
|
||||
(should-contain test-grub @state/grubs))))
|
||||
|
||||
(describe "Complete"
|
||||
(it "should complete a grub in the state when a complete event comes"
|
||||
(let [test-grub {:_id 234243 :grub "testgrub" :completed false}
|
||||
expected-grub (assoc test-grub :completed true)
|
||||
complete-event (-> test-grub
|
||||
(select-keys [:_id])
|
||||
(assoc :event :complete))]
|
||||
(reset! state/grubs [test-grub])
|
||||
(state/handle-event complete-event)
|
||||
(should-contain expected-grub @state/grubs))))
|
||||
|
||||
(describe "Uncomplete"
|
||||
(it "should uncomplete a grub in the state when an uncomplete event comes"
|
||||
(let [test-grub {:_id 234243 :grub "testgrub" :completed true}
|
||||
expected-grub (assoc test-grub :completed false)
|
||||
complete-event (-> test-grub
|
||||
(select-keys [:_id])
|
||||
(assoc :event :uncomplete))]
|
||||
(reset! state/grubs [test-grub])
|
||||
(state/handle-event complete-event)
|
||||
(should-contain expected-grub @state/grubs))))
|
||||
|
||||
(describe "Delete"
|
||||
(it "should delete a grub from the state when a delete event comes"
|
||||
(let [test-grub {:_id 234243 :grub "testgrub" :completed true}
|
||||
delete-event {:_id (:_id test-grub) :event :delete}]
|
||||
(reset! state/grubs [test-grub])
|
||||
(state/handle-event delete-event)
|
||||
(should= [] @state/grubs))))
|
||||
|
||||
(describe "Clear all"
|
||||
(it "should delete all grubs"
|
||||
(let [test-grub {:_id 234243 :grub "testgrub" :completed true}
|
||||
clear-all-event {:event :clear-all}]
|
||||
(reset! state/grubs [test-grub])
|
||||
(state/handle-event clear-all-event)
|
||||
(should= [] @state/grubs))))))
|
|
@ -3,6 +3,8 @@
|
|||
[grub.db :as db]
|
||||
[grub.integration-test :as integration-test]
|
||||
[ring.middleware.reload :as reload]
|
||||
[ring.middleware.file :as file]
|
||||
[ring.util.response :as resp]
|
||||
[compojure.core :refer [defroutes GET POST]]
|
||||
[compojure.handler :as handler]
|
||||
[compojure.route :as route]
|
||||
|
@ -28,6 +30,8 @@
|
|||
(defroutes routes
|
||||
(GET "/ws" [] ws/websocket-handler)
|
||||
(GET "/" [] (index-page))
|
||||
(GET "*/src/cljs/grub/:file" [file] (resp/file-response file {:root "src/cljs/grub"}))
|
||||
(GET "/js/public/js/:file" [file] (resp/redirect (str "/js/" file)))
|
||||
(route/files "/")
|
||||
(route/not-found "<p>Page not found.</p>"))
|
||||
|
||||
|
|
|
@ -6,6 +6,7 @@
|
|||
[clojure.core.async :as async :refer [<! >! >!! chan go close! timeout]]))
|
||||
|
||||
(def grub-collection "grubs")
|
||||
(def recipe-collection "recipes")
|
||||
|
||||
(defn clear-grubs []
|
||||
(mc/drop grub-collection))
|
||||
|
@ -40,6 +41,10 @@
|
|||
(defmethod handle-event :clear-all [event]
|
||||
(clear-grubs))
|
||||
|
||||
(defmethod handle-event :add-recipe [event]
|
||||
(let [recipe (select-keys event [:_id :name :steps])]
|
||||
(mc/insert recipe-collection recipe)))
|
||||
|
||||
(defmethod handle-event :unknown-event [event]
|
||||
(println "Cannot handle unknown event:" event))
|
||||
|
||||
|
@ -59,6 +64,17 @@
|
|||
(>! out grub-event))))
|
||||
out))
|
||||
|
||||
(defn get-current-recipes-as-events []
|
||||
(let [recipes (mc/find-maps recipe-collection)
|
||||
sorted-recipes (sort-by :_id (vec recipes))
|
||||
out (chan)]
|
||||
(go (doseq [recipe sorted-recipes]
|
||||
(let [recipe-event (-> recipe
|
||||
(select-keys [:_id :name :steps])
|
||||
(assoc :event :add-recipe))]
|
||||
(>! out recipe-event))))
|
||||
out))
|
||||
|
||||
(def default-db "grub")
|
||||
|
||||
(defn connect-and-handle-events
|
||||
|
|
|
@ -22,6 +22,9 @@
|
|||
(defn push-current-grubs-to-client [c ws-channel]
|
||||
(copy-chan c (db/get-current-grubs-as-events)))
|
||||
|
||||
(defn push-current-recipes-to-client [c ws-channel]
|
||||
(copy-chan c (db/get-current-recipes-as-events)))
|
||||
|
||||
(defn push-received-events-to-client [c ws-channel]
|
||||
(go-loop (let [event (<! c)
|
||||
event-str (str event)]
|
||||
|
@ -50,6 +53,7 @@
|
|||
(println "Request:" request)
|
||||
(httpkit/on-receive ws-channel #(add-incoming-event % ws-channel-id))
|
||||
(push-current-grubs-to-client c ws-channel)
|
||||
(push-current-recipes-to-client c ws-channel)
|
||||
(push-received-events-to-client c ws-channel))))
|
||||
|
||||
(handle-incoming-events)
|
||||
|
|
|
@ -1,86 +0,0 @@
|
|||
(ns grub.async-utils
|
||||
(:refer-clojure :exclude [map filter])
|
||||
(:require [cljs.core.async :as async :refer [<! >! chan put! alts! close!]])
|
||||
(:require-macros [cljs.core.async.macros :refer [go]]
|
||||
[grub.macros :refer [do-chan]]))
|
||||
|
||||
(defn log [in]
|
||||
(let [out (chan)]
|
||||
(do-chan [e in]
|
||||
(.log js/console e)
|
||||
(>! out e))
|
||||
out))
|
||||
|
||||
(defn put-all! [cs x]
|
||||
(doseq [c cs]
|
||||
(put! c x)))
|
||||
|
||||
(defn fan-out [in cs-or-n]
|
||||
(let [cs (if (number? cs-or-n)
|
||||
(repeatedly cs-or-n chan)
|
||||
cs-or-n)]
|
||||
(go (loop []
|
||||
(let [x (<! in)]
|
||||
(if-not (nil? x)
|
||||
(do
|
||||
(put-all! cs x)
|
||||
(recur))
|
||||
:done))))
|
||||
cs))
|
||||
|
||||
(defn fan-in
|
||||
([ins] (fan-in (chan) ins))
|
||||
([out ins]
|
||||
(go (loop [ins (vec ins)]
|
||||
(when (> (count ins) 0)
|
||||
(let [[x in] (alts! ins)]
|
||||
(when x
|
||||
(>! out x)
|
||||
(recur ins))
|
||||
(recur (vec (disj (set ins) in))))))
|
||||
(close! out))
|
||||
out))
|
||||
|
||||
(defn copy
|
||||
([c]
|
||||
(first (fan-out c 1)))
|
||||
([out c]
|
||||
(first (fan-out c [out]))))
|
||||
|
||||
(defn map [f in]
|
||||
(let [out (chan)]
|
||||
(go (loop []
|
||||
(if-let [x (<! in)]
|
||||
(do (>! out (f x))
|
||||
(recur))
|
||||
(close! out))))
|
||||
out))
|
||||
|
||||
(defn map-filter [f in]
|
||||
(let [out (chan)]
|
||||
(go (loop []
|
||||
(if-let [x (<! in)]
|
||||
(do
|
||||
(when-let [val (f x)]
|
||||
(>! out val))
|
||||
(recur))
|
||||
(close! out))))
|
||||
out))
|
||||
|
||||
(defn filter [pred in]
|
||||
(let [out (chan)]
|
||||
(go (loop []
|
||||
(if-let [x (<! in)]
|
||||
(do (when (pred x) (>! out x))
|
||||
(recur))
|
||||
(close! out))))
|
||||
out))
|
||||
|
||||
(defn siphon
|
||||
([in] (siphon in []))
|
||||
([in coll]
|
||||
(go (loop [coll coll]
|
||||
(if-let [v (<! in)]
|
||||
(recur (conj coll v))
|
||||
coll)))))
|
||||
|
|
@ -1,22 +1,22 @@
|
|||
(ns grub.core
|
||||
(:require [grub.async-utils :as a]
|
||||
[grub.view :as view]
|
||||
(:require [grub.view :as view]
|
||||
[grub.websocket :as ws]
|
||||
[grub.state :as state]
|
||||
[cljs.core.async :refer [<! >! >!! chan close! timeout]]
|
||||
[cljs.core.async :as a :refer [<! >! chan]]
|
||||
[cljs.reader])
|
||||
(:require-macros [grub.macros :refer [log logs go-loop]]
|
||||
[cljs.core.async.macros :refer [go]]))
|
||||
|
||||
(defn handle-grub-events []
|
||||
(a/fan-out view/outgoing-events [state/incoming-events ws/incoming-events])
|
||||
(a/copy state/incoming-events ws/outgoing-events)
|
||||
(a/copy ws/incoming-events state/outgoing-events))
|
||||
(defn wire-channels-together []
|
||||
(let [to-remote (chan)
|
||||
to-state (chan)
|
||||
to-view (chan)
|
||||
from-remote (a/mult (ws/get-remote-chan to-remote))
|
||||
from-view (a/mult (view/setup-and-get-view-events to-view))]
|
||||
(state/handle-incoming-events to-state)
|
||||
(a/tap from-view to-state)
|
||||
(a/tap from-view to-remote)
|
||||
(a/tap from-remote to-state)
|
||||
(a/tap from-remote to-view)))
|
||||
|
||||
(defn init []
|
||||
(ws/connect-to-server)
|
||||
(state/init)
|
||||
(view/init)
|
||||
(handle-grub-events))
|
||||
|
||||
(init)
|
||||
(wire-channels-together)
|
||||
|
|
|
@ -1,7 +1,6 @@
|
|||
(ns grub.dom
|
||||
(:require [grub.async-utils :as a]
|
||||
[dommy.core :as dommy]
|
||||
[cljs.core.async :refer [<! >! chan timeout close!]])
|
||||
(:require [dommy.core :as dommy]
|
||||
[cljs.core.async :as a :refer [<! >! chan]])
|
||||
(:require-macros [grub.macros :refer [log logs go-loop]]
|
||||
[dommy.macros :refer [deftemplate sel1 node]]
|
||||
[cljs.core.async.macros :refer [go]]))
|
||||
|
@ -12,7 +11,7 @@
|
|||
([el type f out]
|
||||
(let [push-fn (fn [e] (when f (f e)) (go (>! out e)))
|
||||
unlisten #(do (dommy/unlisten! el type push-fn)
|
||||
(close! out))]
|
||||
(a/close! out))]
|
||||
(dommy/listen! el type push-fn)
|
||||
{:chan out :unlisten unlisten})))
|
||||
|
||||
|
@ -22,7 +21,7 @@
|
|||
([el type f out]
|
||||
(let [push-fn (fn [e] (when f (f e)) (go (>! out e)))
|
||||
unlisten #(do (dommy/unlisten! el type push-fn)
|
||||
(close! out))]
|
||||
(a/close! out))]
|
||||
(dommy/listen-once! el type push-fn)
|
||||
{:chan out :unlisten unlisten})))
|
||||
|
||||
|
@ -49,10 +48,40 @@
|
|||
(defn grubs-selector []
|
||||
[(sel1 :#grub-list) :.grub-item])
|
||||
|
||||
(defn make-recipe-node [id name steps]
|
||||
(node [:div.panel.panel-default.recipe-panel {:id id}
|
||||
[:div.panel-heading.recipe-header
|
||||
[:input.form-control.recipe-header-input
|
||||
{:id "recipe-name"
|
||||
:type "text"
|
||||
:placeholder "Grub pie"
|
||||
:value name}]]
|
||||
[:div.panel-body.recipe-steps.hidden
|
||||
[:textarea.form-control.recipe-steps-input
|
||||
{:id "recipe-steps"
|
||||
:rows 3
|
||||
:placeholder "2 grubs"
|
||||
:value steps}]]
|
||||
[:button.btn.btn-primary.recipe-done-btn.hidden {:type "button"} "Done"]]))
|
||||
|
||||
(defn add-new-recipe [id name steps]
|
||||
(log "add new recipe:" name)
|
||||
(let [node (make-recipe-node id name steps)
|
||||
recipe-list (sel1 :#recipe-list)]
|
||||
(logs "node:" node)
|
||||
(logs "recipe-list:" recipe-list)
|
||||
(dommy/append! recipe-list node)
|
||||
node))
|
||||
|
||||
(def new-recipe (make-recipe-node "new-recipe" "" ""))
|
||||
|
||||
(defn recipes-selector []
|
||||
[(sel1 :#recipe-list) :.recipe-panel])
|
||||
|
||||
(deftemplate main-template []
|
||||
[:div.container
|
||||
[:div.row.show-grid
|
||||
[:div.col-lg-4]
|
||||
[:div.col-lg-2]
|
||||
[:div.col-lg-4
|
||||
[:h3 "Grub List"]
|
||||
[:div.input-group
|
||||
|
@ -61,7 +90,10 @@
|
|||
add-grub-btn]]
|
||||
[:ul#grub-list.list-group]
|
||||
clear-all-btn]
|
||||
[:div.col-lg-4]
|
||||
[:div.col-lg-4
|
||||
[:h3 "Recipes"]
|
||||
new-recipe
|
||||
[:ul#recipe-list.list-group.recipe-list]]
|
||||
[:div.col-lg-2]]])
|
||||
|
||||
(defn render-body []
|
||||
|
@ -94,6 +126,13 @@
|
|||
(-set-editing! [view])
|
||||
(-unset-editing! [view]))
|
||||
|
||||
(defprotocol IExpandable
|
||||
(-expand! [view])
|
||||
(-unexpand! [view]))
|
||||
|
||||
(defprotocol IClearable
|
||||
(-clear! [view]))
|
||||
|
||||
(extend-type js/HTMLElement
|
||||
IActivatable
|
||||
(-activate! [view]
|
||||
|
@ -117,4 +156,18 @@
|
|||
(-unset-editing! [view]
|
||||
(dommy/remove-class! view :edit)))
|
||||
|
||||
(extend-type js/HTMLDivElement
|
||||
IExpandable
|
||||
(-expand! [view]
|
||||
(dommy/remove-class! (sel1 view ".recipe-steps") :hidden)
|
||||
(dommy/remove-class! (sel1 view ".recipe-done-btn") :hidden))
|
||||
(-unexpand! [view]
|
||||
(dommy/add-class! (sel1 view ".recipe-steps") :hidden)
|
||||
(dommy/add-class! (sel1 view ".recipe-done-btn") :hidden)))
|
||||
|
||||
(extend-type js/HTMLDivElement
|
||||
IClearable
|
||||
(-clear! [view]
|
||||
(dommy/set-value! (sel1 view "#recipe-name") "")
|
||||
(dommy/set-value! (sel1 view "#recipe-steps") "")))
|
||||
|
||||
|
|
|
@ -6,18 +6,3 @@
|
|||
(defmacro logs [& args]
|
||||
(let [strings (map (fn [a] `(pr-str ~a)) args)]
|
||||
`(.log js/console ~@strings)))
|
||||
|
||||
(defmacro go-loop [& body]
|
||||
`(cljs.core.async.macros/go
|
||||
(while true
|
||||
~@body)))
|
||||
|
||||
(defmacro do-chan [[binding chan] & body]
|
||||
`(let [chan# ~chan]
|
||||
(cljs.core.async.macros/go
|
||||
(loop []
|
||||
(if-let [~binding (cljs.core.async/<! chan#)]
|
||||
(do
|
||||
~@body
|
||||
(recur))
|
||||
:done)))))
|
||||
|
|
|
@ -1,11 +1,8 @@
|
|||
(ns grub.state
|
||||
(:require [grub.async-utils :as a]
|
||||
[cljs.core.async :refer [chan <!]])
|
||||
(:require-macros [grub.macros :refer [log logs go-loop]]
|
||||
[cljs.core.async.macros :refer [go]]))
|
||||
(:require [cljs.core.async :as a :refer [chan <!]])
|
||||
(:require-macros [grub.macros :refer [log logs]]
|
||||
[cljs.core.async.macros :refer [go go-loop]]))
|
||||
|
||||
(def incoming-events (chan))
|
||||
(def outgoing-events (chan))
|
||||
|
||||
(def grubs (atom []))
|
||||
|
||||
|
@ -56,9 +53,8 @@
|
|||
(defmethod handle-event :unknown-event [event]
|
||||
(logs "Cannot handle unknown event:" event))
|
||||
|
||||
(defn handle-incoming-events []
|
||||
(go-loop (let [event (<! incoming-events)]
|
||||
(handle-event event))))
|
||||
|
||||
(defn init []
|
||||
(handle-incoming-events))
|
||||
(defn handle-incoming-events [incoming-events]
|
||||
(go-loop []
|
||||
(let [event (<! incoming-events)]
|
||||
(handle-event event)
|
||||
(recur))))
|
||||
|
|
|
@ -1,30 +1,36 @@
|
|||
(ns grub.view
|
||||
(:require [grub.async-utils :as a]
|
||||
[grub.state :as state]
|
||||
(:require [grub.state :as state]
|
||||
[grub.dom :as dom]
|
||||
[dommy.core :as dommy]
|
||||
[cljs.core.async :refer [<! >! chan timeout close!]])
|
||||
(:require-macros [grub.macros :refer [log logs go-loop do-chan]]
|
||||
[cljs.core.async :as a :refer [<! >! chan]])
|
||||
(:require-macros [grub.macros :refer [log logs do-chan]]
|
||||
[dommy.macros :refer [deftemplate sel1 node]]
|
||||
[cljs.core.async.macros :refer [go]]))
|
||||
[cljs.core.async.macros :refer [go go-loop]]))
|
||||
|
||||
(def outgoing-events (chan))
|
||||
(defn re-render-when-state-changes []
|
||||
(add-watch state/grubs
|
||||
:grub-add-watch
|
||||
(fn [key ref old new]
|
||||
(if (empty? new)
|
||||
(dom/-hide! dom/clear-all-btn)
|
||||
(dom/-show! dom/clear-all-btn))
|
||||
(dom/render-grub-list new))))
|
||||
|
||||
(defn get-grubs-from-clicks []
|
||||
(->> (:chan (dom/listen dom/add-grub-btn :click))
|
||||
(a/map #(dom/get-add-grub-text))))
|
||||
(a/map< #(dom/get-add-grub-text))))
|
||||
|
||||
(defn get-grubs-from-enter []
|
||||
(->> (:chan (dom/listen dom/add-grub-text :keyup))
|
||||
(a/filter #(= (.-keyIdentifier %) "Enter"))
|
||||
(a/map #(dom/get-add-grub-text))))
|
||||
(a/filter< #(= (.-keyIdentifier %) "Enter"))
|
||||
(a/map< #(dom/get-add-grub-text))))
|
||||
|
||||
(defn get-created-events []
|
||||
(let [grubs (a/fan-in [(get-grubs-from-clicks)
|
||||
(let [grubs (a/merge [(get-grubs-from-clicks)
|
||||
(get-grubs-from-enter)])]
|
||||
(->> grubs
|
||||
(a/filter #(not (empty? %)))
|
||||
(a/map (fn [g] {:grub g})))))
|
||||
(a/filter< #(not (empty? %)))
|
||||
(a/map< (fn [g] {:grub g})))))
|
||||
|
||||
(defn get-clear-all-events []
|
||||
(:chan (dom/listen dom/clear-all-btn :click)))
|
||||
|
@ -43,7 +49,14 @@
|
|||
|
||||
(defn get-enters []
|
||||
(->> (:chan (dom/listen (sel1 :body) :keyup))
|
||||
(a/filter #(= (.-keyIdentifier %) "Enter"))))
|
||||
(a/filter< #(= (.-keyIdentifier %) "Enter"))))
|
||||
|
||||
(defn get-new-recipe-clicks []
|
||||
(:chan (dom/listen dom/new-recipe :click)))
|
||||
|
||||
(defn get-edit-recipe-clicks []
|
||||
(->> (:chan (dom/listen (dom/recipes-selector) :click))
|
||||
(a/map< (fn [e] (log "edit-recipe-click:" (.-selectedTarget e)) {:elem (.-selectedTarget e)}))))
|
||||
|
||||
(defn parse-completed-event [event]
|
||||
(let [target (.-selectedTarget event)
|
||||
|
@ -52,84 +65,226 @@
|
|||
event-type (if completed :uncomplete :complete)]
|
||||
{:_id id :event event-type}))
|
||||
|
||||
(defn re-render-when-state-changes []
|
||||
(add-watch state/grubs
|
||||
:grub-add-watch
|
||||
(fn [key ref old new]
|
||||
(if (empty? new)
|
||||
(dom/-hide! dom/clear-all-btn)
|
||||
(dom/-show! dom/clear-all-btn))
|
||||
(dom/render-grub-list new))))
|
||||
(defmulti enter-state
|
||||
(fn [old-state new-state-name args]
|
||||
new-state-name)
|
||||
:default :unhandled)
|
||||
|
||||
(defn event-loop []
|
||||
(let [out (chan)
|
||||
created (get-created-events)
|
||||
clear-all (get-clear-all-events)
|
||||
mousedown (get-grub-mousedown-events)
|
||||
mouseup (get-grub-mouseup-events)
|
||||
mouseleave (get-grub-mouseleave-events)
|
||||
body-click (get-body-clicks)
|
||||
edit (chan)
|
||||
enter (get-enters)]
|
||||
(go (loop [mousedown-time nil
|
||||
edit-elem nil]
|
||||
(let [[event c] (alts! [created clear-all mousedown
|
||||
mouseup mouseleave edit body-click
|
||||
enter])]
|
||||
(if edit-elem
|
||||
(cond
|
||||
(or (and (= c body-click)
|
||||
(not (dommy/descendant? (.-target event) edit-elem)))
|
||||
(= c enter))
|
||||
(do (dom/-unset-editing! edit-elem)
|
||||
(let [grub-text (.-value (sel1 edit-elem :.grub-input))
|
||||
id (.-id edit-elem)]
|
||||
(>! out {:event :update
|
||||
:grub grub-text
|
||||
:_id id}))
|
||||
(recur nil nil))
|
||||
(defmethod enter-state :unhandled [old-state new-state-name args]
|
||||
(logs "Unhandled enter transition from " (:name old-state) "to" new-state-name)
|
||||
old-state)
|
||||
|
||||
:else (recur nil edit-elem))
|
||||
(cond
|
||||
(= c created)
|
||||
(defmulti exit-state
|
||||
(fn [state]
|
||||
(:name state))
|
||||
:default :unhandled)
|
||||
|
||||
(defmethod exit-state :unhandled [state]
|
||||
(logs "Unhandled exit transition from " (:name state))
|
||||
state)
|
||||
|
||||
(defn transition [state new-state-name & args]
|
||||
(logs "transition from" (:name state) "to" new-state-name)
|
||||
(-> state
|
||||
(exit-state)
|
||||
(enter-state new-state-name args)
|
||||
(assoc :name new-state-name)))
|
||||
|
||||
(defmulti handle-event
|
||||
(fn [state event]
|
||||
[(:name state) (:event event)])
|
||||
:default [:unhandled-state :unhandled-event])
|
||||
|
||||
(defmethod handle-event [:unhandled-state :unhandled-event] [state event]
|
||||
(logs "Unhandled event [" (:name state) (:event event) "]")
|
||||
state)
|
||||
|
||||
|
||||
|
||||
|
||||
(defmethod handle-event [:default :created] [state event]
|
||||
(let [add-event (-> event
|
||||
(assoc :event :add)
|
||||
(assoc :_id (str "grub-" (.now js/Date)))
|
||||
(assoc :completed false))]
|
||||
(>! out add-event)
|
||||
(go (>! (:out state) add-event))
|
||||
(dom/clear-add-grub-text)
|
||||
(recur mousedown-time edit-elem))
|
||||
state))
|
||||
|
||||
(= c clear-all)
|
||||
(do (>! out {:event :clear-all})
|
||||
(recur mousedown-time edit-elem))
|
||||
(defmethod handle-event [:default :clear-all] [state event]
|
||||
(go (>! (:out state) {:event :clear-all}))
|
||||
state)
|
||||
|
||||
(= c mousedown)
|
||||
(do (dom/-activate! (.-selectedTarget event))
|
||||
(let [now (.now js/Date)]
|
||||
(go (<! (timeout 500))
|
||||
(>! edit {:mousedown-time now :elem (.-selectedTarget event)}))
|
||||
(recur now edit-elem)))
|
||||
(defmethod handle-event [:default :mousedown] [state event]
|
||||
(let [mouseevent (:data event)]
|
||||
(dom/-activate! (.-selectedTarget mouseevent))
|
||||
(let [now (.now js/Date)
|
||||
new-state (assoc state :mousedown-time now)]
|
||||
(go (<! (a/timeout 500))
|
||||
(>! (:edit (:channels state))
|
||||
{:mousedown-time now :elem (.-selectedTarget mouseevent)}))
|
||||
new-state)))
|
||||
|
||||
(= c mouseup)
|
||||
(do (dom/-deactivate! (.-selectedTarget event))
|
||||
(>! out (parse-completed-event event))
|
||||
(recur nil edit-elem))
|
||||
(defmethod handle-event [:default :mouseup] [state event]
|
||||
(dom/-deactivate! (.-selectedTarget (:data event)))
|
||||
(go (>! (:out state) (parse-completed-event (:data event))))
|
||||
(let [new-state (assoc state :mousedown-time nil)]
|
||||
new-state))
|
||||
|
||||
(= c mouseleave)
|
||||
(do (dom/-deactivate! (.-selectedTarget event))
|
||||
(recur nil edit-elem))
|
||||
(defmethod handle-event [:default :mouseleave] [state event]
|
||||
(dom/-deactivate! (.-selectedTarget (:data event)))
|
||||
state)
|
||||
|
||||
(= c edit)
|
||||
(if (and mousedown-time (= (:mousedown-time event) mousedown-time))
|
||||
(do (dom/-set-editing! (:elem event))
|
||||
(recur nil (:elem event)))
|
||||
(recur nil edit-elem))
|
||||
(defmethod handle-event [:default :edit] [state event]
|
||||
(if (and (:mousedown-time state)
|
||||
(= (:mousedown-time event)
|
||||
(:mousedown-time state)))
|
||||
(transition state :edit-grub (:elem event))
|
||||
state))
|
||||
|
||||
:else (recur mousedown-time edit-elem))))))
|
||||
(defmethod handle-event [:default :new-recipe-click] [state event]
|
||||
(transition state :new-recipe))
|
||||
|
||||
(defmethod handle-event [:default :edit-recipe-click] [state event]
|
||||
(transition state :edit-recipe (:elem event)))
|
||||
|
||||
(defmethod handle-event [:default :add-recipe] [state event]
|
||||
(log "handle event add-recipe")
|
||||
(dom/add-new-recipe (:_id event) (:name event) (:steps event))
|
||||
state)
|
||||
|
||||
|
||||
|
||||
|
||||
(defmethod enter-state :edit-grub [old-state new-state-name [edit-elem]]
|
||||
(dom/-set-editing! edit-elem)
|
||||
(assoc old-state :edit-elem edit-elem))
|
||||
|
||||
(defmethod exit-state :edit-grub [state]
|
||||
(let [edit-elem (:edit-elem state)]
|
||||
(dom/-unset-editing! edit-elem)
|
||||
(let [grub-text (.-value (sel1 edit-elem :.grub-input))
|
||||
id (.-id edit-elem)
|
||||
update-event {:event :update :grub grub-text :_id id}
|
||||
new-state (dissoc state :edit-elem)]
|
||||
(go (>! (:out state) update-event))
|
||||
new-state)))
|
||||
|
||||
(defmethod handle-event [:edit-grub :body-click] [state event]
|
||||
(let [clicked-elem (.-target (:data event))
|
||||
edit-elem (:edit-elem state)]
|
||||
(if (dommy/descendant? clicked-elem edit-elem)
|
||||
state
|
||||
(transition state :default))))
|
||||
|
||||
(defmethod handle-event [:edit-grub :enter] [state event]
|
||||
(transition state :default))
|
||||
|
||||
|
||||
|
||||
|
||||
(defmethod enter-state :new-recipe [old-state new-state-name args]
|
||||
(dom/-expand! dom/new-recipe)
|
||||
old-state)
|
||||
|
||||
(defn get-new-recipe-info []
|
||||
(let [name (.-value (sel1 dom/new-recipe "#recipe-name"))
|
||||
steps (.-value (sel1 dom/new-recipe "#recipe-steps"))]
|
||||
(when (not (or (empty? name) (empty? steps)))
|
||||
(let [id (str "recipe-" (.now js/Date))]
|
||||
{:name name :steps steps :_id id}))))
|
||||
|
||||
(defmethod exit-state :new-recipe [state]
|
||||
(dom/-unexpand! dom/new-recipe)
|
||||
(let [recipe-info (get-new-recipe-info)]
|
||||
(if recipe-info
|
||||
(let [recipe-node (dom/add-new-recipe (:_id recipe-info)
|
||||
(:name recipe-info)
|
||||
(:steps recipe-info))]
|
||||
(log "new recipe name:" (:name recipe-info) "steps" (:steps recipe-info))
|
||||
(dom/-clear! dom/new-recipe)
|
||||
(go (>! (:out state) (assoc recipe-info :event :add-recipe)))
|
||||
(assoc state
|
||||
:recipes (assoc (:recipes state) (.-id recipe-node) recipe-node)))
|
||||
state)))
|
||||
|
||||
(defmethod handle-event [:new-recipe :body-click] [state event]
|
||||
(log "new-recipe body click")
|
||||
(let [clicked-elem (.-target (:data event))
|
||||
recipe-panel (sel1 ".recipe-panel")]
|
||||
(if (dommy/descendant? clicked-elem recipe-panel)
|
||||
state
|
||||
(transition state :default))))
|
||||
|
||||
|
||||
(defmethod enter-state :edit-recipe [old-state new-state-name [elem]]
|
||||
(dom/-expand! elem)
|
||||
(assoc old-state :edit-elem elem))
|
||||
|
||||
(defmethod exit-state :edit-recipe [state]
|
||||
(let [recipe-node (:edit-elem state)
|
||||
recipe-name (.-value (sel1 recipe-node "#recipe-name"))
|
||||
recipe-steps (.-value (sel1 recipe-node "#recipe-steps"))]
|
||||
(log "update recipe new name:" recipe-name "new steps:" recipe-steps)
|
||||
(dom/-unexpand! recipe-node)
|
||||
(-> state
|
||||
(dissoc :edit-elem))))
|
||||
|
||||
(defmethod handle-event [:edit-recipe :body-click] [state event]
|
||||
(log "edit-recipe body click")
|
||||
(let [clicked-elem (.-target (:data event))
|
||||
recipe-node (:edit-elem state)]
|
||||
(if (dommy/descendant? clicked-elem recipe-node)
|
||||
state
|
||||
(transition state :default))))
|
||||
|
||||
|
||||
|
||||
|
||||
(defn main-loop [channels]
|
||||
(let [out (chan)
|
||||
events (chan)
|
||||
event-mix (a/mix events)]
|
||||
(doseq [[name c] (seq channels)] (a/admix event-mix c))
|
||||
(go-loop [state {:name :default
|
||||
:channels channels
|
||||
:out out
|
||||
:recipes {}}]
|
||||
(let [event (<! events)]
|
||||
(logs "handle event" (:name state) event)
|
||||
(recur (handle-event state event))))
|
||||
out))
|
||||
|
||||
(defn init []
|
||||
|
||||
|
||||
(defn get-raw-view-channels []
|
||||
{:created (get-created-events)
|
||||
:clear-all (get-clear-all-events)
|
||||
:mousedown (get-grub-mousedown-events)
|
||||
:mouseup (get-grub-mouseup-events)
|
||||
:mouseleave (get-grub-mouseleave-events)
|
||||
:body-click (get-body-clicks)
|
||||
:edit (chan)
|
||||
:enter (get-enters)
|
||||
:new-recipe-click (get-new-recipe-clicks)
|
||||
:edit-recipe-click (get-edit-recipe-clicks)})
|
||||
|
||||
(defn append-event-name-to-channel-events [channels]
|
||||
(into {}
|
||||
(for [[name c] channels]
|
||||
[name (a/map< (fn [e]
|
||||
(if (map? e)
|
||||
(assoc e :event name)
|
||||
{:event name :data e}))
|
||||
c)])))
|
||||
|
||||
(defn get-named-channels [remote-channel]
|
||||
(let [raw-view-channels (get-raw-view-channels)
|
||||
named-view-channels (append-event-name-to-channel-events raw-view-channels)]
|
||||
(assoc named-view-channels :remote-channel remote-channel)))
|
||||
|
||||
|
||||
(defn setup-and-get-view-events [remote-channel]
|
||||
(dom/render-body)
|
||||
(re-render-when-state-changes)
|
||||
(a/copy outgoing-events (event-loop)))
|
||||
(main-loop (get-named-channels remote-channel)))
|
||||
|
|
|
@ -1,31 +1,31 @@
|
|||
(ns grub.websocket
|
||||
(:require [grub.async-utils
|
||||
:refer [fan-in fan-out event-chan filter-chan do-chan do-chan! map-chan]]
|
||||
[cljs.core.async :refer [<! >! >!! chan close! timeout]]
|
||||
(:require [cljs.core.async :as a :refer [<! >! chan]]
|
||||
[cljs.reader])
|
||||
(:require-macros [cljs.core.async.macros :refer [go]]
|
||||
[grub.macros :refer [log logs go-loop]]))
|
||||
|
||||
(def incoming-events (chan))
|
||||
(def outgoing-events (chan))
|
||||
(:require-macros [cljs.core.async.macros :refer [go go-loop]]
|
||||
[grub.macros :refer [log logs]]))
|
||||
|
||||
(def websocket* (atom nil))
|
||||
|
||||
(defn handle-incoming-events []
|
||||
(go-loop
|
||||
(let [event (<! incoming-events)]
|
||||
(.send @websocket* event))))
|
||||
(defn send-outgoing-events [ch]
|
||||
(go-loop []
|
||||
(let [event (<! ch)]
|
||||
(.send @websocket* event)
|
||||
(recur))))
|
||||
|
||||
(defn handle-outgoing-events []
|
||||
(aset @websocket* "onmessage" (fn [event]
|
||||
(defn receive-remote-events []
|
||||
(let [out (chan)]
|
||||
(aset @websocket*
|
||||
"onmessage"
|
||||
(fn [event]
|
||||
(let [grub-event (cljs.reader/read-string (.-data event))]
|
||||
(go (>! outgoing-events grub-event))))))
|
||||
(go (>! out grub-event)))))
|
||||
out))
|
||||
|
||||
(defn connect-to-server []
|
||||
(defn get-remote-chan [to-remote]
|
||||
(let [server-url (str "ws://" (.-host (.-location js/document)) "/ws")]
|
||||
(reset! websocket* (js/WebSocket. server-url))
|
||||
(aset @websocket* "onopen" (fn [event] (log "Connected:" event)))
|
||||
(aset @websocket* "onclose" (fn [event] (log "Connection closed:" event)))
|
||||
(aset @websocket* "onerror" (fn [event] (log "Connection error:" event)))
|
||||
(handle-incoming-events)
|
||||
(handle-outgoing-events)))
|
||||
(send-outgoing-events to-remote)
|
||||
(receive-remote-events)))
|
||||
|
|
Loading…
Reference in a new issue