rustybuzz/hb/
ot_shaper_hebrew.rs1use super::ot_shape_normalize::*;
2use super::ot_shaper::*;
3use super::{hb_tag_t, unicode};
4use crate::hb::buffer::hb_buffer_t;
5use crate::hb::ot_layout::_hb_glyph_info_get_modified_combining_class;
6use crate::hb::ot_shape_plan::hb_ot_shape_plan_t;
7use crate::hb::unicode::modified_combining_class;
8use unicode_ccc::CanonicalCombiningClass;
9
10pub const HEBREW_SHAPER: hb_ot_shaper_t = hb_ot_shaper_t {
11 collect_features: None,
12 override_features: None,
13 create_data: None,
14 preprocess_text: None,
15 postprocess_glyphs: None,
16 normalization_preference: HB_OT_SHAPE_NORMALIZATION_MODE_AUTO,
17 decompose: None,
18 compose: Some(compose),
19 setup_masks: None,
20 gpos_tag: Some(hb_tag_t::from_bytes(b"hebr")),
21 reorder_marks: Some(reorder_marks_hebrew),
22 zero_width_marks: HB_OT_SHAPE_ZERO_WIDTH_MARKS_BY_GDEF_LATE,
23 fallback_position: true,
24};
25
26fn reorder_marks_hebrew(
27 _: &hb_ot_shape_plan_t,
28 buffer: &mut hb_buffer_t,
29 start: usize,
30 end: usize,
31) {
32 for i in start + 2..end {
33 let c0 = buffer.info[i - 2];
34 let c1 = buffer.info[i - 1];
35 let c2 = buffer.info[i - 0];
36
37 if (_hb_glyph_info_get_modified_combining_class(&c0) == modified_combining_class::CCC17
38 || _hb_glyph_info_get_modified_combining_class(&c0) == modified_combining_class::CCC18) &&
40 (_hb_glyph_info_get_modified_combining_class(&c1) == modified_combining_class::CCC10
41 || _hb_glyph_info_get_modified_combining_class(&c1) == modified_combining_class::CCC14) &&
42 (_hb_glyph_info_get_modified_combining_class(&c2) == modified_combining_class::CCC22
43 || _hb_glyph_info_get_modified_combining_class(&c2) == CanonicalCombiningClass::Below as u8)
44 {
46 buffer.merge_clusters(i - 1, i + 1);
47 buffer.info.swap(i - 1, i);
48 break;
49 }
50 }
51}
52
53const S_DAGESH_FORMS: &[char] = &[
54 '\u{FB30}', '\u{FB31}', '\u{FB32}', '\u{FB33}', '\u{FB34}', '\u{FB35}', '\u{FB36}', '\u{0000}', '\u{FB38}', '\u{FB39}', '\u{FB3A}', '\u{FB3B}', '\u{FB3C}', '\u{0000}', '\u{FB3E}', '\u{0000}', '\u{FB40}', '\u{FB41}', '\u{0000}', '\u{FB43}', '\u{FB44}', '\u{0000}', '\u{FB46}', '\u{FB47}', '\u{FB48}', '\u{FB49}', '\u{FB4A}', ];
82
83fn compose(ctx: &hb_ot_shape_normalize_context_t, a: char, b: char) -> Option<char> {
84 match unicode::compose(a, b) {
89 Some(c) => Some(c),
90 None if !ctx.plan.has_gpos_mark => {
91 let a = a as u32;
94 let b = b as u32;
95 match b {
96 0x05B4 => {
97 match a {
99 0x05D9 => Some('\u{FB1D}'), _ => None,
101 }
102 }
103 0x05B7 => {
104 match a {
106 0x05D9 => Some('\u{FB1F}'), 0x05D0 => Some('\u{FB2E}'), _ => None,
109 }
110 }
111 0x05B8 => {
112 match a {
114 0x05D0 => Some('\u{FB2F}'), _ => None,
116 }
117 }
118 0x05B9 => {
119 match a {
121 0x05D5 => Some('\u{FB4B}'), _ => None,
123 }
124 }
125 0x05BC => {
126 match a {
128 0x05D0..=0x05EA => {
129 let c = S_DAGESH_FORMS[a as usize - 0x05D0];
130 if c != '\0' {
131 Some(c)
132 } else {
133 None
134 }
135 }
136 0xFB2A => Some('\u{FB2C}'), 0xFB2B => Some('\u{FB2D}'), _ => None,
139 }
140 }
141 0x05BF => {
142 match a {
144 0x05D1 => Some('\u{FB4C}'), 0x05DB => Some('\u{FB4D}'), 0x05E4 => Some('\u{FB4E}'), _ => None,
148 }
149 }
150 0x05C1 => {
151 match a {
153 0x05E9 => Some('\u{FB2A}'), 0xFB49 => Some('\u{FB2C}'), _ => None,
156 }
157 }
158 0x05C2 => {
159 match a {
161 0x05E9 => Some('\u{FB2B}'), 0xFB49 => Some('\u{FB2D}'), _ => None,
164 }
165 }
166 _ => None,
167 }
168 }
169 None => None,
170 }
171}