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
pub mod gdef;
pub mod gpos;
pub mod gsub;
#[cfg(test)]
#[path = "./tests/layout.rs"]
mod tests;
include!("../generated/generated_layout.rs");
impl<'a, T: FontRead<'a>> Lookup<'a, T> {
pub fn get_subtable(&self, offset: Offset16) -> Result<T, ReadError> {
self.resolve_offset(offset)
}
}
pub enum FeatureParams<'a> {
StylisticSet(StylisticSetParams<'a>),
Size(SizeParams<'a>),
CharacterVariant(CharacterVariantParams<'a>),
}
impl ReadArgs for FeatureParams<'_> {
type Args = Tag;
}
impl<'a> FontReadWithArgs<'a> for FeatureParams<'a> {
fn read_with_args(bytes: FontData<'a>, args: &Tag) -> Result<FeatureParams<'a>, ReadError> {
match *args {
t if t == Tag::new(b"size") => SizeParams::read(bytes).map(Self::Size),
t if &t.to_raw()[..2] == b"ss" => {
StylisticSetParams::read(bytes).map(Self::StylisticSet)
}
t if &t.to_raw()[..2] == b"cv" => {
CharacterVariantParams::read(bytes).map(Self::CharacterVariant)
}
_ => Err(ReadError::InvalidFormat(0xdead)),
}
}
}
#[cfg(feature = "traversal")]
impl<'a> SomeTable<'a> for FeatureParams<'a> {
fn type_name(&self) -> &str {
match self {
FeatureParams::StylisticSet(table) => table.type_name(),
FeatureParams::Size(table) => table.type_name(),
FeatureParams::CharacterVariant(table) => table.type_name(),
}
}
fn get_field(&self, idx: usize) -> Option<Field<'a>> {
match self {
FeatureParams::StylisticSet(table) => table.get_field(idx),
FeatureParams::Size(table) => table.get_field(idx),
FeatureParams::CharacterVariant(table) => table.get_field(idx),
}
}
}
impl FeatureTableSubstitutionRecord {
pub fn alternate_feature<'a>(&self, data: FontData<'a>) -> Result<Feature<'a>, ReadError> {
self.alternate_feature_offset()
.resolve_with_args(data, &Tag::new(b"NULL"))
}
}
impl CoverageTable<'_> {
pub fn iter(&self) -> impl Iterator<Item = GlyphId> + '_ {
let (iter1, iter2) = match self {
CoverageTable::Format1(t) => (Some(t.glyph_array().iter().map(|g| g.get())), None),
CoverageTable::Format2(t) => {
let iter = t.range_records().iter().flat_map(RangeRecord::iter);
(None, Some(iter))
}
};
iter1
.into_iter()
.flatten()
.chain(iter2.into_iter().flatten())
}
}
impl RangeRecord {
fn iter(&self) -> impl Iterator<Item = GlyphId> + '_ {
(self.start_glyph_id().to_u16()..=self.end_glyph_id().to_u16()).map(GlyphId::new)
}
}
impl Default for DeltaFormat {
fn default() -> Self {
DeltaFormat::Local2BitDeltas
}
}
fn delta_value_count(start_size: u16, end_size: u16, delta_format: DeltaFormat) -> usize {
let range_len = end_size.saturating_add(1).saturating_sub(start_size) as usize;
let val_per_word = match delta_format {
DeltaFormat::Local2BitDeltas => 8,
DeltaFormat::Local4BitDeltas => 4,
DeltaFormat::Local8BitDeltas => 2,
_ => return 0,
};
let count = range_len / val_per_word;
let extra = (range_len % val_per_word).min(1);
count + extra
}