openfree's picture
Deploy from GitHub repository
2409829 verified
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(" ")
}