mirror of
https://github.com/penpot/penpot.git
synced 2025-01-23 23:18:48 -05:00
Implement shapes moving using mouse.
This commit is contained in:
parent
17eafb8563
commit
a5dc634e35
4 changed files with 126 additions and 27 deletions
|
@ -63,6 +63,7 @@
|
||||||
:project project
|
:project project
|
||||||
:created (time/now :unix)
|
:created (time/now :unix)
|
||||||
:shapes []
|
:shapes []
|
||||||
|
:shapes-by-id {}
|
||||||
:name name
|
:name name
|
||||||
:width width
|
:width width
|
||||||
:height height}]
|
:height height}]
|
||||||
|
|
|
@ -74,11 +74,13 @@
|
||||||
(reify
|
(reify
|
||||||
rs/UpdateEvent
|
rs/UpdateEvent
|
||||||
(-apply-update [_ state]
|
(-apply-update [_ state]
|
||||||
(println "add-shape")
|
(let [id (random-uuid)
|
||||||
(if-let [pageid (get-in state [:workspace :page])]
|
pageid (get-in state [:workspace :page])
|
||||||
(update-in state [:pages-by-id pageid :shapes] conj
|
_ (assert pageid)
|
||||||
(merge shape props))
|
shape (merge shape props {:id id})]
|
||||||
state))
|
(as-> state $
|
||||||
|
(update-in $ [:pages-by-id pageid :shapes] conj id)
|
||||||
|
(update-in $ [:pages-by-id pageid :shapes-by-id] assoc id shape))))
|
||||||
|
|
||||||
IPrintWithWriter
|
IPrintWithWriter
|
||||||
(-pr-writer [mv writer _]
|
(-pr-writer [mv writer _]
|
||||||
|
@ -101,3 +103,29 @@
|
||||||
(-pr-writer [mv writer _]
|
(-pr-writer [mv writer _]
|
||||||
(-write writer "#<event:u.d.w/initialize>"))))
|
(-write writer "#<event:u.d.w/initialize>"))))
|
||||||
|
|
||||||
|
(defn apply-delta
|
||||||
|
"Mark a shape selected for drawing in the canvas."
|
||||||
|
[shapeid [dx dy :as delta]]
|
||||||
|
(reify
|
||||||
|
rs/UpdateEvent
|
||||||
|
(-apply-update [_ state]
|
||||||
|
;; (println "apply-delta" shapeid delta)
|
||||||
|
(let [pageid (get-in state [:workspace :page])
|
||||||
|
_ (assert pageid)
|
||||||
|
shape (get-in state [:pages-by-id pageid :shapes-by-id shapeid])]
|
||||||
|
(update-in state [:pages-by-id pageid :shapes-by-id shapeid] merge
|
||||||
|
{:x (+ (:x shape) dx)
|
||||||
|
:y (+ (:y shape) dy)})))))
|
||||||
|
|
||||||
|
|
||||||
|
;; (defn apply-delta'
|
||||||
|
;; "Mark a shape selected for drawing in the canvas."
|
||||||
|
;; [shapeid [dx dy :as delta]]
|
||||||
|
;; (reify
|
||||||
|
;; rs/UpdateEvent
|
||||||
|
;; (-apply-update [_ state]
|
||||||
|
;; ;; (println "apply-delta'" shapeid delta)
|
||||||
|
;; (let [pageid (get-in state [:workspace :page])
|
||||||
|
;; shape (get-in state [:pages-by-id pageid :shapes-by-id shapeid])]
|
||||||
|
;; (update-in state [:pages-by-id pageid :shapes-by-id shapeid] merge
|
||||||
|
;; {:x dx :y dy})))))
|
||||||
|
|
|
@ -4,6 +4,7 @@
|
||||||
[uxbox.rstore :as rs]
|
[uxbox.rstore :as rs]
|
||||||
[uxbox.state :as s]
|
[uxbox.state :as s]
|
||||||
[uxbox.data.projects :as dp]
|
[uxbox.data.projects :as dp]
|
||||||
|
[uxbox.data.workspace :as dw]
|
||||||
[uxbox.util.lens :as ul]
|
[uxbox.util.lens :as ul]
|
||||||
[uxbox.ui.util :as util]
|
[uxbox.ui.util :as util]
|
||||||
[goog.events :as events])
|
[goog.events :as events])
|
||||||
|
@ -54,20 +55,57 @@
|
||||||
;; Mouse Position Stream
|
;; Mouse Position Stream
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
|
|
||||||
;; (defn- coords-delta
|
(def immediate-scheduler js/Rx.Scheduler.immediate)
|
||||||
;; [[old new]]
|
(def current-thread-scheduler js/Rx.Scheduler.currentThread)
|
||||||
;; (let [[oldx oldy] old
|
|
||||||
;; [newx newy] new]
|
|
||||||
;; [(* 2 (- newx oldx))
|
|
||||||
;; (* 2 (- newy oldy))]))
|
|
||||||
|
|
||||||
;; (def ^{:doc "A stream of mouse coordinate deltas as `[dx dy]` vectors."}
|
(defn observe-on
|
||||||
;; delta
|
[scheduler ob]
|
||||||
;; (s/map coords-delta (s/partition 2 client-position)))
|
(.observeOn ob scheduler))
|
||||||
|
|
||||||
|
(defn subscribe-on
|
||||||
|
[scheduler ob]
|
||||||
|
(.subscribeOn ob scheduler))
|
||||||
|
|
||||||
|
;; (defn window
|
||||||
|
;; [n ob]
|
||||||
|
;; (.windowWithCount ob n))
|
||||||
|
|
||||||
|
(defonce selected-shape-b (rx/bus))
|
||||||
|
|
||||||
|
(defonce mouse-b (rx/bus))
|
||||||
|
(defonce mouse-s (rx/dedupe mouse-b))
|
||||||
|
|
||||||
|
;; Deltas
|
||||||
|
|
||||||
|
(defn- coords-delta
|
||||||
|
[[old new]]
|
||||||
|
(let [[oldx oldy] old
|
||||||
|
[newx newy] new]
|
||||||
|
[(- newx oldx)
|
||||||
|
(- newy oldy)]))
|
||||||
|
|
||||||
|
(defonce mouse-delta-s
|
||||||
|
(->> mouse-s
|
||||||
|
(rx/sample 10)
|
||||||
|
(rx/buffer 2 1)
|
||||||
|
(rx/map coords-delta)))
|
||||||
|
|
||||||
|
(defonce _subscription_
|
||||||
|
(as-> (rx/with-latest-from vector selected-shape-b mouse-delta-s) $
|
||||||
|
(rx/filter #(not= :nothing (second %)) $)
|
||||||
|
;; (observe-on current-thread-scheduler $)
|
||||||
|
(rx/on-value $ (fn [[delta shape]]
|
||||||
|
(rs/emit! (dw/apply-delta (:id shape) delta))))))
|
||||||
|
|
||||||
|
;; (rx/on-value mouse-delta-s
|
||||||
|
;; (fn [val]
|
||||||
|
;; (println "delta" val))))
|
||||||
|
|
||||||
|
;; Materialized views
|
||||||
|
|
||||||
|
(defonce mouse-position (rx/to-atom (rx/sample 50 mouse-s)))
|
||||||
|
;; (defonce mouse-position2 (rx/to-atom mouse-s))
|
||||||
|
|
||||||
(defonce mouse-bus (rx/bus))
|
|
||||||
(defonce mouse-s (rx/dedupe mouse-bus))
|
|
||||||
(defonce mouse-position (rx/to-atom (rx/throttle 50 mouse-s)))
|
|
||||||
|
|
||||||
(defn- mouse-mixin-did-mount
|
(defn- mouse-mixin-did-mount
|
||||||
[own]
|
[own]
|
||||||
|
@ -78,7 +116,7 @@
|
||||||
offset-y (.-top brect)
|
offset-y (.-top brect)
|
||||||
x (.-clientX event)
|
x (.-clientX event)
|
||||||
y (.-clientY event)]
|
y (.-clientY event)]
|
||||||
(rx/push! mouse-bus [(- x offset-x)
|
(rx/push! mouse-b [(- x offset-x)
|
||||||
(- y offset-y)])))]
|
(- y offset-y)])))]
|
||||||
(let [key (events/listen js/document EventType.MOUSEMOVE on-mousemove)]
|
(let [key (events/listen js/document EventType.MOUSEMOVE on-mousemove)]
|
||||||
(js/console.log "mouse-mixin-did-mount" key)
|
(js/console.log "mouse-mixin-did-mount" key)
|
||||||
|
|
|
@ -1,14 +1,16 @@
|
||||||
(ns uxbox.ui.workspace.canvas
|
(ns uxbox.ui.workspace.canvas
|
||||||
(:require [sablono.core :as html :refer-macros [html]]
|
(:require [sablono.core :as html :refer-macros [html]]
|
||||||
[rum.core :as rum]
|
[rum.core :as rum]
|
||||||
|
[beicon.core :as rx]
|
||||||
[uxbox.router :as r]
|
[uxbox.router :as r]
|
||||||
[uxbox.rstore :as rs]
|
[uxbox.rstore :as rs]
|
||||||
[uxbox.state :as s]
|
[uxbox.state :as s]
|
||||||
[uxbox.shapes :as shapes]
|
[uxbox.shapes :as shapes]
|
||||||
[uxbox.library.icons :as _icons]
|
[uxbox.library.icons :as _icons]
|
||||||
[uxbox.ui.mixins :as mx]
|
|
||||||
[uxbox.ui.util :as util]
|
|
||||||
[uxbox.data.projects :as dp]
|
[uxbox.data.projects :as dp]
|
||||||
|
[uxbox.ui.mixins :as mx]
|
||||||
|
[uxbox.ui.dom :as dom]
|
||||||
|
[uxbox.ui.util :as util]
|
||||||
[uxbox.ui.workspace.base :as wb]
|
[uxbox.ui.workspace.base :as wb]
|
||||||
[uxbox.ui.workspace.rules :as wr]
|
[uxbox.ui.workspace.rules :as wr]
|
||||||
[uxbox.ui.workspace.toolboxes :as toolboxes]))
|
[uxbox.ui.workspace.toolboxes :as toolboxes]))
|
||||||
|
@ -47,9 +49,33 @@
|
||||||
:stroke "gray"})
|
:stroke "gray"})
|
||||||
|
|
||||||
(defn- shape-render
|
(defn- shape-render
|
||||||
[own shape]
|
[own {:keys [x y] :as shape}]
|
||||||
(let [local (:rum/local own)]
|
(let [local (:rum/local own)]
|
||||||
(shapes/render shape)))
|
(html
|
||||||
|
[:g {
|
||||||
|
:on-mouse-down
|
||||||
|
(fn [event]
|
||||||
|
(println "mouse-down")
|
||||||
|
(rx/push! wb/selected-shape-b shape))
|
||||||
|
|
||||||
|
;; :on-mouse-move
|
||||||
|
;; (fn [event]
|
||||||
|
;; (when (:drop @local)
|
||||||
|
;; (println "mouse-move" @wb/mouse-position2)
|
||||||
|
;; (let [target (.-currentTarget event)
|
||||||
|
;; [nx ny] @wb/mouse-position2
|
||||||
|
;; svg (aget (.-childNodes target) 0)]
|
||||||
|
;; (.setAttribute svg "x" nx)
|
||||||
|
;; (.setAttribute svg "y" ny)))
|
||||||
|
;; false)
|
||||||
|
|
||||||
|
:on-mouse-up
|
||||||
|
(fn [event]
|
||||||
|
(println "mouse-up")
|
||||||
|
(rx/push! wb/selected-shape-b :nothing)
|
||||||
|
(dom/stop-propagation event))
|
||||||
|
}
|
||||||
|
(shapes/render shape)])))
|
||||||
|
|
||||||
;; (defn- shape-render
|
;; (defn- shape-render
|
||||||
;; [own shape]
|
;; [own shape]
|
||||||
|
@ -78,7 +104,7 @@
|
||||||
(util/component
|
(util/component
|
||||||
{:render shape-render
|
{:render shape-render
|
||||||
:name "shape"
|
:name "shape"
|
||||||
:mixins [mx/static]}))
|
:mixins [mx/static (mx/local {})]}))
|
||||||
|
|
||||||
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
|
||||||
;; Canvas
|
;; Canvas
|
||||||
|
@ -106,13 +132,19 @@
|
||||||
:ref "canvas"
|
:ref "canvas"
|
||||||
:width page-width
|
:width page-width
|
||||||
:height page-height
|
:height page-height
|
||||||
|
:on-mouse-up
|
||||||
|
(fn [event]
|
||||||
|
(rx/push! wb/selected-shape-b :nothing)
|
||||||
|
(dom/stop-propagation event))
|
||||||
|
|
||||||
;; :on-mouse-down cs/on-mouse-down
|
;; :on-mouse-down cs/on-mouse-down
|
||||||
;; :on-mouse-up cs/on-mouse-up
|
;; :on-mouse-up cs/on-mouse-up
|
||||||
}
|
}
|
||||||
(background)
|
(background)
|
||||||
[:svg.page-layout
|
[:svg.page-layout {}
|
||||||
(for [item (:shapes page)]
|
(for [shapeid (:shapes page)
|
||||||
(shape item))]
|
:let [item (get-in page [:shapes-by-id shapeid])]]
|
||||||
|
(rum/with-key (shape item) (str shapeid)))]
|
||||||
#_(apply vector :svg#page-layout (map shapes/shape->svg raw-shapes))
|
#_(apply vector :svg#page-layout (map shapes/shape->svg raw-shapes))
|
||||||
#_(when-let [shape (rum/react drawing)]
|
#_(when-let [shape (rum/react drawing)]
|
||||||
(shapes/shape->drawing-svg shape))
|
(shapes/shape->drawing-svg shape))
|
||||||
|
|
Loading…
Add table
Reference in a new issue