1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
use core::convert::TryFrom;
use core::num::NonZeroU16;
use crate::{GlyphId, RasterGlyphImage, RasterImageFormat, Tag};
use crate::parser::{Stream, FromData, Offset, Offset32};
pub fn parse(
data: &[u8],
number_of_glyphs: NonZeroU16,
glyph_id: GlyphId,
pixels_per_em: u16,
depth: u8,
) -> Option<RasterGlyphImage> {
if depth == 10 {
return None;
}
let total_glyphs = u32::from(number_of_glyphs.get().checked_add(1)?);
let mut s = Stream::new(data);
let version: u16 = s.read()?;
if version != 1 {
return None;
}
s.skip::<u16>();
let count: u32 = s.read()?;
if count == 0 {
return None;
}
let strikes = s.read_array32::<Offset32>(count)?;
let mut idx = 0;
let mut max_ppem = 0;
{
for (i, offset) in strikes.into_iter().enumerate() {
let mut s = Stream::new_at(data, offset.to_usize())?;
let ppem: u16 = s.read()?;
s.skip::<u16>();
if (pixels_per_em <= ppem && ppem < max_ppem) ||
(pixels_per_em > max_ppem && ppem > max_ppem)
{
idx = i as u32;
max_ppem = ppem;
}
}
}
let offset = strikes.get(idx)?;
let mut s = Stream::new_at(data, offset.to_usize())?;
s.skip::<u16>();
s.skip::<u16>();
let glyph_offsets = s.read_array32::<Offset32>(total_glyphs)?;
let start = glyph_offsets.get(u32::from(glyph_id.0))?.to_usize();
let end = glyph_offsets.get(u32::from(glyph_id.0.checked_add(1)?))?.to_usize();
if start == end {
return None;
}
let data_len = end.checked_sub(start)?.checked_sub(8)?;
let mut s = Stream::new_at(data, offset.to_usize() + start)?;
let x: i16 = s.read()?;
let y: i16 = s.read()?;
let image_type: Tag = s.read()?;
let image_data = s.read_bytes(data_len)?;
let format = match &image_type.to_bytes() {
b"png " => RasterImageFormat::PNG,
b"dupe" => {
let glyph_id = GlyphId::parse(image_data)?;
return parse(data, number_of_glyphs, glyph_id, pixels_per_em, depth + 1);
}
_ => {
return None;
}
};
let (width, height) = png_size(image_data)?;
Some(RasterGlyphImage {
x,
y,
width,
height,
pixels_per_em: max_ppem,
format,
data: image_data,
})
}
fn png_size(data: &[u8]) -> Option<(u16, u16)> {
let mut s = Stream::new_at(data, 16)?;
let width: u32 = s.read()?;
let height: u32 = s.read()?;
Some((
u16::try_from(width).ok()?,
u16::try_from(height).ok()?,
))
}