use super::layer_info::LayerData; use super::style::{self, PathStyle, ViewMode}; use crate::intersection::{intersect_quad_bez_path, Quad}; use crate::LayerId; use glam::{DAffine2, DMat2, DVec2}; use kurbo::{Affine, BezPath, Shape as KurboShape}; use serde::{Deserialize, Serialize}; use std::fmt::Write; fn glam_to_kurbo(transform: DAffine2) -> Affine { Affine::new(transform.to_cols_array()) } #[derive(Debug, Clone, PartialEq, Deserialize, Serialize)] pub struct Shape { pub path: BezPath, pub style: style::PathStyle, pub render_index: i32, } impl LayerData for Shape { fn render(&mut self, svg: &mut String, transforms: &mut Vec, view_mode: ViewMode) { let mut path = self.path.clone(); let transform = self.transform(transforms, view_mode); let inverse = transform.inverse(); if !inverse.is_finite() { let _ = write!(svg, ""); return; } path.apply_affine(glam_to_kurbo(transform)); let _ = writeln!(svg, r#""#); let _ = write!(svg, r#""#, path.to_svg(), self.style.render(view_mode)); let _ = svg.write_str(""); } fn bounding_box(&self, transform: glam::DAffine2) -> Option<[DVec2; 2]> { use kurbo::Shape; let mut path = self.path.clone(); if transform.matrix2 == DMat2::ZERO { return None; } path.apply_affine(glam_to_kurbo(transform)); let kurbo::Rect { x0, y0, x1, y1 } = path.bounding_box(); Some([(x0, y0).into(), (x1, y1).into()]) } fn intersects_quad(&self, quad: Quad, path: &mut Vec, intersections: &mut Vec>) { if intersect_quad_bez_path(quad, &self.path, self.style.fill().is_some()) { intersections.push(path.clone()); } } } impl Shape { pub fn transform(&self, transforms: &[DAffine2], mode: ViewMode) -> DAffine2 { let start = match (mode, self.render_index) { (ViewMode::Outline, _) => 0, (_, -1) => 0, (_, x) => (transforms.len() as i32 - x).max(0) as usize, }; transforms.iter().skip(start).cloned().reduce(|a, b| a * b).unwrap_or(DAffine2::IDENTITY) } pub fn from_bez_path(bez_path: BezPath, style: PathStyle, closed: bool) -> Self { Self { path: bez_path, style, render_index: 1, } } pub fn ngon(sides: u8, style: PathStyle) -> Self { use std::f64::consts::{FRAC_PI_2, TAU}; fn unit_rotation(theta: f64) -> DVec2 { DVec2::new(theta.sin(), theta.cos()) } let mut path = kurbo::BezPath::new(); let apothem_offset_angle = TAU / (sides as f64); // Rotate odd sided shapes by 90 degrees let offset = ((sides + 1) % 2) as f64 * FRAC_PI_2; let relative_points = (0..sides).map(|i| apothem_offset_angle * i as f64 + offset).map(unit_rotation); let min = relative_points.clone().reduce(|a, b| a.min(b)).unwrap_or_default(); let transform = DAffine2::from_scale_angle_translation(DVec2::ONE / 2., 0., -min / 2.); let point = |vec: DVec2| kurbo::Point::new(vec.x, vec.y); let mut relative_points = relative_points.map(|p| point(transform.transform_point2(p))); path.move_to(relative_points.next().expect("Tried to create an ngon with 0 sides")); relative_points.for_each(|p| path.line_to(p)); path.close_path(); Self { path, style, render_index: 1 } } pub fn rectangle(style: PathStyle) -> Self { Self { path: kurbo::Rect::new(0., 0., 1., 1.).to_path(0.01), style, render_index: 1, } } pub fn ellipse(style: PathStyle) -> Self { Self { path: kurbo::Ellipse::from_rect(kurbo::Rect::new(0., 0., 1., 1.)).to_path(0.01), style, render_index: 1, } } pub fn line(style: PathStyle) -> Self { Self { path: kurbo::Line::new((0., 0.), (1., 0.)).to_path(0.01), style, render_index: 1, } } pub fn poly_line(points: Vec>, style: PathStyle) -> Self { let mut path = kurbo::BezPath::new(); points .into_iter() .map(|v| v.into()) .map(|v: DVec2| kurbo::Point { x: v.x, y: v.y }) .enumerate() .for_each(|(i, p)| if i == 0 { path.move_to(p) } else { path.line_to(p) }); Self { path, style, render_index: 0 } } }