use crate::BooleanError; use crate::path::{Path, path_from_commands, path_to_commands}; use crate::path_command::{AbsolutePathCommand, PathCommand, RelativePathCommand}; use glam::DVec2; use regex::Regex; pub fn commands_from_path_data(d: &str) -> Result, BooleanError> { let re_float = Regex::new(r"^\s*,?\s*(-?\d*(?:\d\.|\.\d|\d)\d*(?:[eE][+\-]?\d+)?)").unwrap(); let re_cmd = Regex::new(r"^\s*([MLCSQTAZHVmlhvcsqtaz])").unwrap(); let re_bool = Regex::new(r"^\s*,?\s*([01])").unwrap(); let mut i = 0; let mut last_cmd = 'M'; let mut commands = Vec::new(); let get_cmd = |i: &mut usize, last_cmd: char| -> Option { if *i >= d.len() - 1.min(d.len()) { return None; } if let Some(cap) = re_cmd.captures(&d[*i..]) { *i += cap[0].len(); Some(cap[1].chars().next().unwrap()) } else { match last_cmd { 'M' => Some('L'), 'm' => Some('l'), 'z' | 'Z' => None, _ => Some(last_cmd), } } }; let get_float = |i: &mut usize| -> f64 { if let Some(cap) = re_float.captures(&d[*i..]) { *i += cap[0].len(); cap[1].parse().unwrap() } else { panic!("Invalid path data. Expected a number at index {}, got {}", i, &d[*i..]); } }; let get_bool = |i: &mut usize| -> bool { if let Some(cap) = re_bool.captures(&d[*i..]) { *i += cap[0].len(); &cap[1] == "1" } else { panic!("Invalid path data. Expected a flag at index {}", i); } }; while let Some(cmd) = get_cmd(&mut i, last_cmd) { last_cmd = cmd; match cmd { 'M' => commands.push(PathCommand::Absolute(AbsolutePathCommand::M(DVec2::new(get_float(&mut i), get_float(&mut i))))), 'L' => commands.push(PathCommand::Absolute(AbsolutePathCommand::L(DVec2::new(get_float(&mut i), get_float(&mut i))))), 'C' => commands.push(PathCommand::Absolute(AbsolutePathCommand::C( DVec2::new(get_float(&mut i), get_float(&mut i)), DVec2::new(get_float(&mut i), get_float(&mut i)), DVec2::new(get_float(&mut i), get_float(&mut i)), ))), 'S' => commands.push(PathCommand::Absolute(AbsolutePathCommand::S( DVec2::new(get_float(&mut i), get_float(&mut i)), DVec2::new(get_float(&mut i), get_float(&mut i)), ))), 'Q' => commands.push(PathCommand::Absolute(AbsolutePathCommand::Q( DVec2::new(get_float(&mut i), get_float(&mut i)), DVec2::new(get_float(&mut i), get_float(&mut i)), ))), 'T' => commands.push(PathCommand::Absolute(AbsolutePathCommand::T(DVec2::new(get_float(&mut i), get_float(&mut i))))), 'A' => commands.push(PathCommand::Absolute(AbsolutePathCommand::A( get_float(&mut i), get_float(&mut i), get_float(&mut i), get_bool(&mut i), get_bool(&mut i), DVec2::new(get_float(&mut i), get_float(&mut i)), ))), 'Z' | 'z' => commands.push(PathCommand::Absolute(AbsolutePathCommand::Z)), 'H' => commands.push(PathCommand::Absolute(AbsolutePathCommand::H(get_float(&mut i)))), 'V' => commands.push(PathCommand::Absolute(AbsolutePathCommand::V(get_float(&mut i)))), 'm' => commands.push(PathCommand::Relative(RelativePathCommand::M(get_float(&mut i), get_float(&mut i)))), 'l' => commands.push(PathCommand::Relative(RelativePathCommand::L(get_float(&mut i), get_float(&mut i)))), 'h' => commands.push(PathCommand::Relative(RelativePathCommand::H(get_float(&mut i)))), 'v' => commands.push(PathCommand::Relative(RelativePathCommand::V(get_float(&mut i)))), 'c' => commands.push(PathCommand::Relative(RelativePathCommand::C( get_float(&mut i), get_float(&mut i), get_float(&mut i), get_float(&mut i), get_float(&mut i), get_float(&mut i), ))), 's' => commands.push(PathCommand::Relative(RelativePathCommand::S( get_float(&mut i), get_float(&mut i), get_float(&mut i), get_float(&mut i), ))), 'q' => commands.push(PathCommand::Relative(RelativePathCommand::Q( get_float(&mut i), get_float(&mut i), get_float(&mut i), get_float(&mut i), ))), 't' => commands.push(PathCommand::Relative(RelativePathCommand::T(get_float(&mut i), get_float(&mut i)))), 'a' => commands.push(PathCommand::Relative(RelativePathCommand::A( get_float(&mut i), get_float(&mut i), get_float(&mut i), get_bool(&mut i), get_bool(&mut i), get_float(&mut i), get_float(&mut i), ))), _ => return Err(BooleanError::InvalidPathCommand(cmd)), } } Ok(commands) } pub fn path_from_path_data(d: &str) -> Result { Ok(path_from_commands(commands_from_path_data(d)?).collect()) } pub fn path_to_path_data(path: &Path, eps: f64) -> String { path_to_commands(path.iter(), eps) .map(|cmd| match cmd { PathCommand::Absolute(abs_cmd) => match abs_cmd { AbsolutePathCommand::H(dx) => format!("H {:.12}", dx), AbsolutePathCommand::V(dy) => format!("V {:.12}", dy), AbsolutePathCommand::M(p) => format!("M {:.12},{:.12}", p.x, p.y), AbsolutePathCommand::L(p) => format!("L {:.12},{:.12}", p.x, p.y), AbsolutePathCommand::C(p1, p2, p3) => format!("C {:.12},{:.12} {:.12},{:.12} {:.12},{:.12}", p1.x, p1.y, p2.x, p2.y, p3.x, p3.y), AbsolutePathCommand::S(p1, p2) => { format!("S {:.12},{:.12} {:.12},{:.12}", p1.x, p1.y, p2.x, p2.y) } AbsolutePathCommand::Q(p1, p2) => { format!("Q {:.12},{:.12} {:.12},{:.12}", p1.x, p1.y, p2.x, p2.y) } AbsolutePathCommand::T(p) => format!("T {:.12},{:.12}", p.x, p.y), AbsolutePathCommand::A(rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, p) => { format!("A {:.12} {:.12} {:.12} {} {} {:.12},{:.12}", rx, ry, x_axis_rotation, large_arc_flag as u8, sweep_flag as u8, p.x, p.y) } AbsolutePathCommand::Z => "Z".to_string(), }, PathCommand::Relative(rel_cmd) => match rel_cmd { RelativePathCommand::M(dx, dy) => format!("m {:.12},{:.12}", dx, dy), RelativePathCommand::L(dx, dy) => format!("l {:.12},{:.12}", dx, dy), RelativePathCommand::H(dx) => format!("h {:.12}", dx), RelativePathCommand::V(dy) => format!("v {:.12}", dy), RelativePathCommand::C(dx1, dy1, dx2, dy2, dx, dy) => format!("c{:.12},{:.12} {:.12},{:.12} {:.12},{:.12}", dx1, dy1, dx2, dy2, dx, dy), RelativePathCommand::S(dx2, dy2, dx, dy) => { format!("s {:.12},{:.12} {:.12},{:.12}", dx2, dy2, dx, dy) } RelativePathCommand::Q(dx1, dy1, dx, dy) => { format!("q {:.12},{:.12} {:.12},{:.12}", dx1, dy1, dx, dy) } RelativePathCommand::T(dx, dy) => format!("t{:.12},{:.12}", dx, dy), RelativePathCommand::A(rx, ry, x_axis_rotation, large_arc_flag, sweep_flag, dx, dy) => { format!("a {:.12} {:.12} {:.12} {} {} {:.12},{:.12}", rx, ry, x_axis_rotation, large_arc_flag as u8, sweep_flag as u8, dx, dy) } }, }) .collect::>() .join(" ") }