mirror of
https://github.com/penpot/penpot.git
synced 2025-04-29 14:16:21 +02:00
♻️ Adapt linear gradient type so it can be used for radial too (wasm)
This commit is contained in:
parent
f40ef26c69
commit
abcd050c69
6 changed files with 32 additions and 28 deletions
|
@ -50,12 +50,16 @@
|
||||||
(def GRID-LAYOUT-ROW-ENTRY-SIZE 5)
|
(def GRID-LAYOUT-ROW-ENTRY-SIZE 5)
|
||||||
(def GRID-LAYOUT-COLUMN-ENTRY-SIZE 5)
|
(def GRID-LAYOUT-COLUMN-ENTRY-SIZE 5)
|
||||||
(def GRID-LAYOUT-CELL-ENTRY-SIZE 37)
|
(def GRID-LAYOUT-CELL-ENTRY-SIZE 37)
|
||||||
(def GRADIENT-STOP-SIZE 5)
|
|
||||||
(def LINEAR-FILL-BASE-SIZE 21)
|
|
||||||
|
|
||||||
(defn gradient-stop-get-entries-size
|
(defn gradient-stop-get-entries-size
|
||||||
[stops]
|
[stops]
|
||||||
(mem/get-list-size stops GRADIENT-STOP-SIZE))
|
(mem/get-list-size stops sr-fills/GRADIENT-STOP-SIZE))
|
||||||
|
|
||||||
|
(defn gradient-byte-size
|
||||||
|
[gradient]
|
||||||
|
(let [stops (:stops gradient)]
|
||||||
|
(+ sr-fills/GRADIENT-BASE-SIZE (* (count stops) sr-fills/GRADIENT-STOP-SIZE))))
|
||||||
|
|
||||||
(defn modifier-get-entries-size
|
(defn modifier-get-entries-size
|
||||||
"Returns the list of a modifier list in bytes"
|
"Returns the list of a modifier list in bytes"
|
||||||
|
@ -232,8 +236,7 @@
|
||||||
(some? gradient)
|
(some? gradient)
|
||||||
(case (:type gradient)
|
(case (:type gradient)
|
||||||
:linear
|
:linear
|
||||||
(let [stops (:stops gradient)
|
(let [size (gradient-byte-size gradient)
|
||||||
size (+ LINEAR-FILL-BASE-SIZE (* (count stops) GRADIENT-STOP-SIZE))
|
|
||||||
offset (mem/alloc-bytes size)
|
offset (mem/alloc-bytes size)
|
||||||
heap (mem/get-heap-u8)]
|
heap (mem/get-heap-u8)]
|
||||||
(sr-fills/serialize-linear-fill gradient opacity heap offset)
|
(sr-fills/serialize-linear-fill gradient opacity heap offset)
|
||||||
|
|
|
@ -10,8 +10,6 @@
|
||||||
[app.common.uuid :as uuid]
|
[app.common.uuid :as uuid]
|
||||||
[cuerdas.core :as str]))
|
[cuerdas.core :as str]))
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
(defn u8
|
(defn u8
|
||||||
[value]
|
[value]
|
||||||
(let [u8-arr (js/Uint8Array. 1)]
|
(let [u8-arr (js/Uint8Array. 1)]
|
||||||
|
|
|
@ -3,6 +3,9 @@
|
||||||
[app.common.data.macros :as dm]
|
[app.common.data.macros :as dm]
|
||||||
[app.render-wasm.serializers.color :as clr]))
|
[app.render-wasm.serializers.color :as clr]))
|
||||||
|
|
||||||
|
(def GRADIENT-STOP-SIZE 5)
|
||||||
|
(def GRADIENT-BASE-SIZE 24)
|
||||||
|
|
||||||
(defn serialize-linear-fill
|
(defn serialize-linear-fill
|
||||||
[gradient opacity heap-u8 offset]
|
[gradient opacity heap-u8 offset]
|
||||||
(let [dview (js/DataView. (.-buffer heap-u8))
|
(let [dview (js/DataView. (.-buffer heap-u8))
|
||||||
|
@ -10,13 +13,14 @@
|
||||||
start-y (dm/get-prop gradient :start-y)
|
start-y (dm/get-prop gradient :start-y)
|
||||||
end-x (dm/get-prop gradient :end-x)
|
end-x (dm/get-prop gradient :end-x)
|
||||||
end-y (dm/get-prop gradient :end-y)
|
end-y (dm/get-prop gradient :end-y)
|
||||||
stops (dm/get-prop gradient :stops)]
|
stops (dm/get-prop gradient :stops)
|
||||||
|
width 0]
|
||||||
(.setFloat32 dview offset start-x true)
|
(.setFloat32 dview offset start-x true)
|
||||||
(.setFloat32 dview (+ offset 4) start-y true)
|
(.setFloat32 dview (+ offset 4) start-y true)
|
||||||
(.setFloat32 dview (+ offset 8) end-x true)
|
(.setFloat32 dview (+ offset 8) end-x true)
|
||||||
(.setFloat32 dview (+ offset 12) end-y true)
|
(.setFloat32 dview (+ offset 12) end-y true)
|
||||||
(.setFloat32 dview (+ offset 16) opacity true)
|
(.setFloat32 dview (+ offset 16) opacity true)
|
||||||
(.setUint8 dview (+ offset 20) (count stops))
|
(.setFloat32 dview (+ offset 20) width true)
|
||||||
(loop [stops (seq stops) idx 0]
|
(loop [stops (seq stops) idx 0]
|
||||||
(when-not (empty? stops)
|
(when-not (empty? stops)
|
||||||
(let [stop (first stops)
|
(let [stop (first stops)
|
||||||
|
@ -24,7 +28,7 @@
|
||||||
opacity (dm/get-prop stop :opacity)
|
opacity (dm/get-prop stop :opacity)
|
||||||
rgba (clr/hex->u32argb hex-color opacity)
|
rgba (clr/hex->u32argb hex-color opacity)
|
||||||
stop-offset (* 100 (dm/get-prop stop :offset))
|
stop-offset (* 100 (dm/get-prop stop :offset))
|
||||||
dview-offset (+ (* idx 5) offset 21)]
|
dview-offset (+ (* idx 5) offset 24)]
|
||||||
(.setUint32 dview dview-offset rgba true)
|
(.setUint32 dview dview-offset rgba true)
|
||||||
(.setUint8 dview (+ dview-offset 4) stop-offset)
|
(.setUint8 dview (+ dview-offset 4) stop-offset)
|
||||||
(recur (rest stops) (+ idx 1)))))))
|
(recur (rest stops) (+ idx 1)))))))
|
|
@ -17,8 +17,8 @@ mod wasm;
|
||||||
|
|
||||||
use crate::mem::SerializableResult;
|
use crate::mem::SerializableResult;
|
||||||
use crate::shapes::{
|
use crate::shapes::{
|
||||||
BoolType, ConstraintH, ConstraintV, StructureEntry, TransformEntry, Type,
|
BoolType, ConstraintH, ConstraintV, StructureEntry, TransformEntry, Type, RAW_FILL_DATA_SIZE,
|
||||||
RAW_LINEAR_FILL_DATA_SIZE, RAW_STOP_DATA_SIZE,
|
RAW_STOP_DATA_SIZE,
|
||||||
};
|
};
|
||||||
use crate::utils::uuid_from_u32_quartet;
|
use crate::utils::uuid_from_u32_quartet;
|
||||||
use crate::uuid::Uuid;
|
use crate::uuid::Uuid;
|
||||||
|
@ -260,12 +260,11 @@ pub extern "C" fn add_shape_solid_fill(raw_color: u32) {
|
||||||
#[no_mangle]
|
#[no_mangle]
|
||||||
pub extern "C" fn add_shape_linear_fill() {
|
pub extern "C" fn add_shape_linear_fill() {
|
||||||
with_current_shape!(state, |shape: &mut Shape| {
|
with_current_shape!(state, |shape: &mut Shape| {
|
||||||
let stops_offset = RAW_LINEAR_FILL_DATA_SIZE;
|
|
||||||
let bytes = mem::bytes();
|
let bytes = mem::bytes();
|
||||||
let raw_fill_data: [u8; RAW_LINEAR_FILL_DATA_SIZE] =
|
let raw_gradient_bytes: [u8; RAW_FILL_DATA_SIZE] =
|
||||||
bytes[0..stops_offset].try_into().unwrap();
|
bytes[0..RAW_FILL_DATA_SIZE].try_into().unwrap();
|
||||||
let raw_fill = shapes::RawLinearFillData::from(raw_fill_data);
|
let raw_gradient = shapes::RawGradientData::from(raw_gradient_bytes);
|
||||||
let stops: Vec<shapes::RawStopData> = bytes[stops_offset..]
|
let stops: Vec<shapes::RawStopData> = bytes[RAW_FILL_DATA_SIZE..]
|
||||||
.chunks(RAW_STOP_DATA_SIZE)
|
.chunks(RAW_STOP_DATA_SIZE)
|
||||||
.map(|chunk| {
|
.map(|chunk| {
|
||||||
let data: [u8; RAW_STOP_DATA_SIZE] = chunk.try_into().unwrap();
|
let data: [u8; RAW_STOP_DATA_SIZE] = chunk.try_into().unwrap();
|
||||||
|
@ -274,9 +273,9 @@ pub extern "C" fn add_shape_linear_fill() {
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
shape.add_fill(shapes::Fill::new_linear_gradient_with_stops(
|
shape.add_fill(shapes::Fill::new_linear_gradient_with_stops(
|
||||||
raw_fill.start(),
|
raw_gradient.start(),
|
||||||
raw_fill.end(),
|
raw_gradient.end(),
|
||||||
raw_fill.opacity(),
|
raw_gradient.opacity(),
|
||||||
stops,
|
stops,
|
||||||
));
|
));
|
||||||
});
|
});
|
||||||
|
|
|
@ -20,7 +20,7 @@ pub extern "C" fn alloc_bytes(len: usize) -> *mut u8 {
|
||||||
if ptr.is_null() {
|
if ptr.is_null() {
|
||||||
panic!("Allocation failed");
|
panic!("Allocation failed");
|
||||||
}
|
}
|
||||||
// TODO: Esto quizá se podría eliminar.
|
// TODO: Maybe this could be removed.
|
||||||
ptr::write_bytes(ptr, 0, len);
|
ptr::write_bytes(ptr, 0, len);
|
||||||
*guard = Some(Box::new(Vec::from_raw_parts(ptr, len, len)));
|
*guard = Some(Box::new(Vec::from_raw_parts(ptr, len, len)));
|
||||||
ptr
|
ptr
|
||||||
|
|
|
@ -3,33 +3,33 @@ use skia_safe::{self as skia, Rect};
|
||||||
use super::Color;
|
use super::Color;
|
||||||
use crate::uuid::Uuid;
|
use crate::uuid::Uuid;
|
||||||
|
|
||||||
pub const RAW_LINEAR_FILL_DATA_SIZE: usize = 21;
|
pub const RAW_FILL_DATA_SIZE: usize = 24;
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
#[repr(C)]
|
#[repr(C)]
|
||||||
pub struct RawLinearFillData {
|
pub struct RawGradientData {
|
||||||
start_x: f32,
|
start_x: f32,
|
||||||
start_y: f32,
|
start_y: f32,
|
||||||
end_x: f32,
|
end_x: f32,
|
||||||
end_y: f32,
|
end_y: f32,
|
||||||
opacity: f32,
|
opacity: f32,
|
||||||
stop_count: u8,
|
width: f32,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<[u8; 21]> for RawLinearFillData {
|
impl From<[u8; RAW_FILL_DATA_SIZE]> for RawGradientData {
|
||||||
fn from(bytes: [u8; 21]) -> Self {
|
fn from(bytes: [u8; RAW_FILL_DATA_SIZE]) -> Self {
|
||||||
Self {
|
Self {
|
||||||
start_x: f32::from_le_bytes([bytes[0], bytes[1], bytes[2], bytes[3]]),
|
start_x: f32::from_le_bytes([bytes[0], bytes[1], bytes[2], bytes[3]]),
|
||||||
start_y: f32::from_le_bytes([bytes[4], bytes[5], bytes[6], bytes[7]]),
|
start_y: f32::from_le_bytes([bytes[4], bytes[5], bytes[6], bytes[7]]),
|
||||||
end_x: f32::from_le_bytes([bytes[8], bytes[9], bytes[10], bytes[11]]),
|
end_x: f32::from_le_bytes([bytes[8], bytes[9], bytes[10], bytes[11]]),
|
||||||
end_y: f32::from_le_bytes([bytes[12], bytes[13], bytes[14], bytes[15]]),
|
end_y: f32::from_le_bytes([bytes[12], bytes[13], bytes[14], bytes[15]]),
|
||||||
opacity: f32::from_le_bytes([bytes[16], bytes[17], bytes[18], bytes[19]]),
|
opacity: f32::from_le_bytes([bytes[16], bytes[17], bytes[18], bytes[19]]),
|
||||||
stop_count: bytes[20],
|
width: f32::from_le_bytes([bytes[20], bytes[21], bytes[22], bytes[23]]),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RawLinearFillData {
|
impl RawGradientData {
|
||||||
pub fn start(&self) -> (f32, f32) {
|
pub fn start(&self) -> (f32, f32) {
|
||||||
(self.start_x, self.start_y)
|
(self.start_x, self.start_y)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Reference in a new issue