| 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<Vec<PathCommand>, 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<char> { |
| 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<Path, BooleanError> { |
| 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::<Vec<String>>() |
| .join(" ") |
| } |
|
|