finish glyph parsing
This commit is contained in:
parent
c994f3ec02
commit
7cbb264410
|
@ -51,12 +51,13 @@ impl Font {
|
||||||
pub fn new(file: File) -> bincode::Result<Self> {
|
pub fn new(file: File) -> bincode::Result<Self> {
|
||||||
let mut buf_reader = BufReader::new(file.try_clone()?);
|
let mut buf_reader = BufReader::new(file.try_clone()?);
|
||||||
let table_directory = TableDirectory::new(&mut buf_reader)?;
|
let table_directory = TableDirectory::new(&mut buf_reader)?;
|
||||||
|
debug!("{table_directory:#?}");
|
||||||
|
|
||||||
let font_header: FontHeader = table_directory.get_table(&file)?;
|
let font_header: FontHeader = table_directory.get_table(&file)?;
|
||||||
debug!("{font_header:#?}");
|
debug!("{font_header:#?}");
|
||||||
|
|
||||||
let character_map = table_directory.get_table(&file)?;
|
let character_map = table_directory.get_table(&file)?;
|
||||||
debug!("{character_map:#?}");
|
// debug!("{character_map:#?}");
|
||||||
|
|
||||||
let maximum_profile: MaximumProfile = table_directory.get_table(&file)?;
|
let maximum_profile: MaximumProfile = table_directory.get_table(&file)?;
|
||||||
debug!("{maximum_profile:#?}");
|
debug!("{maximum_profile:#?}");
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
use std::{collections::HashMap, fs::File, io::{BufReader, Read, Seek, SeekFrom}, iter::zip, slice::Iter};
|
use std::{collections::HashMap, fs::File, io::{BufReader, Read, Seek, SeekFrom}, iter::{zip, Peekable}, slice::Iter};
|
||||||
|
|
||||||
use bincode::{ErrorKind, Result};
|
use bincode::{ErrorKind, Result};
|
||||||
use bitfield::bitfield;
|
use bitfield::bitfield;
|
||||||
use log::{debug, info};
|
use log::{debug, info, warn};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
|
|
||||||
use super::{deserialize, deserialize_vec, index_to_location::IndexToLocation, table_directory::{FontTable, TableDirectoryRecord}};
|
use super::{deserialize, deserialize_vec, index_to_location::IndexToLocation, table_directory::{FontTable, TableDirectoryRecord}};
|
||||||
|
@ -56,35 +56,37 @@ impl<'a> Iterator for GlyphFlags<'a> {
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
if self.current_item >= self.flags.len() {
|
if self.current_item >= self.flags.len() {
|
||||||
|
debug!("no more logical flags");
|
||||||
return None;
|
return None;
|
||||||
}
|
}
|
||||||
|
|
||||||
let current = &self.flags[self.current_item];
|
let current = &self.flags[self.current_item];
|
||||||
|
|
||||||
if self.repetitions < current.repeat as u16 {
|
if self.repetitions <= current.repeat as u16 {
|
||||||
|
if self.repetitions > 0 {
|
||||||
|
debug!("flag #{} repeated {} times", self.current_item, self.repetitions);
|
||||||
|
} else {
|
||||||
|
debug!("yield flag #{}", self.current_item);
|
||||||
|
}
|
||||||
|
|
||||||
self.repetitions += 1;
|
self.repetitions += 1;
|
||||||
return Some(current.flags);
|
return Some(current.flags);
|
||||||
}
|
}
|
||||||
|
|
||||||
self.current_item += 1;
|
|
||||||
if self.current_item >= self.flags.len() {
|
|
||||||
return None;
|
|
||||||
}
|
|
||||||
|
|
||||||
let current = &self.flags[self.current_item];
|
|
||||||
self.repetitions = 0;
|
self.repetitions = 0;
|
||||||
|
self.current_item += 1;
|
||||||
Some(current.flags)
|
self.next()
|
||||||
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug, Copy, Clone)]
|
||||||
pub struct GlyphPoint {
|
pub struct GlyphPoint {
|
||||||
pub x: i32,
|
pub x: i32,
|
||||||
pub y: i32,
|
pub y: i32,
|
||||||
|
|
||||||
pub on_curve: bool
|
pub on_curve: bool,
|
||||||
|
pub is_virtual: bool,
|
||||||
|
pub is_endpoint: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
|
@ -100,7 +102,7 @@ impl SimpleGlyph {
|
||||||
let mut flags: Vec<GlyphFlag> = vec![];
|
let mut flags: Vec<GlyphFlag> = vec![];
|
||||||
|
|
||||||
let mut points_processed: usize = 0;
|
let mut points_processed: usize = 0;
|
||||||
while points_processed < num_points {
|
while points_processed <= num_points {
|
||||||
let flag: GlyphFlagData = deserialize(reader.by_ref())?;
|
let flag: GlyphFlagData = deserialize(reader.by_ref())?;
|
||||||
|
|
||||||
let repetitions: u8 = if flag.repeat() { deserialize(reader.by_ref())? } else { 0 };
|
let repetitions: u8 = if flag.repeat() { deserialize(reader.by_ref())? } else { 0 };
|
||||||
|
@ -118,28 +120,39 @@ impl SimpleGlyph {
|
||||||
}
|
}
|
||||||
|
|
||||||
fn read_coordinates<R: Read + Seek>(reader: &mut R, flags: GlyphFlags, is_y: bool) -> Result<Vec<(i32, bool)>> {
|
fn read_coordinates<R: Read + Seek>(reader: &mut R, flags: GlyphFlags, is_y: bool) -> Result<Vec<(i32, bool)>> {
|
||||||
|
debug!("reading batch of {} coordinates at {}", if is_y { "y" } else { "x" }, reader.stream_position()?);
|
||||||
let mut coordinates: Vec<(i32, bool)> = vec![];
|
let mut coordinates: Vec<(i32, bool)> = vec![];
|
||||||
let mut current_coordinate = 0i32;
|
let mut current_coordinate = 0i32;
|
||||||
|
|
||||||
for flag in flags {
|
for flag in flags {
|
||||||
|
debug!("reading next coordinate");
|
||||||
|
|
||||||
let short_vec = if is_y { flag.y_short() } else { flag.x_short() };
|
let short_vec = if is_y { flag.y_short() } else { flag.x_short() };
|
||||||
let same_or_positive = if is_y { flag.y_same_or_positive() } else { flag.x_same_or_positive() };
|
let same_or_positive = if is_y { flag.y_same_or_positive() } else { flag.x_same_or_positive() };
|
||||||
|
|
||||||
|
debug!("short vec: {short_vec}, same or positive: {same_or_positive}");
|
||||||
|
|
||||||
if short_vec {
|
if short_vec {
|
||||||
let sign: i32 = if same_or_positive { 1 } else { -1 };
|
let sign: i32 = if same_or_positive { 1 } else { -1 };
|
||||||
let delta = sign * (deserialize::<_, u8>(reader.by_ref())? as i32);
|
let delta = sign * (deserialize::<_, u8>(reader.by_ref())? as i32);
|
||||||
|
|
||||||
|
debug!("delta = (u8){delta}");
|
||||||
|
|
||||||
current_coordinate += delta;
|
current_coordinate += delta;
|
||||||
coordinates.push((current_coordinate, flag.on_curve()));
|
coordinates.push((current_coordinate, flag.on_curve()));
|
||||||
} else {
|
} else {
|
||||||
if same_or_positive {
|
if same_or_positive {
|
||||||
|
debug!("coordinate is repeated");
|
||||||
coordinates.push((current_coordinate, flag.on_curve()));
|
coordinates.push((current_coordinate, flag.on_curve()));
|
||||||
} else {
|
} else {
|
||||||
let delta: i16 = deserialize(reader.by_ref())?;
|
let delta: i16 = deserialize(reader.by_ref())?;
|
||||||
|
debug!("delta = (i16){delta}");
|
||||||
current_coordinate += delta as i32;
|
current_coordinate += delta as i32;
|
||||||
coordinates.push((current_coordinate, flag.on_curve()));
|
coordinates.push((current_coordinate, flag.on_curve()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
debug!("current coordinate is now {current_coordinate}");
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(coordinates)
|
Ok(coordinates)
|
||||||
|
@ -158,11 +171,24 @@ impl SimpleGlyph {
|
||||||
let x_coords = SimpleGlyph::read_coordinates(reader, GlyphFlags::new(&flags), false)?;
|
let x_coords = SimpleGlyph::read_coordinates(reader, GlyphFlags::new(&flags), false)?;
|
||||||
let y_coords = SimpleGlyph::read_coordinates(reader, GlyphFlags::new(&flags), true)?;
|
let y_coords = SimpleGlyph::read_coordinates(reader, GlyphFlags::new(&flags), true)?;
|
||||||
|
|
||||||
|
if x_coords.len() != *end_points.last().unwrap() as usize + 1 {
|
||||||
|
warn!("expected {} x coordinates, got {} instead", end_points.last().unwrap() + 1, x_coords.len());
|
||||||
|
}
|
||||||
|
|
||||||
|
if y_coords.len() != *end_points.last().unwrap() as usize + 1 {
|
||||||
|
warn!("expected {} y coordinates, got {} instead", end_points.last().unwrap() + 1, y_coords.len());
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut current_point: u16 = 0;
|
||||||
let coordinates = zip(x_coords.iter().cloned(), y_coords.iter().cloned())
|
let coordinates = zip(x_coords.iter().cloned(), y_coords.iter().cloned())
|
||||||
.map(|(x, y)| {
|
.map(|(x, y)| {
|
||||||
GlyphPoint {
|
let point = GlyphPoint {
|
||||||
x: x.0, y: y.0, on_curve: x.1
|
x: x.0, y: y.0, on_curve: x.1, is_endpoint: end_points.contains(¤t_point), is_virtual: false
|
||||||
}
|
};
|
||||||
|
|
||||||
|
current_point += 1;
|
||||||
|
|
||||||
|
point
|
||||||
});
|
});
|
||||||
|
|
||||||
// coordinates.for_each(|point| debug!("({}, {}, {})", point.x, point.y, if point.on_curve { "on" } else { "off" }));
|
// coordinates.for_each(|point| debug!("({}, {}, {})", point.x, point.y, if point.on_curve { "on" } else { "off" }));
|
||||||
|
@ -173,21 +199,44 @@ impl SimpleGlyph {
|
||||||
pub fn iter<'a>(&'a self) -> SimpleGlyphIter<'a> {
|
pub fn iter<'a>(&'a self) -> SimpleGlyphIter<'a> {
|
||||||
SimpleGlyphIter {
|
SimpleGlyphIter {
|
||||||
glyph: self,
|
glyph: self,
|
||||||
iter: self.points.iter()
|
last_point: None,
|
||||||
|
iter: self.points.iter().peekable()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct SimpleGlyphIter<'a> {
|
struct SimpleGlyphIter<'a> {
|
||||||
glyph: &'a SimpleGlyph,
|
glyph: &'a SimpleGlyph,
|
||||||
iter: Iter<'a, GlyphPoint>
|
last_point: Option<GlyphPoint>,
|
||||||
|
iter: Peekable<Iter<'a, GlyphPoint>>
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<'a> Iterator for SimpleGlyphIter<'a> {
|
impl<'a> Iterator for SimpleGlyphIter<'a> {
|
||||||
type Item = &'a GlyphPoint;
|
type Item = GlyphPoint;
|
||||||
|
|
||||||
fn next(&mut self) -> Option<Self::Item> {
|
fn next(&mut self) -> Option<Self::Item> {
|
||||||
self.iter.next()
|
if let Some(last_point) = self.last_point {
|
||||||
|
let Some(&next_point) = self.iter.peek() else {
|
||||||
|
return None
|
||||||
|
};
|
||||||
|
|
||||||
|
if !last_point.on_curve && !next_point.on_curve {
|
||||||
|
// Interpolate a point
|
||||||
|
let virtual_point = GlyphPoint {
|
||||||
|
x: last_point.x + (next_point.x - last_point.x) / 2,
|
||||||
|
y: last_point.y + (next_point.y - last_point.y) / 2,
|
||||||
|
is_endpoint: false,
|
||||||
|
is_virtual: true,
|
||||||
|
on_curve: true
|
||||||
|
};
|
||||||
|
|
||||||
|
self.last_point = Some(virtual_point);
|
||||||
|
return self.last_point;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
self.last_point = self.iter.next().copied();
|
||||||
|
self.last_point
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -204,7 +253,7 @@ pub struct Glyph {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Glyph {
|
impl Glyph {
|
||||||
pub fn iter<'a>(&'a self) -> Box<dyn Iterator<Item = &'a GlyphPoint> + 'a> {
|
pub fn iter<'a>(&'a self) -> Box<dyn Iterator<Item = GlyphPoint> + 'a> {
|
||||||
Box::new(
|
Box::new(
|
||||||
match &self.data {
|
match &self.data {
|
||||||
GlyphData::Simple(glyph) => glyph.iter(),
|
GlyphData::Simple(glyph) => glyph.iter(),
|
||||||
|
@ -236,14 +285,28 @@ impl GlyphTable {
|
||||||
let header: GlyphHeader = deserialize(self.data_source.get_ref())?;
|
let header: GlyphHeader = deserialize(self.data_source.get_ref())?;
|
||||||
debug!("glyph header: {header:#?}");
|
debug!("glyph header: {header:#?}");
|
||||||
|
|
||||||
if header.num_countours >= 0 {
|
let result = if header.num_countours >= 0 {
|
||||||
Ok(Glyph {
|
Ok(Glyph {
|
||||||
data: GlyphData::Simple(SimpleGlyph::new(&header, self.data_source.by_ref())?),
|
data: GlyphData::Simple(SimpleGlyph::new(&header, self.data_source.by_ref())?),
|
||||||
header
|
header
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
todo!();
|
todo!();
|
||||||
|
};
|
||||||
|
|
||||||
|
// sanity check
|
||||||
|
let stream_pos = self.data_source.stream_position()?;
|
||||||
|
let Some(next_offset) = location.get_offset(glyph_id + 1) else {
|
||||||
|
warn!("Failed to perform sanity check after glyph parsing");
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
let remaining_bytes = stream_pos as i64 - (self.offset + next_offset) as i64;
|
||||||
|
if remaining_bytes != 0 {
|
||||||
|
warn!("sanity check failed, there are still {remaining_bytes} bytes remaining in this glyph entry (read head: {}, table off: {}, location: [{offset}, {next_offset}])", stream_pos, self.offset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
result
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -11,8 +11,8 @@ pub trait Visitor {
|
||||||
|
|
||||||
fn write(&mut self, glyph: &Glyph) {
|
fn write(&mut self, glyph: &Glyph) {
|
||||||
self.write_prefix(&glyph.header);
|
self.write_prefix(&glyph.header);
|
||||||
|
|
||||||
glyph.iter().for_each(|point| self.write_point(point));
|
glyph.iter().for_each(|point| self.write_point(&point));
|
||||||
|
|
||||||
self.write_suffix();
|
self.write_suffix();
|
||||||
}
|
}
|
||||||
|
|
|
@ -6,6 +6,9 @@ use super::Visitor;
|
||||||
|
|
||||||
pub struct SvgWriter {
|
pub struct SvgWriter {
|
||||||
first_point: bool,
|
first_point: bool,
|
||||||
|
last_on_curve: bool,
|
||||||
|
final_point: Option<(i32, i32)>,
|
||||||
|
final_control: Option<(i32, i32)>,
|
||||||
writer: BufWriter<File>
|
writer: BufWriter<File>
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -13,6 +16,9 @@ impl SvgWriter {
|
||||||
pub fn new(file: File) -> Self {
|
pub fn new(file: File) -> Self {
|
||||||
SvgWriter {
|
SvgWriter {
|
||||||
first_point: true,
|
first_point: true,
|
||||||
|
last_on_curve: false,
|
||||||
|
final_control: None,
|
||||||
|
final_point: None,
|
||||||
writer: BufWriter::new(file)
|
writer: BufWriter::new(file)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -23,26 +29,56 @@ impl Visitor for SvgWriter {
|
||||||
write!(
|
write!(
|
||||||
self.writer,
|
self.writer,
|
||||||
"<svg width=\"1000\" height=\"1000\" viewBox=\"{} {} {} {}\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"",
|
"<svg width=\"1000\" height=\"1000\" viewBox=\"{} {} {} {}\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"",
|
||||||
header.xmin, header.ymin,
|
header.xmin, -header.ymax,
|
||||||
header.xmax - header.xmin,
|
header.xmax - header.xmin,
|
||||||
header.ymax - header.ymin
|
header.ymax - header.ymin
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_point(&mut self, point: &GlyphPoint) {
|
fn write_point(&mut self, point: &GlyphPoint) {
|
||||||
if !point.on_curve {
|
if self.final_point.is_none() && point.on_curve {
|
||||||
return
|
self.final_point = Some((point.x, point.y));
|
||||||
}
|
}
|
||||||
|
|
||||||
let prefix = if self.first_point { "M" } else { "L" };
|
if self.first_point && self.final_control.is_none() && !point.on_curve {
|
||||||
|
self.final_control = Some((point.x, point.y));
|
||||||
|
}
|
||||||
|
|
||||||
|
if self.first_point {
|
||||||
|
if !point.on_curve {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
write!(self.writer, "M{} {} ", point.x, -point.y);
|
||||||
|
self.first_point = false;
|
||||||
|
self.last_on_curve = true;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
let prefix = if point.on_curve { if self.last_on_curve { "L" } else { "" } } else { "Q" };
|
||||||
|
self.last_on_curve = point.on_curve;
|
||||||
|
|
||||||
write!(
|
write!(
|
||||||
self.writer,
|
self.writer,
|
||||||
"{prefix}{} {} ",
|
"{prefix}{} {} ",
|
||||||
point.x, point.y
|
point.x, -point.y
|
||||||
);
|
);
|
||||||
|
|
||||||
self.first_point = false;
|
if point.is_endpoint {
|
||||||
|
if let Some(final_control) = self.final_control {
|
||||||
|
if let Some(final_point) = self.final_point {
|
||||||
|
write!(self.writer, "Q{} {} {} {} ", final_control.0, -final_control.1, final_point.0, -final_point.1);
|
||||||
|
} else {
|
||||||
|
write!(self.writer, "Z ");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
write!(self.writer, "Z ");
|
||||||
|
}
|
||||||
|
|
||||||
|
self.final_point = None;
|
||||||
|
self.final_control = None;
|
||||||
|
}
|
||||||
|
self.first_point = point.is_endpoint;
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_suffix(&mut self) {
|
fn write_suffix(&mut self) {
|
||||||
|
|
Loading…
Reference in a new issue