mirror of
https://github.com/penpot/penpot.git
synced 2025-01-25 07:58:49 -05:00
125 lines
3.9 KiB
Clojure
125 lines
3.9 KiB
Clojure
(ns uxbox.data.projects
|
|
(:require [uxbox.rstore :as rs]
|
|
[uxbox.router :as r]
|
|
[uxbox.state :as st]
|
|
[uxbox.schema :as sc]
|
|
[bouncer.validators :as v]))
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;; Schemas
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(def ^:static +project-schema+
|
|
{:name [v/required v/string]
|
|
:width [v/required v/integer]
|
|
:height [v/required v/integer]
|
|
:layout [v/required sc/keyword]})
|
|
|
|
(def ^:static +page-schema+
|
|
{:name [v/required v/string]
|
|
:width [v/required v/integer]
|
|
:height [v/required v/integer]
|
|
:project [v/required sc/uuid]})
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;; Helpers
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(defn assoc-project
|
|
"A reduce function for assoc the project
|
|
to the state map."
|
|
[state proj]
|
|
(let [uuid (:id proj)]
|
|
(update-in state [:projects-by-id] assoc uuid proj)))
|
|
|
|
(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)))
|
|
|
|
;; (defn project-pages
|
|
;; "Get a ordered list of pages that
|
|
;; belongs to a specified project."
|
|
;; [state projectid]
|
|
;; (let [xf (comp
|
|
;; (filter #(= projectid (:project %)))
|
|
;; (map #(get-in state [:pages-by-id %])))]
|
|
;; (into [] xf (:pages state))))
|
|
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
;; Events
|
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
|
|
|
(defn create-page
|
|
[{:keys [name width height project] :as data}]
|
|
(sc/validate! +page-schema+ data)
|
|
(reify
|
|
rs/UpdateEvent
|
|
(-apply-update [_ state]
|
|
(let [page {:id (random-uuid)
|
|
:project project
|
|
:name name
|
|
:width width
|
|
:height height}]
|
|
(assoc-page state page)))
|
|
|
|
IPrintWithWriter
|
|
(-pr-writer [mv writer _]
|
|
(-write writer "#<event:u.s.p/create-page>"))))
|
|
|
|
(defn create-project
|
|
[{:keys [name width height layout] :as data}]
|
|
(sc/validate! +project-schema+ data)
|
|
(let [uuid (random-uuid)]
|
|
(reify
|
|
rs/UpdateEvent
|
|
(-apply-update [_ state]
|
|
(let [proj {:id uuid
|
|
:name name
|
|
:width width
|
|
:height height
|
|
:layout layout}]
|
|
(assoc-project state proj)))
|
|
|
|
rs/EffectEvent
|
|
(-apply-effect [_ state]
|
|
(rs/emit! (create-page {:name "Page 1"
|
|
:width width
|
|
:height height
|
|
:project uuid})))
|
|
|
|
IPrintWithWriter
|
|
(-pr-writer [mv writer _]
|
|
(-write writer "#<event:u.s.p/create-project>")))))
|
|
|
|
(defn initialize-workspace
|
|
[projectid pageid]
|
|
(reify
|
|
rs/UpdateEvent
|
|
(-apply-update [_ state]
|
|
(assoc state :workspace {:project projectid :page pageid}))
|
|
|
|
IPrintWithWriter
|
|
(-pr-writer [mv writer _]
|
|
(-write writer "#<event:u.s.p/initialize-workspace>"))))
|
|
|
|
(defn go-to-project
|
|
"A shortcut event that redirects the user to the
|
|
first page of the project."
|
|
([projectid]
|
|
(go-to-project projectid nil))
|
|
([projectid pageid]
|
|
(reify
|
|
rs/EffectEvent
|
|
(-apply-effect [_ state]
|
|
(if pageid
|
|
(rs/emit! (r/navigate :main/page {:project-uuid projectid
|
|
:page-uuid pageid}))
|
|
(let [pages (get-in state [:projects-by-id projectid :pages])]
|
|
(rs/emit! (r/navigate :main/page {:project-uuid projectid
|
|
:page-uuid (first pages)})))))
|
|
IPrintWithWriter
|
|
(-pr-writer [mv writer _]
|
|
(-write writer "#<event:u.s.p/go-to-project")))))
|