mirror of
https://github.com/penpot/penpot.git
synced 2025-01-25 07:58:49 -05:00
355 lines
12 KiB
Clojure
355 lines
12 KiB
Clojure
;; This Source Code Form is subject to the terms of the Mozilla Public
|
|
;; License, v. 2.0. If a copy of the MPL was not distributed with this
|
|
;; file, You can obtain one at http://mozilla.org/MPL/2.0/.
|
|
;;
|
|
;; Copyright (c) UXBOX Labs SL
|
|
|
|
(ns debug
|
|
(:require
|
|
[app.common.data :as d]
|
|
[app.common.logging :as l]
|
|
[app.common.pages.helpers :as cph]
|
|
[app.common.transit :as t]
|
|
[app.common.uuid :as uuid]
|
|
[app.main.data.dashboard.shortcuts]
|
|
[app.main.data.viewer.shortcuts]
|
|
[app.main.data.workspace :as dw]
|
|
[app.main.data.workspace.changes :as dwc]
|
|
[app.main.data.workspace.path.shortcuts]
|
|
[app.main.data.workspace.shortcuts]
|
|
[app.main.store :as st]
|
|
[app.util.dom :as dom]
|
|
[app.util.object :as obj]
|
|
[app.util.timers :as timers]
|
|
[beicon.core :as rx]
|
|
[cljs.pprint :refer [pprint]]
|
|
[cuerdas.core :as str]
|
|
[potok.core :as ptk]
|
|
[promesa.core :as p]))
|
|
|
|
(defn ^:export set-logging
|
|
([level]
|
|
(l/set-level! :app (keyword level)))
|
|
([ns level]
|
|
(l/set-level! (keyword ns) (keyword level))))
|
|
|
|
(def debug-options
|
|
#{;; Displays the bounding box for the shapes
|
|
:bounding-boxes
|
|
|
|
;; Displays an overlay over the groups
|
|
:group
|
|
|
|
;; Displays in the console log the events through the application
|
|
:events
|
|
|
|
;; Display the boxes that represent the rotation and resize handlers
|
|
:handlers
|
|
|
|
;; Displays the center of a selection
|
|
:selection-center
|
|
|
|
;; When active the single selection will not take into account previous transformations
|
|
;; this is useful to debug transforms
|
|
:simple-selection
|
|
|
|
;; When active the thumbnails will be displayed with a sepia filter
|
|
:thumbnails
|
|
|
|
;; When active we can check in the browser the export values
|
|
:show-export-metadata
|
|
|
|
;; Show text fragments outlines
|
|
:text-outline
|
|
|
|
;; Disable thumbnail cache
|
|
:disable-thumbnail-cache
|
|
|
|
;; Disable frame thumbnails
|
|
:disable-frame-thumbnails
|
|
})
|
|
|
|
;; These events are excluded when we activate the :events flag
|
|
(def debug-exclude-events
|
|
#{:app.main.data.workspace.notifications/handle-pointer-update
|
|
:app.main.data.workspace.selection/change-hover-state})
|
|
|
|
(defonce ^:dynamic *debug* (atom #{#_:events}))
|
|
|
|
(defn debug-all! [] (reset! *debug* debug-options))
|
|
(defn debug-none! [] (reset! *debug* #{}))
|
|
(defn debug! [option] (swap! *debug* conj option))
|
|
(defn -debug! [option] (swap! *debug* disj option))
|
|
|
|
(defn ^:export ^boolean debug?
|
|
[option]
|
|
(if *assert*
|
|
(boolean (@*debug* option))
|
|
false))
|
|
|
|
(defn ^:export toggle-debug [name] (let [option (keyword name)]
|
|
(if (debug? option)
|
|
(-debug! option)
|
|
(debug! option))))
|
|
(defn ^:export debug-all [] (debug-all!))
|
|
(defn ^:export debug-none [] (debug-none!))
|
|
|
|
(defn ^:export tap
|
|
"Transducer function that can execute a side-effect `effect-fn` per input"
|
|
[effect-fn]
|
|
(fn [rf]
|
|
(fn
|
|
([] (rf))
|
|
([result] (rf result))
|
|
([result input]
|
|
(effect-fn input)
|
|
(rf result input)))))
|
|
|
|
(defn ^:export logjs
|
|
([str] (tap (partial logjs str)))
|
|
([str val]
|
|
(js/console.log str (clj->js val))
|
|
val))
|
|
|
|
(when (exists? js/window)
|
|
(set! (.-dbg ^js js/window) clj->js)
|
|
(set! (.-pp ^js js/window) pprint))
|
|
|
|
|
|
(defonce widget-style "
|
|
background: black;
|
|
bottom: 10px;
|
|
color: white;
|
|
height: 20px;
|
|
padding-left: 8px;
|
|
position: absolute;
|
|
right: 10px;
|
|
width: 40px;
|
|
z-index: 99999;
|
|
opacity: 0.5;
|
|
")
|
|
|
|
(defn ^:export fps
|
|
"Adds a widget to keep track of the average FPS's"
|
|
[]
|
|
(let [last (volatile! (.now js/performance))
|
|
avg (volatile! 0)
|
|
node (-> (.createElement js/document "div")
|
|
(obj/set! "id" "fps")
|
|
(obj/set! "style" widget-style))
|
|
body (obj/get js/document "body")
|
|
|
|
do-thing (fn do-thing []
|
|
(timers/raf
|
|
(fn []
|
|
(let [cur (.now js/performance)
|
|
ts (/ 1000 (* (- cur @last)))
|
|
val (+ @avg (* (- ts @avg) 0.1))]
|
|
|
|
(obj/set! node "innerText" val)
|
|
(vreset! last cur)
|
|
(vreset! avg val)
|
|
(do-thing)))))]
|
|
|
|
(.appendChild body node)
|
|
(do-thing)))
|
|
|
|
(defn ^:export dump-state []
|
|
(logjs "state" @st/state)
|
|
nil)
|
|
|
|
(defn ^:export dump-buffer []
|
|
(logjs "last-events" @st/last-events)
|
|
nil)
|
|
|
|
(defn ^:export get-state [str-path]
|
|
(let [path (->> (str/split str-path " ")
|
|
(map d/read-string)
|
|
vec)]
|
|
(js/console.log (clj->js (get-in @st/state path))))
|
|
nil)
|
|
|
|
(defn dump-objects'
|
|
[state]
|
|
(let [page-id (get state :current-page-id)
|
|
objects (get-in state [:workspace-data :pages-index page-id :objects])]
|
|
(logjs "objects" objects)
|
|
nil))
|
|
|
|
(defn ^:export dump-objects
|
|
[]
|
|
(dump-objects' @st/state))
|
|
|
|
(defn dump-object'
|
|
[state name]
|
|
(let [page-id (get state :current-page-id)
|
|
objects (get-in state [:workspace-data :pages-index page-id :objects])
|
|
result (or (d/seek (fn [[_ shape]] (= name (:name shape))) objects)
|
|
(get objects (uuid/uuid name)))]
|
|
(logjs name result)
|
|
nil))
|
|
|
|
(defn ^:export dump-object
|
|
[name]
|
|
(dump-object' @st/state name))
|
|
|
|
(defn dump-selected'
|
|
[state]
|
|
(let [page-id (get state :current-page-id)
|
|
objects (get-in state [:workspace-data :pages-index page-id :objects])
|
|
selected (get-in state [:workspace-local :selected])
|
|
result (->> selected (map (d/getf objects)))]
|
|
(logjs "selected" result)
|
|
nil))
|
|
|
|
(defn ^:export dump-selected
|
|
[]
|
|
(dump-selected' @st/state))
|
|
|
|
(defn dump-tree'
|
|
([state ] (dump-tree' state false false))
|
|
([state show-ids] (dump-tree' state show-ids false))
|
|
([state show-ids show-touched]
|
|
(let [page-id (get state :current-page-id)
|
|
objects (get-in state [:workspace-data :pages-index page-id :objects])
|
|
components (get-in state [:workspace-data :components])
|
|
libraries (get state :workspace-libraries)
|
|
root (d/seek #(nil? (:parent-id %)) (vals objects))]
|
|
|
|
(letfn [(show-shape [shape-id level objects]
|
|
(let [shape (get objects shape-id)]
|
|
(println (str/pad (str (str/repeat " " level)
|
|
(:name shape)
|
|
(when (seq (:touched shape)) "*")
|
|
(when show-ids (str/format " <%s>" (:id shape))))
|
|
{:length 20
|
|
:type :right})
|
|
(show-component shape objects))
|
|
(when show-touched
|
|
(when (seq (:touched shape))
|
|
(println (str (str/repeat " " level)
|
|
" "
|
|
(str (:touched shape)))))
|
|
(when (:remote-synced? shape)
|
|
(println (str (str/repeat " " level)
|
|
" (remote-synced)"))))
|
|
(when (:shapes shape)
|
|
(dorun (for [shape-id (:shapes shape)]
|
|
(show-shape shape-id (inc level) objects))))))
|
|
|
|
(show-component [shape objects]
|
|
(if (nil? (:shape-ref shape))
|
|
""
|
|
(let [root-shape (cph/get-component-shape objects shape)
|
|
component-id (when root-shape (:component-id root-shape))
|
|
component-file-id (when root-shape (:component-file root-shape))
|
|
component-file (when component-file-id (get libraries component-file-id nil))
|
|
component (when component-id
|
|
(if component-file
|
|
(get-in component-file [:data :components component-id])
|
|
(get components component-id)))
|
|
component-shape (when (and component (:shape-ref shape))
|
|
(get-in component [:objects (:shape-ref shape)]))]
|
|
(str/format " %s--> %s%s%s"
|
|
(cond (:component-root? shape) "#"
|
|
(:component-id shape) "@"
|
|
:else "-")
|
|
(when component-file (str/format "<%s> " (:name component-file)))
|
|
(or (:name component-shape) "?")
|
|
(if (or (:component-root? shape)
|
|
(nil? (:component-id shape))
|
|
true)
|
|
""
|
|
(let [component-id (:component-id shape)
|
|
component-file-id (:component-file shape)
|
|
component-file (when component-file-id (get libraries component-file-id nil))
|
|
component (if component-file
|
|
(get-in component-file [:data :components component-id])
|
|
(get components component-id))]
|
|
(str/format " (%s%s)"
|
|
(when component-file (str/format "<%s> " (:name component-file)))
|
|
(:name component))))))))]
|
|
|
|
(println "[Page]")
|
|
(show-shape (:id root) 0 objects)
|
|
|
|
(dorun (for [component (vals components)]
|
|
(do
|
|
(println)
|
|
(println (str/format "[%s]" (:name component)))
|
|
(show-shape (:id component) 0 (:objects component)))))))))
|
|
|
|
(defn ^:export dump-tree
|
|
([] (dump-tree' @st/state))
|
|
([show-ids] (dump-tree' @st/state show-ids))
|
|
([show-ids show-touched] (dump-tree' @st/state show-ids show-touched)))
|
|
|
|
(when *assert*
|
|
(defonce debug-subscription
|
|
(->> st/stream
|
|
(rx/filter ptk/event?)
|
|
(rx/filter (fn [s] (and (debug? :events)
|
|
(not (debug-exclude-events (ptk/type s))))))
|
|
(rx/subs #(println "[stream]: " (ptk/repr-event %))))))
|
|
|
|
(defn ^:export apply-changes
|
|
"Takes a Transit JSON changes"
|
|
[^string changes*]
|
|
|
|
(let [file-id (:current-file-id @st/state)
|
|
changes (t/decode-str changes*)]
|
|
(st/emit! (dwc/commit-changes {:redo-changes changes
|
|
:undo-changes []
|
|
:save-undo? true
|
|
:file-id file-id}))))
|
|
|
|
(defn ^:export fetch-apply
|
|
[^string url]
|
|
(-> (p/let [response (js/fetch url)]
|
|
(.text response))
|
|
(p/then apply-changes)))
|
|
|
|
(defn ^:export reset-viewport
|
|
[]
|
|
(st/emit!
|
|
dw/reset-zoom
|
|
(dw/update-viewport-position {:x (constantly 0) :y (constantly 0)})))
|
|
|
|
(defn ^:export hide-ui
|
|
[]
|
|
(st/emit!
|
|
(dw/toggle-layout-flag :hide-ui)))
|
|
|
|
|
|
(defn ^:export shortcuts
|
|
[]
|
|
|
|
(letfn [(print-shortcuts [shortcuts]
|
|
(.table js/console
|
|
(->> shortcuts
|
|
(map (fn [[key {:keys [command]}]]
|
|
[(d/name key)
|
|
(if (vector? command)
|
|
(str/join " | " command)
|
|
command)]))
|
|
(into {})
|
|
(clj->js))))]
|
|
(let [style "font-weight: bold; font-size: 1.25rem;"]
|
|
(.log js/console "%c Dashboard" style)
|
|
(print-shortcuts app.main.data.dashboard.shortcuts/shortcuts)
|
|
|
|
(.log js/console "%c Workspace" style)
|
|
(print-shortcuts app.main.data.workspace.shortcuts/shortcuts)
|
|
|
|
(.log js/console "%c Path" style)
|
|
(print-shortcuts app.main.data.workspace.path.shortcuts/shortcuts)
|
|
|
|
(.log js/console "%c Viewer" style)
|
|
(print-shortcuts app.main.data.viewer.shortcuts/shortcuts)))
|
|
nil)
|
|
|
|
(defn ^:export nodeStats
|
|
[]
|
|
(let [root-node (dom/query ".viewport .render-shapes")
|
|
num-nodes (->> (dom/seq-nodes root-node) count)]
|
|
#js {:number num-nodes}))
|