1 | use crate::hb::ot_layout_gpos_table::ValueRecordExt; |
2 | use crate::hb::ot_layout_gsubgpos::OT::hb_ot_apply_context_t; |
3 | use crate::hb::ot_layout_gsubgpos::{skipping_iterator_t, Apply}; |
4 | use ttf_parser::gpos::{PairAdjustment, ValueRecord}; |
5 | |
6 | impl Apply for PairAdjustment<'_> { |
7 | fn apply(&self, ctx: &mut hb_ot_apply_context_t) -> Option<()> { |
8 | let first_glyph = ctx.buffer.cur(0).as_glyph(); |
9 | let first_glyph_coverage_index = self.coverage().get(first_glyph)?; |
10 | |
11 | let mut iter = skipping_iterator_t::new(ctx, ctx.buffer.idx, false); |
12 | |
13 | let mut unsafe_to = 0; |
14 | if !iter.next(Some(&mut unsafe_to)) { |
15 | ctx.buffer |
16 | .unsafe_to_concat(Some(ctx.buffer.idx), Some(unsafe_to)); |
17 | return None; |
18 | } |
19 | |
20 | let second_glyph_index = iter.index(); |
21 | let second_glyph = ctx.buffer.info[second_glyph_index].as_glyph(); |
22 | |
23 | let finish = |ctx: &mut hb_ot_apply_context_t, iter_index: &mut usize, has_record2| { |
24 | if has_record2 { |
25 | *iter_index += 1; |
26 | // https://github.com/harfbuzz/harfbuzz/issues/3824 |
27 | // https://github.com/harfbuzz/harfbuzz/issues/3888#issuecomment-1326781116 |
28 | ctx.buffer |
29 | .unsafe_to_break(Some(ctx.buffer.idx), Some(*iter_index + 1)); |
30 | } |
31 | |
32 | ctx.buffer.idx = *iter_index; |
33 | |
34 | Some(()) |
35 | }; |
36 | |
37 | let boring = |ctx: &mut hb_ot_apply_context_t, iter_index: &mut usize, has_record2| { |
38 | ctx.buffer |
39 | .unsafe_to_concat(Some(ctx.buffer.idx), Some(second_glyph_index + 1)); |
40 | finish(ctx, iter_index, has_record2) |
41 | }; |
42 | |
43 | let success = |
44 | |ctx: &mut hb_ot_apply_context_t, iter_index: &mut usize, flag1, flag2, has_record2| { |
45 | if flag1 || flag2 { |
46 | ctx.buffer |
47 | .unsafe_to_break(Some(ctx.buffer.idx), Some(second_glyph_index + 1)); |
48 | finish(ctx, iter_index, has_record2) |
49 | } else { |
50 | boring(ctx, iter_index, has_record2) |
51 | } |
52 | }; |
53 | |
54 | let bail = |ctx: &mut hb_ot_apply_context_t, |
55 | iter_index: &mut usize, |
56 | records: (ValueRecord, ValueRecord)| { |
57 | let has_record1 = !records.0.is_empty(); |
58 | let has_record2 = !records.1.is_empty(); |
59 | |
60 | let flag1 = has_record1 && records.0.apply(ctx, ctx.buffer.idx); |
61 | let flag2 = has_record2 && records.1.apply(ctx, second_glyph_index); |
62 | |
63 | success(ctx, iter_index, flag1, flag2, has_record2) |
64 | }; |
65 | |
66 | let records = match self { |
67 | Self::Format1 { sets, .. } => { |
68 | sets.get(first_glyph_coverage_index)?.get(second_glyph)? |
69 | } |
70 | Self::Format2 { |
71 | classes, matrix, .. |
72 | } => { |
73 | let classes = (classes.0.get(first_glyph), classes.1.get(second_glyph)); |
74 | |
75 | let records = match matrix.get(classes) { |
76 | Some(v) => v, |
77 | None => { |
78 | ctx.buffer |
79 | .unsafe_to_concat(Some(ctx.buffer.idx), Some(iter.index() + 1)); |
80 | return None; |
81 | } |
82 | }; |
83 | |
84 | return bail(ctx, &mut iter.buf_idx, records); |
85 | } |
86 | }; |
87 | |
88 | bail(ctx, &mut iter.buf_idx, records) |
89 | } |
90 | } |
91 | |