2015-12-20 13:19:07 -05:00
|
|
|
(ns uxbox.data.dashboard
|
|
|
|
(:require [uxbox.rstore :as rs]
|
|
|
|
[uxbox.router :as r]
|
|
|
|
[uxbox.state :as st]
|
|
|
|
[uxbox.schema :as sc]
|
|
|
|
[uxbox.time :as time]
|
|
|
|
[bouncer.validators :as v]))
|
|
|
|
|
2015-12-24 12:32:26 -05:00
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Schemas
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
(def ^:static +color-replace-schema+
|
|
|
|
{:id [v/required sc/uuid]
|
|
|
|
:from [sc/color]
|
|
|
|
:to [v/required sc/color]})
|
|
|
|
|
|
|
|
(def ^:static +remove-color-schema+
|
|
|
|
{:id [v/required sc/uuid]
|
|
|
|
:color [v/required sc/color]})
|
|
|
|
|
2015-12-20 13:19:07 -05:00
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Helpers
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
|
|
|
(defn assoc-page
|
|
|
|
"A reduce function for assoc the page
|
|
|
|
to the state map."
|
|
|
|
[state page]
|
|
|
|
(let [uuid (:id page)]
|
|
|
|
(update-in state [:pages-by-id] assoc uuid page)))
|
|
|
|
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
;; Events
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
|
2015-12-21 12:16:39 -05:00
|
|
|
(defn merge-if-not-exists
|
|
|
|
[map & maps]
|
|
|
|
(let [result (transient map)]
|
|
|
|
(loop [maps maps]
|
|
|
|
(if-let [nextval (first maps)]
|
|
|
|
(do
|
|
|
|
(run! (fn [[key value]]
|
|
|
|
(when-not (contains? result key)
|
|
|
|
(assoc! result key value)))
|
|
|
|
nextval)
|
|
|
|
(recur (rest maps)))
|
|
|
|
(persistent! result)))))
|
|
|
|
|
2015-12-20 13:19:07 -05:00
|
|
|
(defn initialize
|
|
|
|
[section]
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
2015-12-21 12:16:39 -05:00
|
|
|
(as-> state $
|
|
|
|
(assoc-in $ [:dashboard :section] section)
|
|
|
|
(update $ :dashboard merge-if-not-exists
|
|
|
|
{:collection-type :builtin
|
|
|
|
:collection-id 1})))
|
|
|
|
|
2015-12-20 13:19:07 -05:00
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/initialize>"))))
|
|
|
|
|
|
|
|
(defn set-collection-type
|
|
|
|
[type]
|
|
|
|
{:pre [(contains? #{:builtin :own} type)]}
|
2015-12-21 12:16:39 -05:00
|
|
|
(letfn [(select-first [state]
|
|
|
|
(if (= type :builtin)
|
|
|
|
(assoc-in state [:dashboard :collection-id] 1)
|
2015-12-24 12:32:56 -05:00
|
|
|
(let [colls (sort-by :id (vals (:colors-by-id state)))]
|
|
|
|
(assoc-in state [:dashboard :collection-id] (:id (first colls))))))]
|
2015-12-21 12:16:39 -05:00
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(as-> state $
|
|
|
|
(assoc-in $ [:dashboard :collection-type] type)
|
|
|
|
(select-first $)))
|
2015-12-20 13:19:07 -05:00
|
|
|
|
2015-12-21 12:16:39 -05:00
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/set-collection-type>")))))
|
2015-12-20 13:19:07 -05:00
|
|
|
|
|
|
|
(defn set-collection
|
|
|
|
[id]
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(assoc-in state [:dashboard :collection-id] id))
|
|
|
|
|
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/set-collection>"))))
|
2015-12-24 12:34:46 -05:00
|
|
|
|
|
|
|
(defn mk-color-collection
|
|
|
|
[]
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(let [id (random-uuid)
|
|
|
|
coll {:name "Unnamed collection"
|
|
|
|
:id id :colors #{}}]
|
|
|
|
(-> state
|
|
|
|
(assoc-in [:colors-by-id id] coll)
|
|
|
|
(assoc-in [:dashboard :collection-id] id)
|
|
|
|
(assoc-in [:dashboard :collection-type] :own))))
|
|
|
|
|
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/mk-color-collection>"))))
|
|
|
|
|
|
|
|
(defn rename-color-collection
|
|
|
|
[id name]
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(assoc-in state [:colors-by-id id :name] name))
|
|
|
|
|
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/rename-color-collection>"))))
|
|
|
|
|
|
|
|
(defn delete-color-collection
|
|
|
|
[id]
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(let [state (update state :colors-by-id dissoc id)
|
|
|
|
colls (sort-by :id (vals (:colors-by-id state)))]
|
|
|
|
(assoc-in state [:dashboard :collection-id] (:id (first colls)))))
|
|
|
|
|
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/rename-color-collection>"))))
|
|
|
|
|
|
|
|
(defn replace-color
|
|
|
|
"Add or replace color in a collection."
|
|
|
|
[{:keys [id from to] :as params}]
|
|
|
|
(sc/validate! +color-replace-schema+ params)
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(if-let [colors (get-in state [:colors-by-id id :colors])]
|
|
|
|
(as-> colors $
|
|
|
|
(disj $ from)
|
|
|
|
(conj $ to)
|
|
|
|
(assoc-in state [:colors-by-id id :colors] $))
|
|
|
|
state))
|
|
|
|
|
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/replace-color>"))))
|
|
|
|
|
|
|
|
(defn remove-color
|
|
|
|
"Remove color in a collection."
|
|
|
|
[{:keys [id color] :as params}]
|
|
|
|
(sc/validate! +remove-color-schema+ params)
|
|
|
|
(reify
|
|
|
|
rs/UpdateEvent
|
|
|
|
(-apply-update [_ state]
|
|
|
|
(if-let [colors (get-in state [:colors-by-id id :colors])]
|
|
|
|
(as-> colors $
|
|
|
|
(disj $ color)
|
|
|
|
(assoc-in state [:colors-by-id id :colors] $))
|
|
|
|
state))
|
|
|
|
|
|
|
|
IPrintWithWriter
|
|
|
|
(-pr-writer [mv writer _]
|
|
|
|
(-write writer "#<event:u.d.d/remove-color>"))))
|
|
|
|
|