use super::layer_info::LayerData; use super::style::{PathStyle, RenderData, ViewMode}; use crate::intersection::{intersect_quad_bez_path, Quad}; use crate::LayerId; pub use font_cache::{Font, FontCache}; use graphene_std::vector::subpath::Subpath; use glam::{DAffine2, DMat2, DVec2}; use rustybuzz::Face; use serde::{Deserialize, Serialize}; use std::fmt::Write; mod font_cache; mod to_path; /// A line, or multiple lines, of text drawn in the document. /// Like [ShapeLayers](super::shape_layer::ShapeLayer), [TextLayer] are rendered as /// [``s](https://developer.mozilla.org/en-US/docs/Web/SVG/Element/path). #[derive(Debug, Clone, PartialEq, Deserialize, Serialize)] pub struct TextLayer { /// The string of text, encompassing one or multiple lines. pub text: String, /// Fill color and stroke used to render the text. pub path_style: PathStyle, /// Font size in pixels. pub size: f64, pub line_width: Option, pub font: Font, #[serde(skip)] pub editable: bool, #[serde(skip)] pub cached_path: Option, } impl LayerData for TextLayer { fn render(&mut self, svg: &mut String, svg_defs: &mut String, transforms: &mut Vec, render_data: RenderData) -> bool { let transform = self.transform(transforms, render_data.view_mode); let inverse = transform.inverse(); if !inverse.is_finite() { let _ = write!(svg, ""); return false; } let _ = writeln!(svg, r#""#); if self.editable { let font = render_data.font_cache.resolve_font(&self.font); if let Some(url) = font.and_then(|font| render_data.font_cache.get_preview_url(font)) { let _ = write!(svg, r#""#, url); } let _ = write!( svg, r#""#, transform .to_cols_array() .iter() .enumerate() .map(|(i, entry)| { entry.to_string() + if i == 5 { "" } else { "," } }) .collect::(), font.map(|_| r#" style="font-family: local-font;""#).unwrap_or_default() ); } else { let buzz_face = self.load_face(render_data.font_cache); let mut path = self.to_subpath(buzz_face); let bounds = path.bounding_box().unwrap_or_default(); path.apply_affine(transform); let transformed_bounds = path.bounding_box().unwrap_or_default(); let _ = write!( svg, r#""#, path.to_svg(), self.path_style.render(render_data.view_mode, svg_defs, transform, bounds, transformed_bounds) ); } let _ = svg.write_str(""); false } fn bounding_box(&self, transform: glam::DAffine2, font_cache: &FontCache) -> Option<[DVec2; 2]> { let buzz_face = Some(self.load_face(font_cache)?); if transform.matrix2 == DMat2::ZERO { return None; } Some((transform * self.bounding_box(&self.text, buzz_face)).bounding_box()) } fn intersects_quad(&self, quad: Quad, path: &mut Vec, intersections: &mut Vec>, font_cache: &FontCache) { let buzz_face = self.load_face(font_cache); if intersect_quad_bez_path(quad, &self.bounding_box(&self.text, buzz_face).path(), true) { intersections.push(path.clone()); } } } impl TextLayer { pub fn load_face<'a>(&self, font_cache: &'a FontCache) -> Option> { font_cache.get(&self.font).map(|data| rustybuzz::Face::from_slice(data, 0).expect("Loading font failed")) } pub fn transform(&self, transforms: &[DAffine2], mode: ViewMode) -> DAffine2 { let start = match mode { ViewMode::Outline => 0, _ => (transforms.len() as i32 - 1).max(0) as usize, }; transforms.iter().skip(start).cloned().reduce(|a, b| a * b).unwrap_or(DAffine2::IDENTITY) } pub fn new(text: String, style: PathStyle, size: f64, font: Font, font_cache: &FontCache) -> Self { let mut new = Self { text, path_style: style, size, line_width: None, font, editable: false, cached_path: None, }; new.cached_path = Some(new.generate_path(new.load_face(font_cache))); new } /// Converts to a [Subpath], populating the cache if necessary. #[inline] pub fn to_subpath(&mut self, buzz_face: Option) -> Subpath { if self.cached_path.as_ref().filter(|subpath| !subpath.manipulator_groups().is_empty()).is_none() { let path = self.generate_path(buzz_face); self.cached_path = Some(path.clone()); return path; } self.cached_path.clone().unwrap() } /// Converts to a [Subpath], without populating the cache. #[inline] pub fn to_subpath_nonmut(&self, font_cache: &FontCache) -> Subpath { let buzz_face = self.load_face(font_cache); self.cached_path .clone() .filter(|subpath| !subpath.manipulator_groups().is_empty()) .unwrap_or_else(|| self.generate_path(buzz_face)) } #[inline] pub fn generate_path(&self, buzz_face: Option) -> Subpath { to_path::to_path(&self.text, buzz_face, self.size, self.line_width) } #[inline] pub fn bounding_box(&self, text: &str, buzz_face: Option) -> Quad { let far = to_path::bounding_box(text, buzz_face, self.size, self.line_width); Quad::from_box([DVec2::ZERO, far]) } pub fn update_text(&mut self, text: String, font_cache: &FontCache) { let buzz_face = self.load_face(font_cache); self.text = text; self.cached_path = Some(self.generate_path(buzz_face)); } }