add svg visitor

This commit is contained in:
lauchmelder 2025-03-03 13:18:52 +01:00
parent 6f2cd7da86
commit c994f3ec02
7 changed files with 141 additions and 19891 deletions

2
.gitignore vendored
View file

@ -2,3 +2,5 @@
*.ttf *.ttf
*.otf *.otf
*.svg
*.dump

19873
font.dump

File diff suppressed because it is too large Load diff

View file

@ -17,6 +17,8 @@ use log::{debug, info};
use maximum_profile::MaximumProfile; use maximum_profile::MaximumProfile;
use table_directory::TableDirectory; use table_directory::TableDirectory;
pub use glyph_data::{GlyphData, GlyphHeader, GlyphPoint, Glyph, SimpleGlyph};
fn deserialize<R: Read, T: serde::de::DeserializeOwned>(reader: R) -> bincode::Result<T> { fn deserialize<R: Read, T: serde::de::DeserializeOwned>(reader: R) -> bincode::Result<T> {
bincode::options().with_big_endian().with_fixint_encoding().deserialize_from::<R, T>(reader) bincode::options().with_big_endian().with_fixint_encoding().deserialize_from::<R, T>(reader)
} }
@ -76,7 +78,7 @@ impl Font {
}) })
} }
pub fn get_data_for(&mut self, glyph: char) -> Result<(), Box<dyn Error>> { pub fn get_data_for(&mut self, glyph: char) -> Result<Glyph, Box<dyn Error>> {
let Some(glyph_id) = self.character_map.get_char(glyph) else { let Some(glyph_id) = self.character_map.get_char(glyph) else {
return Err("Failed to get glyph id for char".into()); return Err("Failed to get glyph id for char".into());
}; };
@ -85,7 +87,7 @@ impl Font {
let glyph = self.glyph_table.get_glyph(glyph_id, &mut self.index_to_location)?; let glyph = self.glyph_table.get_glyph(glyph_id, &mut self.index_to_location)?;
Ok(()) Ok(glyph)
} }
} }

View file

@ -1,4 +1,4 @@
use std::{collections::HashMap, fs::File, io::{BufReader, Read, Seek, SeekFrom}, iter::zip}; use std::{collections::HashMap, fs::File, io::{BufReader, Read, Seek, SeekFrom}, iter::zip, slice::Iter};
use bincode::{ErrorKind, Result}; use bincode::{ErrorKind, Result};
use bitfield::bitfield; use bitfield::bitfield;
@ -12,10 +12,10 @@ use super::{deserialize, deserialize_vec, index_to_location::IndexToLocation, ta
pub struct GlyphHeader { pub struct GlyphHeader {
num_countours: i16, num_countours: i16,
xmin: i16, pub xmin: i16,
ymin: i16, pub ymin: i16,
xmax: i16, pub xmax: i16,
ymax: i16 pub ymax: i16
} }
bitfield! { bitfield! {
@ -81,15 +81,14 @@ impl<'a> Iterator for GlyphFlags<'a> {
#[derive(Debug)] #[derive(Debug)]
pub struct GlyphPoint { pub struct GlyphPoint {
x: i32, pub x: i32,
y: i32, pub y: i32,
on_curve: bool pub on_curve: bool
} }
#[derive(Debug)] #[derive(Debug)]
pub struct SimpleGlyph { pub struct SimpleGlyph {
header: GlyphHeader,
instructions: Vec<u8>, instructions: Vec<u8>,
points: Vec<GlyphPoint> points: Vec<GlyphPoint>
@ -146,7 +145,7 @@ impl SimpleGlyph {
Ok(coordinates) Ok(coordinates)
} }
fn new<R: Read + Seek>(header: GlyphHeader, reader: &mut R) -> Result<SimpleGlyph> { fn new<R: Read + Seek>(header: &GlyphHeader, reader: &mut R) -> Result<SimpleGlyph> {
let end_points: Vec<u16> = deserialize_vec(reader, header.num_countours as usize)?; let end_points: Vec<u16> = deserialize_vec(reader, header.num_countours as usize)?;
debug!("glyph end points: {end_points:#?}"); debug!("glyph end points: {end_points:#?}");
@ -168,7 +167,27 @@ impl SimpleGlyph {
// 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" }));
Ok(SimpleGlyph { header, instructions, points: coordinates.collect() }) Ok(SimpleGlyph { instructions, points: coordinates.collect() })
}
pub fn iter<'a>(&'a self) -> SimpleGlyphIter<'a> {
SimpleGlyphIter {
glyph: self,
iter: self.points.iter()
}
}
}
struct SimpleGlyphIter<'a> {
glyph: &'a SimpleGlyph,
iter: Iter<'a, GlyphPoint>
}
impl<'a> Iterator for SimpleGlyphIter<'a> {
type Item = &'a GlyphPoint;
fn next(&mut self) -> Option<Self::Item> {
self.iter.next()
} }
} }
@ -178,16 +197,33 @@ pub enum GlyphData {
Composite Composite
} }
#[derive(Debug)]
pub struct Glyph {
pub header: GlyphHeader,
pub data: GlyphData
}
impl Glyph {
pub fn iter<'a>(&'a self) -> Box<dyn Iterator<Item = &'a GlyphPoint> + 'a> {
Box::new(
match &self.data {
GlyphData::Simple(glyph) => glyph.iter(),
GlyphData::Composite => todo!()
}
)
}
}
#[derive(Debug)] #[derive(Debug)]
pub struct GlyphTable { pub struct GlyphTable {
data_source: BufReader<File>, data_source: BufReader<File>,
offset: u32, offset: u32,
cache: HashMap<u16, GlyphData> cache: HashMap<u16, Glyph>
} }
impl GlyphTable { impl GlyphTable {
pub fn get_glyph(&mut self, glyph_id: u16, location: &mut IndexToLocation) -> Result<GlyphData>{ pub fn get_glyph(&mut self, glyph_id: u16, location: &mut IndexToLocation) -> Result<Glyph>{
let Some(offset) = location.get_offset(glyph_id) else { let Some(offset) = location.get_offset(glyph_id) else {
return Err(ErrorKind::Custom("Failed to look up glyph location".into()).into()); return Err(ErrorKind::Custom("Failed to look up glyph location".into()).into());
}; };
@ -201,7 +237,10 @@ impl GlyphTable {
debug!("glyph header: {header:#?}"); debug!("glyph header: {header:#?}");
if header.num_countours >= 0 { if header.num_countours >= 0 {
Ok(GlyphData::Simple(SimpleGlyph::new(header, self.data_source.by_ref())?)) Ok(Glyph {
data: GlyphData::Simple(SimpleGlyph::new(&header, self.data_source.by_ref())?),
header
})
} else { } else {
todo!(); todo!();
} }

View file

@ -1,4 +1,5 @@
mod font; mod font;
mod writer;
use std::{error::Error, fs::File}; use std::{error::Error, fs::File};
@ -6,6 +7,7 @@ use clap::Parser;
use font::Font; use font::Font;
use log::{error, LevelFilter}; use log::{error, LevelFilter};
use simplelog::{ColorChoice, Config, TermLogger, TerminalMode}; use simplelog::{ColorChoice, Config, TermLogger, TerminalMode};
use writer::{SvgWriter, Visitor};
#[derive(Parser, Debug)] #[derive(Parser, Debug)]
#[command(version, about, long_about = None)] #[command(version, about, long_about = None)]
@ -33,13 +35,18 @@ fn main() -> Result<(), Box<dyn Error>> {
// dbg!(font); // dbg!(font);
match font.get_data_for(args.glyph) { let glyph = match font.get_data_for(args.glyph) {
Ok(()) => {}, Ok(glyph) => glyph,
Err(err) => { Err(err) => {
error!("{err}"); error!("{err}");
return Err(err); return Err(err);
} }
}; };
let svg_file = File::create("out.svg")?;
let mut writer = SvgWriter::new(svg_file);
writer.write(&glyph);
Ok(()) Ok(())
} }

19
src/writer.rs Normal file
View file

@ -0,0 +1,19 @@
mod svg;
use crate::font::{GlyphHeader, GlyphPoint, Glyph};
pub use svg::SvgWriter;
pub trait Visitor {
fn write_prefix(&mut self, header: &GlyphHeader);
fn write_point(&mut self, point: &GlyphPoint);
fn write_suffix(&mut self);
fn write(&mut self, glyph: &Glyph) {
self.write_prefix(&glyph.header);
glyph.iter().for_each(|point| self.write_point(point));
self.write_suffix();
}
}

54
src/writer/svg.rs Normal file
View file

@ -0,0 +1,54 @@
use std::{fs::File, io::{BufWriter, Write}};
use crate::font::{GlyphHeader, GlyphPoint};
use super::Visitor;
pub struct SvgWriter {
first_point: bool,
writer: BufWriter<File>
}
impl SvgWriter {
pub fn new(file: File) -> Self {
SvgWriter {
first_point: true,
writer: BufWriter::new(file)
}
}
}
impl Visitor for SvgWriter {
fn write_prefix(&mut self, header: &GlyphHeader) {
write!(
self.writer,
"<svg width=\"1000\" height=\"1000\" viewBox=\"{} {} {} {}\" xmlns=\"http://www.w3.org/2000/svg\"><path d=\"",
header.xmin, header.ymin,
header.xmax - header.xmin,
header.ymax - header.ymin
);
}
fn write_point(&mut self, point: &GlyphPoint) {
if !point.on_curve {
return
}
let prefix = if self.first_point { "M" } else { "L" };
write!(
self.writer,
"{prefix}{} {} ",
point.x, point.y
);
self.first_point = false;
}
fn write_suffix(&mut self) {
write!(
self.writer,
"\" style=\"fill:none; stroke:black; stroke-width:3;\" /></svg>"
);
}
}