use crate::boolean_ops::BooleanOperation as BooleanOperationType; use crate::layers::blend_mode::BlendMode; use crate::layers::layer_info::Layer; use crate::layers::style::{self, Stroke}; use crate::layers::vector::consts::ManipulatorType; use crate::layers::vector::manipulator_group::ManipulatorGroup; use crate::layers::vector::subpath::Subpath; use crate::LayerId; use serde::{Deserialize, Serialize}; use std::collections::hash_map::DefaultHasher; use std::hash::{Hash, Hasher}; #[repr(C)] #[derive(Debug, Clone, Deserialize, Serialize, PartialEq)] // TODO: Rename all instances of `path` to `layer_path` /// Operations that can be performed to mutate the document. pub enum Operation { AddEllipse { path: Vec, insert_index: isize, transform: [f64; 6], style: style::PathStyle, }, AddRect { path: Vec, insert_index: isize, transform: [f64; 6], style: style::PathStyle, }, AddLine { path: Vec, insert_index: isize, transform: [f64; 6], style: style::PathStyle, }, AddText { path: Vec, transform: [f64; 6], insert_index: isize, text: String, style: style::PathStyle, size: f64, font_name: String, font_style: String, }, AddImage { path: Vec, transform: [f64; 6], insert_index: isize, mime: String, image_data: Vec, }, SetImageBlobUrl { path: Vec, blob_url: String, dimensions: (f64, f64), }, SetTextEditability { path: Vec, editable: bool, }, SetTextContent { path: Vec, new_text: String, }, AddPolyline { path: Vec, transform: [f64; 6], insert_index: isize, points: Vec<(f64, f64)>, style: style::PathStyle, }, AddSpline { path: Vec, transform: [f64; 6], insert_index: isize, points: Vec<(f64, f64)>, style: style::PathStyle, }, AddNgon { path: Vec, insert_index: isize, transform: [f64; 6], sides: u32, style: style::PathStyle, }, AddShape { path: Vec, transform: [f64; 6], insert_index: isize, // TODO This will become a compound path once we support them. subpath: Subpath, style: style::PathStyle, }, BooleanOperation { operation: BooleanOperationType, selected: Vec>, }, DeleteLayer { path: Vec, }, DeleteSelectedManipulatorPoints { layer_paths: Vec>, }, DeselectManipulatorPoints { layer_path: Vec, point_ids: Vec<(u64, ManipulatorType)>, }, DeselectAllManipulatorPoints { layer_path: Vec, }, DuplicateLayer { path: Vec, }, ModifyFont { path: Vec, font_family: String, font_style: String, size: f64, }, MoveSelectedManipulatorPoints { layer_path: Vec, delta: (f64, f64), absolute_position: (f64, f64), }, MoveManipulatorPoint { layer_path: Vec, id: u64, manipulator_type: ManipulatorType, position: (f64, f64), }, RenameLayer { layer_path: Vec, new_name: String, }, InsertLayer { layer: Layer, destination_path: Vec, insert_index: isize, }, CreateFolder { path: Vec, }, TransformLayer { path: Vec, transform: [f64; 6], }, TransformLayerInViewport { path: Vec, transform: [f64; 6], }, SetLayerTransformInViewport { path: Vec, transform: [f64; 6], }, SelectManipulatorPoints { layer_path: Vec, point_ids: Vec<(u64, ManipulatorType)>, add: bool, }, SetShapePath { path: Vec, subpath: Subpath, }, InsertManipulatorGroup { layer_path: Vec, manipulator_group: ManipulatorGroup, after_id: u64, }, PushManipulatorGroup { layer_path: Vec, manipulator_group: ManipulatorGroup, }, RemoveManipulatorGroup { layer_path: Vec, id: u64, }, RemoveManipulatorPoint { layer_path: Vec, id: u64, manipulator_type: ManipulatorType, }, TransformLayerInScope { path: Vec, transform: [f64; 6], scope: [f64; 6], }, SetLayerTransformInScope { path: Vec, transform: [f64; 6], scope: [f64; 6], }, SetLayerTransform { path: Vec, transform: [f64; 6], }, ToggleLayerVisibility { path: Vec, }, SetLayerVisibility { path: Vec, visible: bool, }, SetLayerName { path: Vec, name: String, }, SetLayerBlendMode { path: Vec, blend_mode: BlendMode, }, SetLayerOpacity { path: Vec, opacity: f64, }, SetLayerStyle { path: Vec, style: style::PathStyle, }, SetLayerFill { path: Vec, fill: style::Fill, }, SetLayerStroke { path: Vec, stroke: Stroke, }, SetSelectedHandleMirroring { layer_path: Vec, toggle_distance: bool, toggle_angle: bool, }, } impl Operation { /// Returns the byte representation of the message. /// /// # Safety /// This function reads from uninitialized memory!!! /// Only use if you know what you are doing unsafe fn as_slice(&self) -> &[u8] { core::slice::from_raw_parts(self as *const Operation as *const u8, std::mem::size_of::()) } /// Returns a pseudo hash that should uniquely identify the operation. /// This is needed because `Hash` is not implemented for f64s /// /// # Safety /// This function reads from uninitialized memory but the generated value should be fine. pub fn pseudo_hash(&self) -> u64 { let mut s = DefaultHasher::new(); unsafe { self.as_slice() }.hash(&mut s); s.finish() } }