Render grub list with om

This commit is contained in:
Nicholas Kariniemi 2014-06-29 12:35:35 +03:00
parent 594408ede2
commit 24632d99ce
9 changed files with 17237 additions and 226 deletions

View file

@ -1,161 +1,165 @@
html, body { html, body {
height: 100%; height: 100%;
min-height: 100%; min-height: 100%;
width: 100%; width: 100%;
min-width: 100%; min-width: 100%;
margin: 0px; margin: 0px;
} }
h3 { h3 {
margin-top: 10px; margin-top: 10px;
} }
.recipes-title { .recipes-title {
clear: right; clear: right;
} }
.recipe-list { .recipe-list {
margin-top: 10px; margin-top: 10px;
} }
#recipe-grubs { #recipe-grubs {
width: 100%; width: 100%;
} }
.recipe-add-grubs-btn { .recipe-add-grubs-btn {
float: right; float: right;
clear: both; clear: both;
margin: 2px; margin: 2px;
} }
.hidden { .hidden {
display: none; display: none;
} }
.leftmost-column { .leftmost-column {
margin-bottom: 45px; margin-bottom: 45px;
} }
.grub-close { .grub-close {
display: none; display: none;
} }
tr:hover .grub-close { tr:hover .grub-close {
display: block; display: block;
} }
#grub-list { #grub-list {
margin-top: 10px; margin-top: 10px;
margin-bottom: 10px; margin-bottom: 10px;
}
.add-grub-input-form .input-group-btn {
width: 100%;
} }
.grub-item { .grub-item {
cursor: default; cursor: default;
-webkit-user-select: none; -webkit-user-select: none;
-moz-user-select: none; -moz-user-select: none;
-ms-user-select: none; -ms-user-select: none;
user-select: none; user-select: none;
} }
.grub-item.grub-active { .grub-item.grub-active {
color: #ffffff; color: #ffffff;
background-color: rgb(71, 73, 73); background-color: rgb(71, 73, 73);
} }
.grub-item .input-span { .grub-item .input-span {
display: block; display: block;
overflow: hidden; overflow: hidden;
} }
.grub-item .grub-static { .grub-item .grub-static {
display: inline; display: inline;
} }
.grub-item .grub-input { .grub-item .grub-input {
display: none; display: none;
} }
.grub-item.edit .grub-static { .grub-item.edit .grub-static {
display: none; display: none;
} }
.grub-item.edit .grub-input { .grub-item.edit .grub-input {
display: inline; display: inline;
width: 100%; width: 100%;
} }
.grub-item .glyphicon { .grub-item .glyphicon {
margin-right: 5px; margin-right: 5px;
} }
.completed { .completed {
text-decoration: line-through; text-decoration: line-through;
color: #a9a9a9; color: #a9a9a9;
} }
.completed.edit { .completed.edit {
text-decoration: none; text-decoration: none;
color: default; color: default;
} }
.panel-body { .panel-body {
padding: 10px; padding: 10px;
padding-top: 0px; padding-top: 0px;
} }
.panel-heading { .panel-heading {
background-color: #ffffff; background-color: #ffffff;
} }
.recipe-panel { .recipe-panel {
padding: 0px; padding: 0px;
margin-bottom: -1px; margin-bottom: -1px;
} }
.recipe-panel > .recipe-header { .recipe-panel > .recipe-header {
background-color: #ffffff; background-color: #ffffff;
} }
.recipe-header { .recipe-header {
margin: 0px; margin: 0px;
padding: 0px; padding: 0px;
border-bottom: none; border-bottom: none;
} }
.recipe-header-input { .recipe-header-input {
border: none; border: none;
box-shadow: none; box-shadow: none;
transition: none; transition: none;
border-bottom: none; border-bottom: none;
width: inherit; width: inherit;
display: inline; display: inline;
} }
.recipe-header-input:focus { .recipe-header-input:focus {
border: none; border: none;
box-shadow: none; box-shadow: none;
transition: none; transition: none;
outline: none; outline: none;
} }
.recipe-btn { .recipe-btn {
margin-top: 10px; margin-top: 10px;
} }
.recipe-grubs { .recipe-grubs {
} }
.recipe-grubs-input { .recipe-grubs-input {
border: none; border: none;
box-shadow: none; box-shadow: none;
transition: none; transition: none;
border-bottom: none; border-bottom: none;
resize: none; resize: none;
} }
.recipe-grubs-input:focus { .recipe-grubs-input:focus {
border: none; border: none;
box-shadow: none; box-shadow: none;
transition: none; transition: none;
border-bottom: none; border-bottom: none;
} }

16644
public/js/react-0.9.0.js Normal file

File diff suppressed because it is too large Load diff

21
public/js/react-0.9.0.min.js vendored Normal file

File diff suppressed because one or more lines are too long

View file

@ -25,7 +25,8 @@
(html5 (html5
index-page-header index-page-header
[:body [:body
(include-js "http://fb.me/react-0.9.0.js") [:div#container]
(include-js "/js/react-0.9.0.min.js")
(include-js "/js/jquery.js") (include-js "/js/jquery.js")
(include-js "/js/bootstrap.js") (include-js "/js/bootstrap.js")
(include-js "/js/grub.js")])) (include-js "/js/grub.js")]))
@ -34,7 +35,8 @@
(html5 (html5
index-page-header index-page-header
[:body [:body
(include-js "http://fb.me/react-0.9.0.js") [:div#container]
(include-js "/js/react-0.9.0.js")
(include-js "/js/out/goog/base.js") (include-js "/js/out/goog/base.js")
(include-js "/js/jquery.js") (include-js "/js/jquery.js")
(include-js "/js/bootstrap.js") (include-js "/js/bootstrap.js")

View file

@ -1,5 +1,5 @@
(ns grub.core (ns grub.core
(:require [grub.view :as view] (:require [grub.state :as state]
[grub.websocket :as ws] [grub.websocket :as ws]
[cljs.core.async :as a :refer [<! >! chan]]) [cljs.core.async :as a :refer [<! >! chan]])
(:require-macros [grub.macros :refer [log logs go-loop]] (:require-macros [grub.macros :refer [log logs go-loop]]
@ -7,10 +7,10 @@
(defn wire-channels-together [] (defn wire-channels-together []
(let [to-remote (chan) (let [to-remote (chan)
to-view (chan) to-state (chan)
from-remote (ws/get-remote-chan to-remote) from-remote (ws/get-remote-chan to-remote)
from-view (view/setup-and-get-view-events to-view)] from-state (state/update-state-and-render to-state)]
(a/pipe from-remote to-view) (a/pipe from-remote to-state)
(a/pipe from-view to-remote))) (a/pipe from-state to-remote)))
(wire-channels-together) (wire-channels-together)

59
src/cljs/grub/state.cljs Normal file
View file

@ -0,0 +1,59 @@
(ns grub.state
(:require [grub.view :as view]
[cljs.core.async :as a :refer [<! >! chan]])
(:require-macros [grub.macros :refer [log logs]]
[cljs.core.async.macros :refer [go go-loop]]))
(def default-app-state {:grubs {}
:recipes {}})
(defmulti handle-event (fn [event state] (:event event))
:default :unknown-event)
(defmethod handle-event :unknown-event [event grubs]
(logs "Cannot handle unknown event:" event)
grubs)
(defn new-grub [id grub completed]
{:id id :grub grub :completed completed})
(defmethod handle-event :add-grub [event grubs]
(let [grub (new-grub (:id event) (:grub event) (:completed event))]
(assoc grubs (:id grub) grub)))
(defn assoc-new-grub [current new]
(assoc current (:id new)
(new-grub (:id new) (:grub new) (:completed new))))
(defn make-add-grubs-map [grub-events]
(reduce assoc-new-grub {} grub-events))
(defmethod handle-event :add-grub-list [event grubs]
(let [add-grub-events (:grubs event)
add-grubs (make-add-grubs-map add-grub-events)]
(merge grubs add-grubs)))
(defmethod handle-event :complete-grub [event grubs]
(assoc-in grubs [(:id event) :completed] true))
(defmethod handle-event :uncomplete-grub [event grubs]
(assoc-in grubs [(:id event) :completed] false))
(defmethod handle-event :update-grub [event grubs]
(assoc-in grubs [(:id event) :grub] (:grub event)))
(defmethod handle-event :clear-all-grubs [event grubs]
{})
(defn update-state-and-render [remote-chan]
(let [out (chan)]
(go-loop [state default-app-state]
(let [event (<! remote-chan)
new-grubs (handle-event event (:grubs state))
new-state (assoc state :grubs new-grubs)]
(logs "event:" event)
(logs "new-state")
(logs new-state)
(view/render-body new-state)
(recur new-state)))
out))

View file

@ -1,24 +1,364 @@
(ns grub.view (ns grub.view
(:require [grub.view.dom :as dom] (:require [dommy.core :as dommy]
[grub.view.grub :as grub-view] [cljs.core.async :as a :refer [<! >! chan]]
[grub.view.recipe :as recipe-view] [om.core :as om :include-macros true]
[dommy.core :as dommy] [om.dom :as dom :include-macros true]
[cljs.core.async :as a :refer [<! >! chan]]) [sablono.core :as html :refer-macros [html]])
(:require-macros [grub.macros :refer [log logs and-let]] (:require-macros [grub.macros :refer [log logs]]
[dommy.macros :refer [deftemplate sel1 node]] [dommy.macros :refer [sel1]]
[cljs.core.async.macros :refer [go go-loop]])) [cljs.core.async.macros :refer [go]]))
(defn setup-and-get-view-events [remote-channel] (defn listen
(dom/render-body) ([el type] (listen el type nil))
(let [out (chan) ([el type f] (listen el type f (chan)))
remote (a/mult remote-channel) ([el type f out]
to-grubs (chan) (let [push-fn (fn [e] (when f (f e)) (go (>! out e)))
to-recipes (chan) unlisten #(do (dommy/unlisten! el type push-fn)
from-grubs (grub-view/handle-grubs to-grubs) (a/close! out))]
from-recipes (a/mult (recipe-view/handle-recipes to-recipes))] (dommy/listen! el type push-fn)
(a/tap remote to-grubs) {:chan out :unlisten unlisten})))
(a/tap remote to-recipes)
(a/tap from-recipes to-grubs) (def ENTER-KEYCODE 13)
(a/tap from-recipes out)
(a/pipe from-grubs out) (defn listen-once
out)) ([el type] (listen el type nil))
([el type f] (listen el type f (chan)))
([el type f out]
(let [push-fn (fn [e] (when f (f e)) (go (>! out e)))
unlisten #(do (dommy/unlisten! el type push-fn)
(a/close! out))]
(dommy/listen-once! el type push-fn)
{:chan out :unlisten unlisten})))
(defn get-away-clicks [elem]
(let [{c :chan unlisten :unlisten} (listen (sel1 :body) :click)
filtered-chan (a/filter< #(not (dommy/descendant? (.-target %) elem)) c)]
{:unlisten unlisten :chan filtered-chan}))
(defn get-clicks [elem]
(listen elem :click))
(defn get-enters [elem]
(let [{c :chan unlisten :unlisten} (listen elem :keyup)
filtered-chan (a/filter< #(= (.-which %) ENTER-KEYCODE) c)]
{:unlisten unlisten
:chan filtered-chan}))
(defn get-ctrl-enters []
(let [{c :chan unlisten :unlisten} (listen (sel1 :body) :keyup)
filtered-chan (a/filter< #(and (= (.-which %) ENTER-KEYCODE)
(.-ctrlKey %))
c)]
{:chan filtered-chan :unlisten unlisten}))
(defn get-body-enters []
(get-enters (sel1 :body)))
(def add-grub-btn
[:button.btn.btn-primary {:id "add-grub-btn" :type "button"} "Add"])
(def clear-all-btn
[:button.btn.hidden.pull-right
{:id "clear-all-btn" :type "button"}
"Clear all"])
(defn clear-grubs! []
(dommy/set-html! (sel1 :#grub-list) ""))
(defn get-grub-completed-glyph [completed]
(if completed
[:span.glyphicon.glyphicon-check]
[:span.glyphicon.glyphicon-unchecked]))
(defn make-grub-node [id grub completed]
[:li.list-group-item.grub-item
{:id id
:class (when completed "completed")}
[:span.grub-static
(get-grub-completed-glyph completed)
[:span.grub-text grub]]
[:input.grub-input {:type "text" :value grub}]])
(defn grubs-selector []
[(sel1 :#grub-list) :.grub-item])
(defn make-recipe-node
([id name grubs] (make-recipe-node id name grubs false))
([id name grubs new-recipe]
[: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}]
(when-not new-recipe
[:button.btn.btn-primary.btn-sm.recipe-add-grubs-btn {:type "button"} "Add Grubs"])]
[:div.panel-body.recipe-grubs.hidden
[:textarea.form-control.recipe-grubs-input
{:id "recipe-grubs"
:rows 3
:placeholder "2 grubs"}
grubs]
[:button.btn.btn-primary.hidden.pull-right.recipe-btn.recipe-done-btn
{:type "button"} "Done"]]]))
(def new-recipe (make-recipe-node "new-recipe" "" "" true))
(def new-recipe-done-btn
(sel1 new-recipe ".recipe-done-btn"))
(defn recipes-selector []
[(sel1 :#recipe-list) :.recipe-panel])
(defn recipe-done-btns-selector []
[(sel1 :body) :.recipe-done-btn])
(defn recipe-done-btn-selector [recipe-elem]
(sel1 recipe-elem :.recipe-done-btn))
(defn recipe-add-grubs-btns-selector []
[(sel1 :body) :.recipe-add-grubs-btn])
(defn grub-list-template [grubs]
(for [grub grubs]
(make-grub-node (:id grub) (:grub grub) (:completed grub))))
(defn render-grub-list [grubs]
(let [grub-list (sel1 :#grub-list)]
(aset grub-list "innerHTML" "")
(dommy/replace-contents! grub-list (grub-list-template grubs))))
(defn get-add-grub-text []
(dommy/value (sel1 :#add-grub-input)))
(defn clear-add-grub-text []
(dommy/set-value! (sel1 :#add-grub-input) ""))
(defn get-recipe-add-grubs-clicks []
(->> (:chan (listen (recipe-add-grubs-btns-selector) :click))
(a/map< #(dommy/closest (.-selectedTarget %) :.recipe-panel))))
(defn get-edit-recipe-input-click []
(->> (:chan (listen-once (recipes-selector) :click))
(a/filter< #(not (dommy/has-class? (.-selectedTarget %) :btn)))
(a/map< #(.-selectedTarget %))))
(defprotocol IHideable
(-hide! [this])
(-show! [this]))
(defprotocol IGrub
(-activate! [this])
(-deactivate! [this])
(-id [this])
(-grub-text [this])
(-complete! [this])
(-uncomplete! [this])
(-completed? [this])
(-set-editing! [this])
(-unset-editing! [this])
(-editing? [this])
(-update-grub! [this grub]))
(defprotocol IRecipe
(-expand! [this])
(-unexpand! [this])
(-update-recipe! [this])
(-get-name [this])
(-get-grubs-str [this])
(-get-grubs [this]))
(defprotocol IClearable
(-clear! [this]))
(extend-type js/HTMLElement
IHideable
(-hide! [this]
(dommy/add-class! this :hidden))
(-show! [this]
(dommy/remove-class! this :hidden)))
(extend-type js/HTMLElement
IGrub
(-activate! [this]
(dommy/add-class! this :grub-active))
(-deactivate! [this]
(dommy/remove-class! this :grub-active))
(-id [this]
(.-id this))
(-grub-text [this]
(.-value (sel1 this :.grub-input)))
(-complete! [this]
(dommy/add-class! this :completed)
(dommy/replace! (sel1 this ".glyphicon")
(get-grub-completed-glyph true)))
(-uncomplete! [this]
(dommy/remove-class! this :completed)
(dommy/replace! (sel1 this ".glyphicon")
(get-grub-completed-glyph false)))
(-completed? [this]
(dommy/has-class? this :completed))
(-set-editing! [this]
(-deactivate! this)
(dommy/add-class! this :edit)
(.focus (sel1 this :input)))
(-unset-editing! [this]
(dommy/remove-class! this :edit))
(-editing? [this]
(dommy/has-class? this :edit)))
(defrecord Grub [elem id grub completed]
dommy.template/PElement
(-elem [this] elem)
IGrub
(-set-editing! [this] (-set-editing! elem))
(-unset-editing! [this] (-unset-editing! elem))
(-editing? [this] (-editing? elem))
(-complete! [this] (-complete! elem))
(-uncomplete! [this] (-uncomplete! elem))
(-completed? [this] (-completed? elem))
(-set-editing! [this] (-set-editing! elem))
(-unset-editing! [this] (-unset-editing! elem))
(-editing? [this] (-editing? elem))
(-update-grub! [this grub]
(dommy/set-text! (sel1 elem ".grub-text") grub)
(dommy/set-value! (sel1 elem ".grub-input") grub)))
(defn make-new-grub [id grub completed]
(let [node (make-grub-node id grub completed)
grub (Grub. node id grub completed)
grub-list (sel1 :#grub-list)]
grub))
(defn clear-new-grub-input! []
(dommy/set-value! (sel1 :#add-grub-input) ""))
(defn focus-new-grub-input! []
(.focus (sel1 :#add-grub-input)))
(extend-type js/HTMLDivElement
IRecipe
(-expand! [this]
(dommy/remove-class! (sel1 this ".recipe-grubs") :hidden)
(dommy/remove-class! (sel1 this ".recipe-done-btn") :hidden))
(-unexpand! [this]
(dommy/add-class! (sel1 this ".recipe-grubs") :hidden)
(dommy/add-class! (sel1 this ".recipe-done-btn") :hidden))
(-get-name [this]
(dommy/value (sel1 this :#recipe-name)))
(-get-grubs-str [this]
(dommy/value (sel1 this :#recipe-grubs)))
(-get-grubs [this]
(let [split-grubs (clojure.string/split-lines (-get-grubs-str this))]
(when split-grubs (into [] split-grubs)))))
(extend-type js/HTMLDivElement
IClearable
(-clear! [this]
(dommy/set-value! (sel1 this "#recipe-name") "")
(dommy/set-value! (sel1 this "#recipe-grubs") "")))
(defrecord Recipe [elem id name grubs]
dommy.template/PElement
(-elem [this] elem)
IRecipe
(-expand! [this] (-expand! elem))
(-unexpand! [this] (-unexpand! elem))
(-clear! [this] (-clear! elem))
(-update-recipe! [this]
(dommy/set-value! (sel1 this :#recipe-name) name)
(dommy/set-text! (sel1 this :#recipe-grubs) grubs)))
(defn add-new-recipe! [id name grubs]
(let [node (make-recipe-node id name grubs)
recipe (Recipe. node id name grubs)
recipe-list (sel1 :#recipe-list)]
(dommy/append! recipe-list recipe)
recipe))
(defn grub-view [grub-state owner]
(reify
om/IRender
(render [this]
(let [{:keys [id grub completed]} grub-state]
(html
[:li.list-group-item.grub-item
{:id id
:class (when completed "completed")}
[:span.grub-static
(get-grub-completed-glyph completed)
[:span.grub-text grub]]
[:input.grub-input {:type "text" :value grub}]])))))
(defn get-grub-ingredient [grub]
(let [text (clojure.string/lower-case (:grub grub))
match (re-find #"[a-z]{3}.*$" text)]
match))
(defn sort-grubs [grubs]
(sort-by (juxt :completed get-grub-ingredient) (vals grubs)))
(defn app-view [state owner]
(reify
om/IRender
(render [this]
(let [sorted-grubs (sort-grubs (:grubs state))]
(logs "render app-view state:" state)
(logs "render app-view owner:" owner)
(logs "render grubs:" sorted-grubs)
(html
[:div.container
[:div.row
[:div.col-sm-6.leftmost-column
[:h3 "Grub List"]
[:div.input-group.add-grub-input-form
[:span.input-group-btn
[:input.form-control#add-grub-input {:type "text" :placeholder "2 grubs"}]]
[:button.btn.btn-primary {:id "add-grub-btn" :type "button"} "Add"]]
[:ul#grub-list.list-group
(for [grub (vals (:grubs state))]
(om/build grub-view grub))]
[:button.btn.hidden.pull-right
{:id "clear-all-btn" :type "button"}
"Clear all"]]
[:div.col-sm-6
[:h3.recipes-title "Recipes"]
[:div.panel.panel-default.recipe-panel
[:div.panel-heading.recipe-header
[:input.form-control.recipe-header-input
{:id "recipe-name"
:type "text"
:placeholder "Grub pie"}]
[:button.btn.btn-primary.btn-sm.recipe-add-grubs-btn {:type "button"} "Add Grubs"]]
[:div.panel-body.recipe-grubs.hidden
[:textarea.form-control.recipe-grubs-input
{:id "recipe-grubs"
:rows 3
:placeholder "2 grubs"}]
[:button.btn.btn-primary.hidden.pull-right.recipe-btn.recipe-done-btn
{:type "button"} "Done"]]]
[:ul#recipe-list.list-group.recipe-list]]]])))))
(defn render-body [state]
(logs state)
(om/root app-view state {:target (.getElementById js/document "container")}))

View file

@ -4,8 +4,8 @@
[om.core :as om :include-macros true] [om.core :as om :include-macros true]
[om.dom :as dom :include-macros true] [om.dom :as dom :include-macros true]
[sablono.core :as html :refer-macros [html]]) [sablono.core :as html :refer-macros [html]])
(:require-macros [grub.macros :refer [log logs go-loop]] (:require-macros [grub.macros :refer [log logs]]
[dommy.macros :refer [deftemplate sel1 node]] [dommy.macros :refer [sel1]]
[cljs.core.async.macros :refer [go]])) [cljs.core.async.macros :refer [go]]))
(defn listen (defn listen
@ -54,33 +54,30 @@
(defn get-body-enters [] (defn get-body-enters []
(get-enters (sel1 :body))) (get-enters (sel1 :body)))
(def add-grub-text
[:input.form-control {:id "add-grub-input" :type "text" :placeholder "2 grubs"}])
(def add-grub-btn (def add-grub-btn
[:button.btn.btn-primary {:id "add-grub-btn" :type "button"} "Add"]) [:button.btn.btn-primary {:id "add-grub-btn" :type "button"} "Add"])
(def clear-all-btn (def clear-all-btn
(node [:button.btn.hidden.pull-right [:button.btn.hidden.pull-right
{:id "clear-all-btn" :type "button"} {:id "clear-all-btn" :type "button"}
"Clear all"])) "Clear all"])
(defn clear-grubs! [] (defn clear-grubs! []
(dommy/set-html! (sel1 :#grub-list) "")) (dommy/set-html! (sel1 :#grub-list) ""))
(defn get-grub-completed-glyph [completed] (defn get-grub-completed-glyph [completed]
(node (if completed (if completed
[:span.glyphicon.glyphicon-check] [:span.glyphicon.glyphicon-check]
[:span.glyphicon.glyphicon-unchecked]))) [:span.glyphicon.glyphicon-unchecked]))
(defn make-grub-node [id grub completed] (defn make-grub-node [id grub completed]
(node [:li.list-group-item.grub-item [:li.list-group-item.grub-item
{:id id {:id id
:class (when completed "completed")} :class (when completed "completed")}
[:span.grub-static [:span.grub-static
(get-grub-completed-glyph completed) (get-grub-completed-glyph completed)
[:span.grub-text grub]] [:span.grub-text grub]]
[:input.grub-input {:type "text" :value grub}]])) [:input.grub-input {:type "text" :value grub}]])
(defn grubs-selector [] (defn grubs-selector []
[(sel1 :#grub-list) :.grub-item]) [(sel1 :#grub-list) :.grub-item])
@ -88,24 +85,24 @@
(defn make-recipe-node (defn make-recipe-node
([id name grubs] (make-recipe-node id name grubs false)) ([id name grubs] (make-recipe-node id name grubs false))
([id name grubs new-recipe] ([id name grubs new-recipe]
(node [:div.panel.panel-default.recipe-panel [:div.panel.panel-default.recipe-panel
{:id id} {:id id}
[:div.panel-heading.recipe-header [:div.panel-heading.recipe-header
[:input.form-control.recipe-header-input [:input.form-control.recipe-header-input
{:id "recipe-name" {:id "recipe-name"
:type "text" :type "text"
:placeholder "Grub pie" :placeholder "Grub pie"
:value name}] :value name}]
(when-not new-recipe (when-not new-recipe
[:button.btn.btn-primary.btn-sm.recipe-add-grubs-btn {:type "button"} "Add Grubs"])] [:button.btn.btn-primary.btn-sm.recipe-add-grubs-btn {:type "button"} "Add Grubs"])]
[:div.panel-body.recipe-grubs.hidden [:div.panel-body.recipe-grubs.hidden
[:textarea.form-control.recipe-grubs-input [:textarea.form-control.recipe-grubs-input
{:id "recipe-grubs" {:id "recipe-grubs"
:rows 3 :rows 3
:placeholder "2 grubs"} :placeholder "2 grubs"}
grubs] grubs]
[:button.btn.btn-primary.hidden.pull-right.recipe-btn.recipe-done-btn [:button.btn.btn-primary.hidden.pull-right.recipe-btn.recipe-done-btn
{:type "button"} "Done"]]]))) {:type "button"} "Done"]]]))
(def new-recipe (make-recipe-node "new-recipe" "" "" true)) (def new-recipe (make-recipe-node "new-recipe" "" "" true))
@ -124,36 +121,9 @@
(defn recipe-add-grubs-btns-selector [] (defn recipe-add-grubs-btns-selector []
[(sel1 :body) :.recipe-add-grubs-btn]) [(sel1 :body) :.recipe-add-grubs-btn])
(deftemplate main-template [] (defn grub-list-template [grubs]
[:div.container (for [grub grubs]
[:div.row (make-grub-node (:id grub) (:grub grub) (:completed grub))))
[:div.col-sm-6.leftmost-column
[:h3 "Grub List"]
[:div#addGrubInput]
[:ul#grub-list.list-group]
clear-all-btn]
[:div.col-sm-6
[:h3.recipes-title "Recipes"]
new-recipe
[:ul#recipe-list.list-group.recipe-list]]]])
(deftemplate grub-list-template [grubs]
(node (for [grub grubs]
(make-grub-node (:id grub) (:grub grub) (:completed grub)))))
(defn add-grub-input [data]
(om/component
(html [:div.input-group
add-grub-text
[:span.input-group-btn
add-grub-btn]])))
(defn render-om []
(om/root add-grub-input {} {:target (.getElementById js/document "addGrubInput")}))
(defn render-body []
(dommy/prepend! (sel1 :body) (main-template))
(render-om))
(defn render-grub-list [grubs] (defn render-grub-list [grubs]
(let [grub-list (sel1 :#grub-list)] (let [grub-list (sel1 :#grub-list)]
@ -325,4 +295,40 @@
recipe-list (sel1 :#recipe-list)] recipe-list (sel1 :#recipe-list)]
(dommy/append! recipe-list recipe) (dommy/append! recipe-list recipe)
recipe)) recipe))
(defn grub-view []
(om/component
(html
[:div.container
[:div.row
[:div.col-sm-6.leftmost-column
[:h3 "Grub List"]
[:div.input-group.add-grub-input-form
[:span.input-group-btn
[:input.form-control#add-grub-input {:type "text" :placeholder "2 grubs"}]]
[:button.btn.btn-primary {:id "add-grub-btn" :type "button"} "Add"]]
[:ul#grub-list.list-group]
[:button.btn.hidden.pull-right
{:id "clear-all-btn" :type "button"}
"Clear all"]]
[:div.col-sm-6
[:h3.recipes-title "Recipes"]
[:div.panel.panel-default.recipe-panel
[:div.panel-heading.recipe-header
[:input.form-control.recipe-header-input
{:id "recipe-name"
:type "text"
:placeholder "Grub pie"}]
[:button.btn.btn-primary.btn-sm.recipe-add-grubs-btn {:type "button"} "Add Grubs"]]
[:div.panel-body.recipe-grubs.hidden
[:textarea.form-control.recipe-grubs-input
{:id "recipe-grubs"
:rows 3
:placeholder "2 grubs"}]
[:button.btn.btn-primary.hidden.pull-right.recipe-btn.recipe-done-btn
{:type "button"} "Done"]]]
[:ul#recipe-list.list-group.recipe-list]]]])))
(defn render-body [state]
(logs state)
(om/root grub-view state {:target (.getElementById js/document "container")}))

View file

@ -37,7 +37,7 @@
(a/map< parse-complete-event))) (a/map< parse-complete-event)))
(defn get-clear-all-events [] (defn get-clear-all-events []
(->> (:chan (dom/listen dom/clear-all-btn :click)) (->> (:chan (dom/listen (sel1 :#clear-all-btn) :click))
(a/map< (fn [e] {:event :clear-all-grubs})))) (a/map< (fn [e] {:event :clear-all-grubs}))))
(defn get-grub-mousedown-events [] (defn get-grub-mousedown-events []
@ -127,68 +127,3 @@
(defn sort-and-render-grub-list! [grubs] (defn sort-and-render-grub-list! [grubs]
(let [sorted-grubs (sort-by (juxt :completed get-grub-ingredient) (vals grubs))] (let [sorted-grubs (sort-by (juxt :completed get-grub-ingredient) (vals grubs))]
(dom/render-grub-list sorted-grubs))) (dom/render-grub-list sorted-grubs)))
(defmulti handle-event (fn [event grubs] (:event event))
:default :unknown-event)
(defmethod handle-event :unknown-event [event grubs]
;(logs "Cannot handle unknown event:" event)
grubs)
(defmethod handle-event :add-grub [event grubs]
(let [grub (dom/make-new-grub (:id event) (:grub event) (:completed event))
new-grubs (assoc grubs (:id grub) grub)]
(dom/-show! dom/clear-all-btn)
(sort-and-render-grub-list! new-grubs)
(dom/clear-new-grub-input!)
(dom/focus-new-grub-input!)
new-grubs))
(defn assoc-new-grub [current new]
(assoc current (:id new)
(dom/make-new-grub (:id new) (:grub new) (:completed new))))
(defn make-add-grubs-map [grub-events]
(reduce assoc-new-grub {} grub-events))
(defmethod handle-event :add-grub-list [event grubs]
(let [add-grub-events (:grubs event)
add-grubs (make-add-grubs-map add-grub-events)
new-grubs (merge grubs add-grubs)]
(dom/-show! dom/clear-all-btn)
(sort-and-render-grub-list! new-grubs)
new-grubs))
(defmethod handle-event :complete-grub [event grubs]
(let [grub (get grubs (:id event))
new-grubs (assoc-in grubs [(:id event) :completed] true)]
(sort-and-render-grub-list! new-grubs)
new-grubs))
(defmethod handle-event :uncomplete-grub [event grubs]
(let [new-grubs (assoc-in grubs [(:id event) :completed] false)]
(sort-and-render-grub-list! new-grubs)
new-grubs))
(defmethod handle-event :update-grub [event grubs]
(let [new-grubs (assoc-in grubs [(:id event) :grub] (:grub event))]
(sort-and-render-grub-list! new-grubs)
new-grubs))
(defmethod handle-event :clear-all-grubs [event grubs]
(dom/-hide! dom/clear-all-btn)
(dom/clear-grubs!)
{})
(defn handle-grubs [remote-events]
(let [out (chan)
local-events [(get-create-events)
(get-complete-events)
(get-clear-all-events)
(get-update-events)]]
(go-loop [grubs {}]
(let [[event c] (a/alts! (conj local-events remote-events))]
(when-not (= c remote-events)
(>! out event))
(recur (handle-event event grubs))))
out))