Merge pull request #3678 from penpot/alotor-codegen-fixes

Codegen & grid fixes
This commit is contained in:
Aitor Moreno 2023-10-02 12:46:09 +02:00 committed by GitHub
commit 828082ea47
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
13 changed files with 379 additions and 329 deletions

View file

@ -205,3 +205,6 @@
;; Modifiers ;; Modifiers
(dm/export gsm/set-objects-modifiers) (dm/export gsm/set-objects-modifiers)
;; Rect
(dm/export grc/rect->points)

View file

@ -107,9 +107,7 @@
margin margin
(if ignore-margin? (if ignore-margin?
0 0
(->> strokes (shape-stroke-margin shape stroke-width))
(map #(shape-stroke-margin % stroke-width))
(reduce d/max 0)))
shadow-width shadow-width
(->> (:shadow shape) (->> (:shadow shape)
@ -125,8 +123,8 @@
0)) 0))
(reduce d/max 0))] (reduce d/max 0))]
{:horizontal (+ stroke-width margin shadow-width) {:horizontal (mth/ceil (+ stroke-width margin shadow-width))
:vertical (+ stroke-width margin shadow-height)}))) :vertical (mth/ceil (+ stroke-width margin shadow-height))})))
(defn- add-padding (defn- add-padding
[bounds padding] [bounds padding]

View file

@ -16,6 +16,7 @@
[app.common.types.components-list :as ctkl] [app.common.types.components-list :as ctkl]
[app.common.types.pages-list :as ctpl] [app.common.types.pages-list :as ctpl]
[app.common.types.shape-tree :as ctst] [app.common.types.shape-tree :as ctst]
[app.common.types.shape.layout :as ctl]
[app.common.uuid :as uuid])) [app.common.uuid :as uuid]))
;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;; ;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;
@ -265,7 +266,7 @@
(gpt/add orig-pos delta) (gpt/add orig-pos delta)
{:skip-components? true {:skip-components? true
:bottom-frames? true})) :bottom-frames? true}))
frame-ids-map (volatile! {}) ids-map (volatile! {})
update-new-shape update-new-shape
(fn [new-shape original-shape] (fn [new-shape original-shape]
@ -275,8 +276,7 @@
(when (nil? (:parent-id original-shape)) (when (nil? (:parent-id original-shape))
(vswap! unames conj new-name)) (vswap! unames conj new-name))
(when (= (:type original-shape) :frame) (vswap! ids-map assoc (:id original-shape) (:id new-shape))
(vswap! frame-ids-map assoc (:id original-shape) (:id new-shape)))
(cond-> new-shape (cond-> new-shape
:always :always
@ -307,24 +307,29 @@
(dissoc :component-root)))) (dissoc :component-root))))
[new-shape new-shapes _] [new-shape new-shapes _]
(ctst/clone-object component-shape (ctst/clone-object
nil component-shape
(if components-v2 (:objects component-page) (:objects component)) nil
update-new-shape (if components-v2 (:objects component-page) (:objects component))
(fn [object _] object) update-new-shape
force-id (fn [object _] object)
keep-ids?) force-id
keep-ids?)
;; If frame-id points to a shape inside the component, remap it to the ;; If frame-id points to a shape inside the component, remap it to the
;; corresponding new frame shape. If not, set it to the destination frame. ;; corresponding new frame shape. If not, set it to the destination frame.
;; Also fix empty parent-id. ;; Also fix empty parent-id.
remap-frame-id (fn [shape] remap-ids
(as-> shape $ (fn [shape]
(update $ :frame-id #(get @frame-ids-map % frame-id)) (as-> shape $
(update $ :parent-id #(or % (:frame-id $)))))] (update $ :frame-id #(get @ids-map % frame-id))
(update $ :parent-id #(or % (:frame-id $)))
(cond-> $
(ctl/grid-layout? shape)
(ctl/remap-grid-cells @ids-map))))]
[(remap-frame-id new-shape) [(remap-ids new-shape)
(map remap-frame-id new-shapes)]))) (map remap-ids new-shapes)])))
(defn get-top-instance (defn get-top-instance
"The case of having an instance that contains another instances. "The case of having an instance that contains another instances.

View file

@ -1177,3 +1177,15 @@
(for [r (range first-row (inc last-row)) (for [r (range first-row (inc last-row))
c (range first-column (inc last-column))] c (range first-column (inc last-column))]
[r c])))) [r c]))))
(defn remap-grid-cells
"Remaps the shapes inside the cells"
[shape ids-map]
(let [do-remap-cells
(fn [cell]
(-> cell
(update :shapes #(mapv ids-map %))))
shape
(-> shape
(update :layout-grid-cells update-vals do-remap-cells))]
shape))

View file

@ -6,6 +6,7 @@
(ns common-tests.types-shape-interactions-test (ns common-tests.types-shape-interactions-test
(:require (:require
[app.common.math :as mth]
[app.common.exceptions :as ex] [app.common.exceptions :as ex]
[app.common.geom.point :as gpt] [app.common.geom.point :as gpt]
[app.common.geom.rect :as grc] [app.common.geom.rect :as grc]
@ -332,56 +333,56 @@
(t/testing "Overlay top-center relative to auto" (t/testing "Overlay top-center relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :top-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :top-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 0)) (t/is (mth/close? (:y overlay-pos) 0))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay top-right relative to auto" (t/testing "Overlay top-right relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :top-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :top-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 70)) (t/is (mth/close? (:x overlay-pos) 70))
(t/is (= (:y overlay-pos) 0)) (t/is (mth/close? (:y overlay-pos) 0))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay bottom-left relative to auto" (t/testing "Overlay bottom-left relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :bottom-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :bottom-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 0)) (t/is (mth/close? (:x overlay-pos) 0))
(t/is (= (:y overlay-pos) 80)) (t/is (mth/close? (:y overlay-pos) 80))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay bottom-center relative to auto" (t/testing "Overlay bottom-center relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :bottom-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :bottom-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 80)) (t/is (mth/close? (:y overlay-pos) 80))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay bottom-right relative to auto" (t/testing "Overlay bottom-right relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :bottom-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :bottom-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 70)) (t/is (mth/close? (:x overlay-pos) 70))
(t/is (= (:y overlay-pos) 80)) (t/is (mth/close? (:y overlay-pos) 80))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay center relative to auto" (t/testing "Overlay center relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 40)) (t/is (mth/close? (:y overlay-pos) 40))
(t/is (= snap-v :center)) (t/is (= snap-v :center))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay manual relative to auto" (t/testing "Overlay manual relative to auto"
(let [i2 (ctsi/set-overlay-pos-type interaction-auto :center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-auto :center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 40)) (t/is (mth/close? (:y overlay-pos) 40))
(t/is (= snap-v :center)) (t/is (= snap-v :center))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
@ -390,64 +391,64 @@
(ctsi/set-overlay-pos-type :manual base-frame objects) (ctsi/set-overlay-pos-type :manual base-frame objects)
(ctsi/set-overlay-position (gpt/point 12 62))) (ctsi/set-overlay-position (gpt/point 12 62)))
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 17)) (t/is (mth/close? (:x overlay-pos) 17))
(t/is (= (:y overlay-pos) 67)) (t/is (mth/close? (:y overlay-pos) 67))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay top-left relative to base-frame" (t/testing "Overlay top-left relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :top-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :top-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 5)) (t/is (mth/close? (:x overlay-pos) 5))
(t/is (= (:y overlay-pos) 5)) (t/is (mth/close? (:y overlay-pos) 5))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay top-center relative to base-frame" (t/testing "Overlay top-center relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :top-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :top-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 40)) (t/is (mth/close? (:x overlay-pos) 40))
(t/is (= (:y overlay-pos) 5)) (t/is (mth/close? (:y overlay-pos) 5))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay top-right relative to base-frame" (t/testing "Overlay top-right relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :top-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :top-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 75)) (t/is (mth/close? (:x overlay-pos) 75))
(t/is (= (:y overlay-pos) 5)) (t/is (mth/close? (:y overlay-pos) 5))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay bottom-left relative to base-frame" (t/testing "Overlay bottom-left relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :bottom-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :bottom-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 5)) (t/is (mth/close? (:x overlay-pos) 5))
(t/is (= (:y overlay-pos) 85)) (t/is (mth/close? (:y overlay-pos) 85))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay bottom-center relative to base-frame" (t/testing "Overlay bottom-center relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :bottom-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :bottom-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 40)) (t/is (mth/close? (:x overlay-pos) 40))
(t/is (= (:y overlay-pos) 85)) (t/is (mth/close? (:y overlay-pos) 85))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay bottom-right relative to base-frame" (t/testing "Overlay bottom-right relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :bottom-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :bottom-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 75)) (t/is (mth/close? (:x overlay-pos) 75))
(t/is (= (:y overlay-pos) 85)) (t/is (mth/close? (:y overlay-pos) 85))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay center relative to base-frame" (t/testing "Overlay center relative to base-frame"
(let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-base-frame :center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 40)) (t/is (mth/close? (:x overlay-pos) 40))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :center)) (t/is (= snap-v :center))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
@ -456,64 +457,64 @@
(ctsi/set-overlay-pos-type :manual base-frame objects) (ctsi/set-overlay-pos-type :manual base-frame objects)
(ctsi/set-overlay-position (gpt/point 12 62))) (ctsi/set-overlay-position (gpt/point 12 62)))
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects base-frame base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 17)) (t/is (mth/close? (:x overlay-pos) 17))
(t/is (= (:y overlay-pos) 67)) (t/is (mth/close? (:y overlay-pos) 67))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay top-left relative to popup" (t/testing "Overlay top-left relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 15)) (t/is (mth/close? (:x overlay-pos) 15))
(t/is (= (:y overlay-pos) 15)) (t/is (mth/close? (:y overlay-pos) 15))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay top-center relative to popup" (t/testing "Overlay top-center relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 25)) (t/is (mth/close? (:x overlay-pos) 25))
(t/is (= (:y overlay-pos) 15)) (t/is (mth/close? (:y overlay-pos) 15))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay top-right relative to popup" (t/testing "Overlay top-right relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 15)) (t/is (mth/close? (:y overlay-pos) 15))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay bottom-left relative to popup" (t/testing "Overlay bottom-left relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :bottom-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :bottom-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 15)) (t/is (mth/close? (:x overlay-pos) 15))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay bottom-center relative to popup" (t/testing "Overlay bottom-center relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :bottom-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :bottom-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 25)) (t/is (mth/close? (:x overlay-pos) 25))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay bottom-right relative to popup" (t/testing "Overlay bottom-right relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :bottom-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :bottom-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay center relative to popup" (t/testing "Overlay center relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 25)) (t/is (mth/close? (:x overlay-pos) 25))
(t/is (= (:y overlay-pos) 30)) (t/is (mth/close? (:y overlay-pos) 30))
(t/is (= snap-v :center)) (t/is (= snap-v :center))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
@ -522,64 +523,64 @@
(ctsi/set-overlay-pos-type :manual base-frame objects) (ctsi/set-overlay-pos-type :manual base-frame objects)
(ctsi/set-overlay-position (gpt/point 12 62))) (ctsi/set-overlay-position (gpt/point 12 62)))
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 27)) (t/is (mth/close? (:x overlay-pos) 27))
(t/is (= (:y overlay-pos) 77)) (t/is (mth/close? (:y overlay-pos) 77))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay top-left relative to popup" (t/testing "Overlay top-left relative to popup"
(let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-popup :top-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects popup base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 15)) (t/is (mth/close? (:x overlay-pos) 15))
(t/is (= (:y overlay-pos) 15)) (t/is (mth/close? (:y overlay-pos) 15))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay top-center relative to rect" (t/testing "Overlay top-center relative to rect"
(let [i2 (ctsi/set-overlay-pos-type interaction-rect :top-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-rect :top-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 25)) (t/is (mth/close? (:x overlay-pos) 25))
(t/is (= (:y overlay-pos) 15)) (t/is (mth/close? (:y overlay-pos) 15))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay top-right relative to rect" (t/testing "Overlay top-right relative to rect"
(let [i2 (ctsi/set-overlay-pos-type interaction-rect :top-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-rect :top-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 15)) (t/is (mth/close? (:y overlay-pos) 15))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay bottom-left relative to rect" (t/testing "Overlay bottom-left relative to rect"
(let [i2 (ctsi/set-overlay-pos-type interaction-rect :bottom-left base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-rect :bottom-left base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 15)) (t/is (mth/close? (:x overlay-pos) 15))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :left)))) (t/is (= snap-h :left))))
(t/testing "Overlay bottom-center relative to rect" (t/testing "Overlay bottom-center relative to rect"
(let [i2 (ctsi/set-overlay-pos-type interaction-rect :bottom-center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-rect :bottom-center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 25)) (t/is (mth/close? (:x overlay-pos) 25))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
(t/testing "Overlay bottom-right relative to rect" (t/testing "Overlay bottom-right relative to rect"
(let [i2 (ctsi/set-overlay-pos-type interaction-rect :bottom-right base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-rect :bottom-right base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 35)) (t/is (mth/close? (:x overlay-pos) 35))
(t/is (= (:y overlay-pos) 45)) (t/is (mth/close? (:y overlay-pos) 45))
(t/is (= snap-v :bottom)) (t/is (= snap-v :bottom))
(t/is (= snap-h :right)))) (t/is (= snap-h :right))))
(t/testing "Overlay center relative to rect" (t/testing "Overlay center relative to rect"
(let [i2 (ctsi/set-overlay-pos-type interaction-rect :center base-frame objects) (let [i2 (ctsi/set-overlay-pos-type interaction-rect :center base-frame objects)
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 25)) (t/is (mth/close? (:x overlay-pos) 25))
(t/is (= (:y overlay-pos) 30)) (t/is (mth/close? (:y overlay-pos) 30))
(t/is (= snap-v :center)) (t/is (= snap-v :center))
(t/is (= snap-h :center)))) (t/is (= snap-h :center))))
@ -588,8 +589,8 @@
(ctsi/set-overlay-pos-type :manual base-frame objects) (ctsi/set-overlay-pos-type :manual base-frame objects)
(ctsi/set-overlay-position (gpt/point 12 62))) (ctsi/set-overlay-position (gpt/point 12 62)))
[overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)] [overlay-pos [snap-v snap-h]] (ctsi/calc-overlay-position i2 rect objects rect base-frame overlay-frame frame-offset)]
(t/is (= (:x overlay-pos) 17)) (t/is (mth/close? (:x overlay-pos) 17))
(t/is (= (:y overlay-pos) 67)) (t/is (mth/close? (:y overlay-pos) 67))
(t/is (= snap-v :top)) (t/is (= snap-v :top))
(t/is (= snap-h :left)))))) (t/is (= snap-h :left))))))

View file

@ -330,17 +330,6 @@
(when selected (when selected
(rx/of (select-shape (:id selected)))))))) (rx/of (select-shape (:id selected))))))))
(defn remap-grid-cells
"Remaps the shapes inside the cells"
[shape ids-map]
(let [do-remap-cells
(fn [cell]
(-> cell
(update :shapes #(mapv ids-map %))))]
(update shape :layout-grid-cells update-vals do-remap-cells)))
;; --- Duplicate Shapes ;; --- Duplicate Shapes
(declare prepare-duplicate-shape-change) (declare prepare-duplicate-shape-change)
(declare prepare-duplicate-flows) (declare prepare-duplicate-flows)
@ -463,8 +452,7 @@
(d/update-when :interactions #(ctsi/remap-interactions % ids-map objects)) (d/update-when :interactions #(ctsi/remap-interactions % ids-map objects))
(cond-> (ctl/grid-layout? obj) (cond-> (ctl/grid-layout? obj)
(-> (ctl/check-deassigned-cells) (ctl/remap-grid-cells ids-map)))
(remap-grid-cells ids-map))))
new-obj (cond-> new-obj new-obj (cond-> new-obj
(not duplicating-component?) (not duplicating-component?)
@ -474,7 +462,7 @@
changes (-> (pcb/add-object changes new-obj {:ignore-touched (and duplicating-component? child?)}) changes (-> (pcb/add-object changes new-obj {:ignore-touched (and duplicating-component? child?)})
(pcb/amend-last-change #(assoc % :old-id (:id obj))) (pcb/amend-last-change #(assoc % :old-id (:id obj)))
(cond-> (ctl/grid-layout? objects (:parent-id obj)) (cond-> (ctl/grid-layout? objects (:parent-id obj))
(-> (pcb/update-shapes [(:parent-id obj)] ctl/assign-cells) (-> (pcb/update-shapes [(:parent-id obj)] (fn [shape] (-> shape ctl/assign-cells ctl/check-deassigned-cells)))
(pcb/reorder-grid-children [(:parent-id obj)])))) (pcb/reorder-grid-children [(:parent-id obj)]))))
changes (cond-> changes changes (cond-> changes

View file

@ -25,12 +25,11 @@
handle-load handle-load
(mf/use-callback (mf/use-callback
(fn [data width height] (fn [data width height]
(prn "handle-load" data)
(reset! last-data* data) (reset! last-data* data)
(let [iframe-dom (mf/ref-val iframe-ref)] (let [iframe-dom (mf/ref-val iframe-ref)]
(when iframe-dom (when iframe-dom
(-> iframe-dom (aset "width" width)) (-> iframe-dom (aset "width" (+ width 64)))
(-> iframe-dom (aset "height" height)) (-> iframe-dom (aset "height" (+ height 64)))
(-> iframe-dom .-contentWindow .-document .open) (-> iframe-dom .-contentWindow .-document .open)
(-> iframe-dom .-contentWindow .-document (.write data)) (-> iframe-dom .-contentWindow .-document (.write data))
(-> iframe-dom .-contentWindow .-document .close))))) (-> iframe-dom .-contentWindow .-document .close)))))
@ -59,13 +58,13 @@
[:div {:style {:display "flex" :width "100%" :height "100%" :flex-direction "column" :overflow "auto" :align-items "center"}} [:div {:style {:display "flex" :width "100%" :height "100%" :flex-direction "column" :overflow "auto" :align-items "center"}}
[:input {:id "zoom-input" [:input {:id "zoom-input"
:ref zoom-ref :ref zoom-ref
:type "range" :min 1 :max 200 :default-value 100 :type "range" :min 1 :max 400 :default-value 100
:on-change change-zoom :on-change change-zoom
:style {:max-width "500px"}}] :style {:max-width "500px"}}]
[:div {:style {:width "100%" :height "100%" :overflow "auto"}} [:div {:style {:width "100%" :height "100%" :overflow "auto"}}
[:iframe {:ref load-ref [:iframe {:ref load-ref
:frameborder "0" :frame-border "0"
:scrolling "no" :scrolling "no"
:style {:transform-origin "top center" :style {:transform-origin "top left"
:transform (str "scale(" zoom ")")}}]]])) :transform (str "scale(" zoom ")")}}]]]))

View file

@ -34,6 +34,9 @@
[cuerdas.core :as str] [cuerdas.core :as str]
[rumext.v2 :as mf])) [rumext.v2 :as mf]))
(def small-size-limit 60)
(def medium-size-limit 110)
(defn apply-to-point [result next-fn] (defn apply-to-point [result next-fn]
(conj result (next-fn (last result)))) (conj result (next-fn (last result))))
@ -630,6 +633,10 @@
[(:x text-p) (- (:y text-p) (/ 36 zoom)) (max 0 (:size track-data)) (/ 36 zoom)] [(:x text-p) (- (:y text-p) (/ 36 zoom)) (max 0 (:size track-data)) (/ 36 zoom)]
[(- (:x text-p) (max 0 (:size track-data))) (- (:y text-p) (/ 36 zoom)) (max 0 (:size track-data)) (/ 36 zoom)]) [(- (:x text-p) (max 0 (:size track-data))) (- (:y text-p) (/ 36 zoom)) (max 0 (:size track-data)) (/ 36 zoom)])
trackwidth (* text-width zoom)
medium? (and (>= trackwidth small-size-limit) (< trackwidth medium-size-limit))
small? (< trackwidth small-size-limit)
track-before (get-in layout-data [track-list-prop (dec index)])] track-before (get-in layout-data [track-list-prop (dec index)])]
(mf/use-effect (mf/use-effect
@ -644,28 +651,29 @@
(dm/str (gmt/transform-in text-p (:transform shape))) (dm/str (gmt/transform-in text-p (:transform shape)))
(dm/str (gmt/transform-in text-p (gmt/rotate (:transform shape) -90))))} (dm/str (gmt/transform-in text-p (gmt/rotate (:transform shape) -90))))}
(when hovering? (when (and hovering? (not small?))
[:rect {:x (+ text-x (/ 5 zoom)) [:rect {:x (+ text-x (/ 18 zoom))
:y text-y :y text-y
:width (- text-width (/ 10 zoom)) :width (- text-width (/ 36 zoom))
:height (- text-height (/ 5 zoom)) :height (- text-height (/ 5 zoom))
:rx (/ 3 zoom) :rx (/ 3 zoom)
:fill "var(--color-distance)" :fill "var(--color-distance)"
:opacity 0.2}]) :opacity 0.2}])
[:foreignObject {:x text-x :y text-y :width text-width :height text-height} (when (not small?)
[:div {:class (css :grid-editor-wrapper)} [:foreignObject {:x text-x :y text-y :width text-width :height text-height}
[:input [:div {:class (css :grid-editor-wrapper)}
{:ref track-input-ref [:input
:style {} {:ref track-input-ref
:class (css :grid-editor-label) :style {}
:type "text" :class (css :grid-editor-label)
:default-value (format-size track-data) :type "text"
:data-default-value (format-size track-data) :default-value (format-size track-data)
:on-key-down handle-keydown-track-input :data-default-value (format-size track-data)
:on-blur handle-blur-track-input}] :on-key-down handle-keydown-track-input
(when hovering? :on-blur handle-blur-track-input}]
[:button {:class (css :grid-editor-button) (when (and hovering? (not medium?) (not small?))
:on-click handle-remove-track} i/trash])]]] [:button {:class (css :grid-editor-button)
:on-click handle-remove-track} i/trash])]])]
[:g {:transform (when (= type :row) (dm/fmt "rotate(-90 % %)" (:x marker-p) (:y marker-p)))} [:g {:transform (when (= type :row) (dm/fmt "rotate(-90 % %)" (:x marker-p) (:y marker-p)))}
[:& track-marker [:& track-marker
@ -782,106 +790,107 @@
(fn [] (fn []
#(st/emit! (dwge/stop-grid-layout-editing (:id shape))))) #(st/emit! (dwge/stop-grid-layout-editing (:id shape)))))
[:g.grid-editor {:pointer-events (when view-only "none") (when (and (not (:hidden shape)) (not (:blocked shape)))
:on-pointer-down handle-pointer-down} [:g.grid-editor {:pointer-events (when view-only "none")
[:g.cells :on-pointer-down handle-pointer-down}
(for [cell (ctl/get-cells shape {:sort? true})] [:g.cells
[:& grid-cell {:key (dm/str "cell-" (:id cell)) (for [cell (ctl/get-cells shape {:sort? true})]
:shape base-shape [:& grid-cell {:key (dm/str "cell-" (:id cell))
:layout-data layout-data :shape base-shape
:cell cell :layout-data layout-data
:cell cell
:zoom zoom
:hover? (contains? hover-cells (:id cell))
:selected? (contains? selected-cells (:id cell))}])]
(when-not view-only
[:*
[:& grid-editor-frame {:zoom zoom
:bounds bounds}]
(let [start-p (-> origin (gpt/add (hv width)))]
[:g {:transform (dm/str (gmt/transform-in start-p (:transform shape)))}
[:& plus-btn {:start-p start-p
:zoom zoom
:type :column
:on-click handle-add-column}]])
(let [start-p (-> origin (gpt/add (vv height)))]
[:g {:transform (dm/str (gmt/transform-in start-p (:transform shape)))}
[:& plus-btn {:start-p start-p
:zoom zoom
:type :row
:on-click handle-add-row}]])
(for [[idx column-data] (d/enumerate column-tracks)]
[:& track {:key (dm/str "column-track-" idx)
:shape shape
:zoom zoom :zoom zoom
:hover? (contains? hover-cells (:id cell)) :type :column
:selected? (contains? selected-cells (:id cell))}])] :index idx
(when-not view-only :layout-data layout-data
[:* :snap-pixel? snap-pixel?
[:& grid-editor-frame {:zoom zoom :track-data column-data
:bounds bounds}] :hovering? (contains? hover-columns idx)}])
(let [start-p (-> origin (gpt/add (hv width)))]
[:g {:transform (dm/str (gmt/transform-in start-p (:transform shape)))}
[:& plus-btn {:start-p start-p
:zoom zoom
:type :column
:on-click handle-add-column}]])
(let [start-p (-> origin (gpt/add (vv height)))] ;; Last track resize handler
[:g {:transform (dm/str (gmt/transform-in start-p (:transform shape)))} (when-not (empty? column-tracks)
[:& plus-btn {:start-p start-p (let [last-track (last column-tracks)
:zoom zoom start-p (:start-p last-track)
:type :row end-p (gpt/add start-p (hv (:size last-track)))
:on-click handle-add-row}]]) marker-p (-> (gpo/project-point bounds :h end-p)
(gpt/subtract (vv (/ 20 zoom))))]
[:g.track
[:& track-marker {:center marker-p
:index (count column-tracks)
:shape shape
:snap-pixel? snap-pixel?
:track-before (last column-tracks)
:type :column
:value (dm/str (inc (count column-tracks)))
:zoom zoom}]
[:& resize-track-handler
{:index (count column-tracks)
:last? true
:shape shape
:layout-data layout-data
:snap-pixel? snap-pixel?
:start-p end-p
:type :column
:track-before (last column-tracks)
:zoom zoom}]]))
(for [[idx column-data] (d/enumerate column-tracks)] (for [[idx row-data] (d/enumerate row-tracks)]
[:& track {:key (dm/str "column-track-" idx) [:& track {:index idx
:shape shape :key (dm/str "row-track-" idx)
:zoom zoom :layout-data layout-data
:type :column :shape shape
:index idx :snap-pixel? snap-pixel?
:layout-data layout-data :track-data row-data
:snap-pixel? snap-pixel? :type :row
:track-data column-data :zoom zoom
:hovering? (contains? hover-columns idx)}]) :hovering? (contains? hover-rows idx)}])
(when-not (empty? row-tracks)
;; Last track resize handler (let [last-track (last row-tracks)
(when-not (empty? column-tracks) start-p (:start-p last-track)
(let [last-track (last column-tracks) end-p (gpt/add start-p (vv (:size last-track)))
start-p (:start-p last-track) marker-p (-> (gpo/project-point bounds :v end-p)
end-p (gpt/add start-p (hv (:size last-track))) (gpt/subtract (hv (/ 20 zoom))))]
marker-p (-> (gpo/project-point bounds :h end-p) [:g.track
(gpt/subtract (vv (/ 20 zoom))))] [:g {:transform (dm/fmt "rotate(-90 % %)" (:x marker-p) (:y marker-p))}
[:g.track [:& track-marker {:center marker-p
[:& track-marker {:center marker-p :index (count row-tracks)
:index (count column-tracks) :shape shape
:shape shape :snap-pixel? snap-pixel?
:snap-pixel? snap-pixel? :track-before (last row-tracks)
:track-before (last column-tracks) :type :row
:type :column :value (dm/str (inc (count row-tracks)))
:value (dm/str (inc (count column-tracks))) :zoom zoom}]]
:zoom zoom}] [:& resize-track-handler
[:& resize-track-handler {:index (count row-tracks)
{:index (count column-tracks) :last? true
:last? true :shape shape
:shape shape :layout-data layout-data
:layout-data layout-data :start-p end-p
:snap-pixel? snap-pixel? :type :row
:start-p end-p :track-before (last row-tracks)
:type :column :snap-pixel? snap-pixel?
:track-before (last column-tracks) :zoom zoom}]]))])])))
:zoom zoom}]]))
(for [[idx row-data] (d/enumerate row-tracks)]
[:& track {:index idx
:key (dm/str "row-track-" idx)
:layout-data layout-data
:shape shape
:snap-pixel? snap-pixel?
:track-data row-data
:type :row
:zoom zoom
:hovering? (contains? hover-rows idx)}])
(when-not (empty? row-tracks)
(let [last-track (last row-tracks)
start-p (:start-p last-track)
end-p (gpt/add start-p (vv (:size last-track)))
marker-p (-> (gpo/project-point bounds :v end-p)
(gpt/subtract (hv (/ 20 zoom))))]
[:g.track
[:g {:transform (dm/fmt "rotate(-90 % %)" (:x marker-p) (:y marker-p))}
[:& track-marker {:center marker-p
:index (count row-tracks)
:shape shape
:snap-pixel? snap-pixel?
:track-before (last row-tracks)
:type :row
:value (dm/str (inc (count row-tracks)))
:zoom zoom}]]
[:& resize-track-handler
{:index (count row-tracks)
:last? true
:shape shape
:layout-data layout-data
:start-p end-p
:type :row
:track-before (last row-tracks)
:snap-pixel? snap-pixel?
:zoom zoom}]]))])]))

View file

@ -46,7 +46,7 @@
margin: 0; margin: 0;
padding: 0; padding: 0;
position: absolute; position: absolute;
right: calc(10px / var(--zoom)); right: calc(20px / var(--zoom));
width: calc(20px / var(--zoom)); width: calc(20px / var(--zoom));
height: calc(20px / var(--zoom)); height: calc(20px / var(--zoom));
@ -80,12 +80,12 @@
.grid-cell-outline { .grid-cell-outline {
fill: transparent; fill: transparent;
stroke: var(--color-distance); stroke: var(--color-distance);
stroke-linecap: round; stroke-opacity: 0.5;
stroke-width: calc(2 / var(--zoom)); stroke-width: calc(1 / var(--zoom));
stroke-dasharray: 0 calc(8 / var(--zoom));
&.hover, &.hover,
&.selected { &.selected {
stroke-dasharray: initial; stroke-opacity: 1;
stroke-width: calc(2 / var(--zoom));
} }
} }

View file

@ -58,5 +58,6 @@
[objects shape] [objects shape]
;; Layout children with a transform should be wrapped ;; Layout children with a transform should be wrapped
(and (ctl/any-layout-immediate-child? objects shape) (and (ctl/any-layout-immediate-child? objects shape)
(not (ctl/layout-absolute? shape))
(not (gmt/unit? (:transform shape))))) (not (gmt/unit? (:transform shape)))))

View file

@ -23,68 +23,69 @@
(generate-html objects shape 0)) (generate-html objects shape 0))
([objects shape level] ([objects shape level]
(let [indent (str/repeat " " level) (when (and (some? shape) (some? (:selrect shape)))
maybe-reverse (if (ctl/any-layout? shape) reverse identity) (let [indent (str/repeat " " level)
maybe-reverse (if (ctl/any-layout? shape) reverse identity)
shape-html shape-html
(cond (cond
(cgc/svg-markup? shape) (cgc/svg-markup? shape)
(let [svg-markup (generate-svg objects shape)] (let [svg-markup (generate-svg objects shape)]
(dm/fmt "%<div class=\"%\">\n%\n%</div>"
indent
(dm/str "shape " (d/name (:type shape)) " "
(cgc/shape->selector shape))
svg-markup
indent))
(cph/text-shape? shape)
(let [text-shape-html (rds/renderToStaticMarkup (mf/element text/text-shape #js {:shape shape :code? true}))]
(dm/fmt "%<div class=\"%\">\n%\n%</div>"
indent
(dm/str "shape " (d/name (:type shape)) " "
(cgc/shape->selector shape))
text-shape-html
indent))
(cph/image-shape? shape)
(let [data (or (:metadata shape) (:fill-image shape))
image-url (cfg/resolve-file-media data)]
(dm/fmt "%<img src=\"%\" class=\"%\">\n%</img>"
indent
image-url
(dm/str "shape " (d/name (:type shape)) " "
(cgc/shape->selector shape))
indent))
(empty? (:shapes shape))
(dm/fmt "%<div class=\"%\">\n%</div>"
indent
(dm/str "shape " (d/name (:type shape)) " "
(cgc/shape->selector shape))
indent)
:else
(dm/fmt "%<div class=\"%\">\n%\n%</div>" (dm/fmt "%<div class=\"%\">\n%\n%</div>"
indent indent
(dm/str "shape " (d/name (:type shape)) " " (dm/str (d/name (:type shape)) " "
(cgc/shape->selector shape)) (cgc/shape->selector shape))
svg-markup (->> (:shapes shape)
(maybe-reverse)
(map #(generate-html objects (get objects %) (inc level)))
(str/join "\n"))
indent)) indent))
(cph/text-shape? shape) shape-html
(let [text-shape-html (rds/renderToStaticMarkup (mf/element text/text-shape #js {:shape shape :code? true}))] (if (cgc/has-wrapper? objects shape)
(dm/fmt "%<div class=\"%\">\n%\n%</div>" (dm/fmt "<div class=\"%\">%</div>"
indent (dm/str (cgc/shape->selector shape) "-wrapper")
(dm/str "shape " (d/name (:type shape)) " " shape-html)
(cgc/shape->selector shape))
text-shape-html
indent))
(cph/image-shape? shape) shape-html)]
(let [data (or (:metadata shape) (:fill-image shape)) (dm/fmt "%<!-- % -->\n%" indent (dm/str (d/name (:type shape)) ": " (:name shape)) shape-html)))))
image-url (cfg/resolve-file-media data)]
(dm/fmt "%<img src=\"%\" class=\"%\">\n%</img>"
indent
image-url
(dm/str "shape " (d/name (:type shape)) " "
(cgc/shape->selector shape))
indent))
(empty? (:shapes shape))
(dm/fmt "%<div class=\"%\">\n%</div>"
indent
(dm/str "shape " (d/name (:type shape)) " "
(cgc/shape->selector shape))
indent)
:else
(dm/fmt "%<div class=\"%\">\n%\n%</div>"
indent
(dm/str (d/name (:type shape)) " "
(cgc/shape->selector shape))
(->> (:shapes shape)
(maybe-reverse)
(map #(generate-html objects (get objects %) (inc level)))
(str/join "\n"))
indent))
shape-html
(if (cgc/has-wrapper? objects shape)
(dm/fmt "<div class=\"%\">%</div>"
(dm/str (cgc/shape->selector shape) "-wrapper")
shape-html)
shape-html)]
(dm/fmt "%<!-- % -->\n%" indent (dm/str (d/name (:type shape)) ": " (:name shape)) shape-html))))
(defn generate-markup (defn generate-markup
[objects shapes] [objects shapes]
(->> shapes (->> shapes
(map #(generate-html objects %)) (keep #(generate-html objects %))
(str/join "\n"))) (str/join "\n")))

View file

@ -9,7 +9,8 @@
[app.common.data :as d] [app.common.data :as d]
[app.common.data.macros :as dm] [app.common.data.macros :as dm]
[app.common.geom.matrix :as gmt] [app.common.geom.matrix :as gmt]
[app.common.geom.shapes :as gsh] [app.common.geom.shapes.bounds :as gsb]
[app.common.geom.shapes.points :as gpo]
[app.common.pages.helpers :as cph] [app.common.pages.helpers :as cph]
[app.common.text :as txt] [app.common.text :as txt]
[app.common.types.shape.layout :as ctl] [app.common.types.shape.layout :as ctl]
@ -38,13 +39,6 @@ body {
gap: 2rem; gap: 2rem;
} }
svg {
position: absolute;
left: 50%;
top: 50%;
transform: translate(-50%, -50%);
}
* { * {
box-sizing: border-box; box-sizing: border-box;
} }
@ -64,7 +58,10 @@ svg {
:justify-self :justify-self
:grid-column :grid-column
:grid-row :grid-row
:z-index}) :z-index
:top
:left
:position})
(def shape-css-properties (def shape-css-properties
[:position [:position
@ -139,12 +136,16 @@ svg {
[shape objects] [shape objects]
(when (and (ctl/any-layout-immediate-child? objects shape) (when (and (ctl/any-layout-immediate-child? objects shape)
(not (gmt/unit? (:transform shape)))) (not (gmt/unit? (:transform shape))))
(let [{:keys [width height]} (gsh/shapes->rect [shape])] (let [parent (get objects (:parent-id shape))
(cond-> [[:position "relative"] bounds (gpo/parent-coords-bounds (:points shape) (:points parent))
[:width width] width (gpo/width-points bounds)
height (gpo/height-points bounds)]
(cond-> [[:width width]
[:height height]] [:height height]]
(ctl/flex-layout-immediate-child? objects shape)
(conj [:flex-shrink 0]))))) (or (not (ctl/any-layout-immediate-child? objects shape))
(not (ctl/layout-absolute? shape)))
(conj [:position "relative"])))))
(defn shape->wrapper-child-css-properties (defn shape->wrapper-child-css-properties
[shape objects] [shape objects]
@ -153,6 +154,16 @@ svg {
[:left "50%"] [:left "50%"]
[:top "50%"]])) [:top "50%"]]))
(defn shape->svg-props
[shape objects]
(let [bounds (gsb/get-object-bounds objects shape)]
[[:position "absolute"]
[:top 0]
[:left 0]
[:transform (dm/fmt "translate(%,%)"
(dm/str (- (:x bounds) (-> shape :selrect :x)) "px")
(dm/str (- (:y bounds) (-> shape :selrect :y)) "px"))]]))
(defn shape->css-properties (defn shape->css-properties
"Given a shape extract the CSS properties in the format of list [property value]" "Given a shape extract the CSS properties in the format of list [property value]"
[shape objects properties] [shape objects properties]
@ -235,40 +246,49 @@ svg {
(get-shape-css-selector shape objects nil)) (get-shape-css-selector shape objects nil))
([shape objects options] ([shape objects options]
(let [selector (cgc/shape->selector shape) (when (and (some? shape) (some? (:selrect shape)))
(let [selector (cgc/shape->selector shape)
wrapper? (cgc/has-wrapper? objects shape) wrapper? (cgc/has-wrapper? objects shape)
svg? (cgc/svg-markup? shape)
css-properties css-properties
(if wrapper? (if wrapper?
(filter (complement shape-wrapper-css-properties) shape-css-properties) (filter (complement shape-wrapper-css-properties) shape-css-properties)
shape-css-properties) shape-css-properties)
properties properties
(-> shape
(shape->css-properties objects css-properties)
(format-css-properties options))
wrapper-properties
(when wrapper?
(-> (d/concat-vec
(shape->css-properties shape objects shape-wrapper-css-properties)
(shape->wrapper-css-properties shape objects))
(format-css-properties options)))
wrapper-child-properties
(when wrapper?
(-> shape (-> shape
(shape->wrapper-child-css-properties objects) (shape->css-properties objects css-properties)
(format-css-properties options)))] (format-css-properties options))
(str/join wrapper-properties
"\n" (when wrapper?
(filter some? [(str/fmt "/* %s */" (:name shape)) (-> (d/concat-vec
(when wrapper? (str/fmt ".%s-wrapper {\n%s\n}" selector wrapper-properties)) (shape->css-properties shape objects shape-wrapper-css-properties)
(when wrapper? (str/fmt ".%s-wrapper > * {\n%s\n}" selector wrapper-child-properties)) (shape->wrapper-css-properties shape objects))
(str/fmt ".%s {\n%s\n}" selector properties) (format-css-properties options)))
(when (cph/text-shape? shape) (generate-text-css shape))])))))
wrapper-child-properties
(when wrapper?
(-> shape
(shape->wrapper-child-css-properties objects)
(format-css-properties options)))
svg-child-props
(when svg?
(-> shape
(shape->svg-props objects)
(format-css-properties options)))]
(str/join
"\n"
(filter some? [(str/fmt "/* %s */" (:name shape))
(when wrapper? (str/fmt ".%s-wrapper {\n%s\n}" selector wrapper-properties))
(when wrapper? (str/fmt ".%s-wrapper > * {\n%s\n}" selector wrapper-child-properties))
(when svg? (str/fmt ".%s > svg {\n%s\n}" selector svg-child-props))
(str/fmt ".%s {\n%s\n}" selector properties)
(when (cph/text-shape? shape) (generate-text-css shape))]))))))
(defn get-css-property (defn get-css-property
([objects shape property] ([objects shape property]
@ -294,5 +314,5 @@ svg {
(dm/str (dm/str
prelude prelude
(->> shapes (->> shapes
(map #(get-shape-css-selector % objects options)) (keep #(get-shape-css-selector % objects options))
(str/join "\n\n"))))) (str/join "\n\n")))))

View file

@ -49,12 +49,14 @@
(when (and (not (cph/root-frame? shape)) (when (and (not (cph/root-frame? shape))
(or (not (ctl/any-layout-immediate-child? objects shape)) (or (not (ctl/any-layout-immediate-child? objects shape))
(ctl/layout-absolute? shape))) (ctl/layout-absolute? shape)))
(let [parent (get objects (:parent-id shape)) (let [parent (get objects (:parent-id shape))
parent-value (dm/get-in parent [:selrect coord]) parent-value (dm/get-in parent [:selrect coord])
[selrect _ _] [selrect _ _]
(-> (:points shape) (-> (:points shape)
(gsh/transform-points (gsh/shape->center parent) (:transform-inverse parent)) (gsh/transform-points (gsh/shape->center parent) (:transform-inverse parent (gmt/matrix)))
(gsh/calculate-geometry)) (gsh/calculate-geometry))
shape-value (get selrect coord)] shape-value (get selrect coord)]
@ -117,7 +119,17 @@
(defmethod get-value :transform (defmethod get-value :transform
[_ shape objects] [_ shape objects]
(when-not (cgc/svg-markup? shape) (if (cgc/svg-markup? shape)
(let [parent (get objects (:parent-id shape))
transform
(:transform-inverse parent (gmt/matrix))
transform-str (when-not (gmt/unit? transform) (fmt/format-matrix transform))]
(if (cgc/has-wrapper? objects shape)
(dm/str "translate(-50%, -50%) " (d/nilv transform-str ""))
transform-str))
(let [parent (get objects (:parent-id shape)) (let [parent (get objects (:parent-id shape))
transform transform
@ -207,6 +219,7 @@
(defmethod get-value :overflow (defmethod get-value :overflow
[_ shape _] [_ shape _]
(when (and (cph/frame-shape? shape) (when (and (cph/frame-shape? shape)
(not (cgc/svg-markup? shape))
(not (:show-content shape))) (not (:show-content shape)))
"hidden")) "hidden"))