use crate::cff::charstring::{ArgumentsStack, IsEven};
use crate::cff::outline::Builder;
use crate::cff::{self, CFFError};
use crate::outline::OutlineSink;
pub(crate) struct CharStringParser<'a, B>
where
B: OutlineSink,
{
pub(crate) builder: &'a mut Builder<'a, B>,
pub(crate) x: f32,
pub(crate) y: f32,
pub(crate) has_move_to: bool,
pub(crate) is_first_move_to: bool,
pub(crate) temp: [f32; cff::MAX_OPERANDS],
}
impl<B: OutlineSink> CharStringParser<'_, B> {
pub fn parse_move_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if stack.len() != 2 {
return Err(CFFError::InvalidArgumentsStackLength);
}
if self.is_first_move_to {
self.is_first_move_to = false;
} else {
self.builder.close();
}
self.has_move_to = true;
self.x += stack.at(0);
self.y += stack.at(1);
self.builder.move_to(self.x, self.y);
Ok(())
}
pub fn parse_horizontal_move_to(
&mut self,
stack: &ArgumentsStack<'_, f32>,
) -> Result<(), CFFError> {
if stack.len() != 1 {
return Err(CFFError::InvalidArgumentsStackLength);
}
if self.is_first_move_to {
self.is_first_move_to = false;
} else {
self.builder.close();
}
self.has_move_to = true;
self.x += stack.at(0);
self.builder.move_to(self.x, self.y);
Ok(())
}
pub fn parse_vertical_move_to(
&mut self,
stack: &ArgumentsStack<'_, f32>,
) -> Result<(), CFFError> {
if stack.len() != 1 {
return Err(CFFError::InvalidArgumentsStackLength);
}
if self.is_first_move_to {
self.is_first_move_to = false;
} else {
self.builder.close();
}
self.has_move_to = true;
self.y += stack.at(0);
self.builder.move_to(self.x, self.y);
Ok(())
}
pub fn parse_line_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len().is_odd() {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut i = 0;
while i < stack.len() {
self.x += stack.at(i + 0);
self.y += stack.at(i + 1);
self.builder.line_to(self.x, self.y);
i += 2;
}
Ok(())
}
pub fn parse_horizontal_line_to(
&mut self,
stack: &ArgumentsStack<'_, f32>,
) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.is_empty() {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut i = 0;
while i < stack.len() {
self.x += stack.at(i);
i += 1;
self.builder.line_to(self.x, self.y);
if i == stack.len() {
break;
}
self.y += stack.at(i);
i += 1;
self.builder.line_to(self.x, self.y);
}
Ok(())
}
pub fn parse_vertical_line_to(
&mut self,
stack: &ArgumentsStack<'_, f32>,
) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.is_empty() {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut i = 0;
while i < stack.len() {
self.y += stack.at(i);
i += 1;
self.builder.line_to(self.x, self.y);
if i == stack.len() {
break;
}
self.x += stack.at(i);
i += 1;
self.builder.line_to(self.x, self.y);
}
Ok(())
}
pub fn parse_curve_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() % 6 != 0 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut i = 0;
while i < stack.len() {
let x1 = self.x + stack.at(i + 0);
let y1 = self.y + stack.at(i + 1);
let x2 = x1 + stack.at(i + 2);
let y2 = y1 + stack.at(i + 3);
self.x = x2 + stack.at(i + 4);
self.y = y2 + stack.at(i + 5);
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
i += 6;
}
Ok(())
}
pub fn parse_curve_line(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() < 8 {
return Err(CFFError::InvalidArgumentsStackLength);
}
if (stack.len() - 2) % 6 != 0 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut i = 0;
while i < stack.len() - 2 {
let x1 = self.x + stack.at(i + 0);
let y1 = self.y + stack.at(i + 1);
let x2 = x1 + stack.at(i + 2);
let y2 = y1 + stack.at(i + 3);
self.x = x2 + stack.at(i + 4);
self.y = y2 + stack.at(i + 5);
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
i += 6;
}
self.x += stack.at(i + 0);
self.y += stack.at(i + 1);
self.builder.line_to(self.x, self.y);
Ok(())
}
pub fn parse_line_curve(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() < 8 {
return Err(CFFError::InvalidArgumentsStackLength);
}
if (stack.len() - 6).is_odd() {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut i = 0;
while i < stack.len() - 6 {
self.x += stack.at(i + 0);
self.y += stack.at(i + 1);
self.builder.line_to(self.x, self.y);
i += 2;
}
let x1 = self.x + stack.at(i + 0);
let y1 = self.y + stack.at(i + 1);
let x2 = x1 + stack.at(i + 2);
let y2 = y1 + stack.at(i + 3);
self.x = x2 + stack.at(i + 4);
self.y = y2 + stack.at(i + 5);
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
Ok(())
}
pub fn parse_hh_curve_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
let mut i = 0;
if stack.len().is_odd() {
self.y += stack.at(0);
i += 1;
}
if (stack.len() - i) % 4 != 0 {
return Err(CFFError::InvalidArgumentsStackLength);
}
while i < stack.len() {
let x1 = self.x + stack.at(i + 0);
let y1 = self.y;
let x2 = x1 + stack.at(i + 1);
let y2 = y1 + stack.at(i + 2);
self.x = x2 + stack.at(i + 3);
self.y = y2;
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
i += 4;
}
Ok(())
}
pub fn parse_vv_curve_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
let mut i = 0;
if stack.len().is_odd() {
self.x += stack.at(0);
i += 1;
}
if (stack.len() - i) % 4 != 0 {
return Err(CFFError::InvalidArgumentsStackLength);
}
while i < stack.len() {
let x1 = self.x;
let y1 = self.y + stack.at(i + 0);
let x2 = x1 + stack.at(i + 1);
let y2 = y1 + stack.at(i + 2);
self.x = x2;
self.y = y2 + stack.at(i + 3);
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
i += 4;
}
Ok(())
}
pub fn parse_hv_curve_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() < 4 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut stack = stack.clone_into(self.temp.as_mut_slice());
stack.reverse();
while !stack.is_empty() {
if stack.len() < 4 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let x1 = self.x + stack.pop();
let y1 = self.y;
let x2 = x1 + stack.pop();
let y2 = y1 + stack.pop();
self.y = y2 + stack.pop();
self.x = x2 + if stack.len() == 1 { stack.pop() } else { 0.0 };
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
if stack.is_empty() {
break;
}
if stack.len() < 4 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let x1 = self.x;
let y1 = self.y + stack.pop();
let x2 = x1 + stack.pop();
let y2 = y1 + stack.pop();
self.x = x2 + stack.pop();
self.y = y2 + if stack.len() == 1 { stack.pop() } else { 0.0 };
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
}
debug_assert!(stack.is_empty());
Ok(())
}
pub fn parse_vh_curve_to(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() < 4 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let mut stack = stack.clone_into(self.temp.as_mut_slice());
stack.reverse();
while !stack.is_empty() {
if stack.len() < 4 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let x1 = self.x;
let y1 = self.y + stack.pop();
let x2 = x1 + stack.pop();
let y2 = y1 + stack.pop();
self.x = x2 + stack.pop();
self.y = y2 + if stack.len() == 1 { stack.pop() } else { 0.0 };
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
if stack.is_empty() {
break;
}
if stack.len() < 4 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let x1 = self.x + stack.pop();
let y1 = self.y;
let x2 = x1 + stack.pop();
let y2 = y1 + stack.pop();
self.y = y2 + stack.pop();
self.x = x2 + if stack.len() == 1 { stack.pop() } else { 0.0 };
self.builder.curve_to(x1, y1, x2, y2, self.x, self.y);
}
debug_assert!(stack.is_empty());
Ok(())
}
pub fn parse_flex(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() != 13 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let dx1 = self.x + stack.at(0);
let dy1 = self.y + stack.at(1);
let dx2 = dx1 + stack.at(2);
let dy2 = dy1 + stack.at(3);
let dx3 = dx2 + stack.at(4);
let dy3 = dy2 + stack.at(5);
let dx4 = dx3 + stack.at(6);
let dy4 = dy3 + stack.at(7);
let dx5 = dx4 + stack.at(8);
let dy5 = dy4 + stack.at(9);
self.x = dx5 + stack.at(10);
self.y = dy5 + stack.at(11);
self.builder.curve_to(dx1, dy1, dx2, dy2, dx3, dy3);
self.builder.curve_to(dx4, dy4, dx5, dy5, self.x, self.y);
Ok(())
}
pub fn parse_flex1(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() != 11 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let dx1 = self.x + stack.at(0);
let dy1 = self.y + stack.at(1);
let dx2 = dx1 + stack.at(2);
let dy2 = dy1 + stack.at(3);
let dx3 = dx2 + stack.at(4);
let dy3 = dy2 + stack.at(5);
let dx4 = dx3 + stack.at(6);
let dy4 = dy3 + stack.at(7);
let dx5 = dx4 + stack.at(8);
let dy5 = dy4 + stack.at(9);
if (dx5 - self.x).abs() > (dy5 - self.y).abs() {
self.x = dx5 + stack.at(10);
} else {
self.y = dy5 + stack.at(10);
}
self.builder.curve_to(dx1, dy1, dx2, dy2, dx3, dy3);
self.builder.curve_to(dx4, dy4, dx5, dy5, self.x, self.y);
Ok(())
}
pub fn parse_hflex(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() != 7 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let dx1 = self.x + stack.at(0);
let dy1 = self.y;
let dx2 = dx1 + stack.at(1);
let dy2 = dy1 + stack.at(2);
let dx3 = dx2 + stack.at(3);
let dy3 = dy2;
let dx4 = dx3 + stack.at(4);
let dy4 = dy2;
let dx5 = dx4 + stack.at(5);
let dy5 = self.y;
self.x = dx5 + stack.at(6);
self.builder.curve_to(dx1, dy1, dx2, dy2, dx3, dy3);
self.builder.curve_to(dx4, dy4, dx5, dy5, self.x, self.y);
Ok(())
}
pub fn parse_hflex1(&mut self, stack: &ArgumentsStack<'_, f32>) -> Result<(), CFFError> {
if !self.has_move_to {
return Err(CFFError::MissingMoveTo);
}
if stack.len() != 9 {
return Err(CFFError::InvalidArgumentsStackLength);
}
let dx1 = self.x + stack.at(0);
let dy1 = self.y + stack.at(1);
let dx2 = dx1 + stack.at(2);
let dy2 = dy1 + stack.at(3);
let dx3 = dx2 + stack.at(4);
let dy3 = dy2;
let dx4 = dx3 + stack.at(5);
let dy4 = dy2;
let dx5 = dx4 + stack.at(6);
let dy5 = dy4 + stack.at(7);
self.x = dx5 + stack.at(8);
self.builder.curve_to(dx1, dy1, dx2, dy2, dx3, dy3);
self.builder.curve_to(dx4, dy4, dx5, dy5, self.x, self.y);
Ok(())
}
}