use crate::ttf_parser::GlyphId;
use crate::{Direction, Face};
use crate::face::GlyphExtents;
use crate::buffer::{Buffer, GlyphPosition};
use crate::unicode::{space, modified_combining_class, CanonicalCombiningClass, GeneralCategory};
use crate::plan::ShapePlan;
pub fn recategorize_marks(_: &ShapePlan, _: &Face, buffer: &mut Buffer) {
let len = buffer.len;
for info in &mut buffer.info[..len] {
if info.general_category() == GeneralCategory::NonspacingMark {
let mut class = info.modified_combining_class();
class = recategorize_combining_class(info.glyph_id, class);
info.set_modified_combining_class(class);
}
}
}
fn recategorize_combining_class(u: u32, mut class: u8) -> u8 {
use CanonicalCombiningClass as Class;
use modified_combining_class as mcc;
if class >= 200 {
return class;
}
if u & !0xFF == 0x0E00 {
if class == 0 {
match u {
0x0E31 |
0x0E34 |
0x0E35 |
0x0E36 |
0x0E37 |
0x0E47 |
0x0E4C |
0x0E4D |
0x0E4E => class = Class::AboveRight as u8,
0x0EB1 |
0x0EB4 |
0x0EB5 |
0x0EB6 |
0x0EB7 |
0x0EBB |
0x0ECC |
0x0ECD => class = Class::Above as u8,
0x0EBC => class = Class::Below as u8,
_ => {}
}
} else {
if u == 0x0E3A {
class = Class::BelowRight as u8;
}
}
}
match class {
mcc::CCC10 => Class::Below as u8, mcc::CCC11 => Class::Below as u8, mcc::CCC12 => Class::Below as u8, mcc::CCC13 => Class::Below as u8, mcc::CCC14 => Class::Below as u8, mcc::CCC15 => Class::Below as u8, mcc::CCC16 => Class::Below as u8, mcc::CCC17 => Class::Below as u8, mcc::CCC18 => Class::Below as u8, mcc::CCC20 => Class::Below as u8, mcc::CCC22 => Class::Below as u8, mcc::CCC23 => Class::AttachedAbove as u8, mcc::CCC24 => Class::AboveRight as u8, mcc::CCC25 => Class::AboveLeft as u8, mcc::CCC19 => Class::AboveLeft as u8, mcc::CCC26 => Class::Above as u8, mcc::CCC21 => class,
mcc::CCC27 => Class::Above as u8, mcc::CCC28 => Class::Above as u8, mcc::CCC30 => Class::Above as u8, mcc::CCC31 => Class::Above as u8, mcc::CCC33 => Class::Above as u8, mcc::CCC34 => Class::Above as u8, mcc::CCC35 => Class::Above as u8, mcc::CCC36 => Class::Above as u8, mcc::CCC29 => Class::Below as u8, mcc::CCC32 => Class::Below as u8,
mcc::CCC103 => Class::BelowRight as u8, mcc::CCC107 => Class::AboveRight as u8,
mcc::CCC118 => Class::Below as u8, mcc::CCC122 => Class::Above as u8,
mcc::CCC129 => Class::Below as u8, mcc::CCC130 => Class::Above as u8, mcc::CCC132 => Class::Below as u8,
_ => class,
}
}
pub fn position_marks(
plan: &ShapePlan,
face: &Face,
buffer: &mut Buffer,
adjust_offsets_when_zeroing: bool,
) {
let mut start = 0;
let len = buffer.len;
for i in 1..len {
if !buffer.info[i].is_unicode_mark() {
position_cluster(plan, face, buffer, start, i, adjust_offsets_when_zeroing);
start = i;
}
}
position_cluster(plan, face, buffer, start, len, adjust_offsets_when_zeroing);
}
fn position_cluster(
plan: &ShapePlan,
face: &Face,
buffer: &mut Buffer,
start: usize,
end: usize,
adjust_offsets_when_zeroing: bool,
) {
if end - start < 2 {
return;
}
let mut i = start;
while i < end {
if !buffer.info[i].is_unicode_mark() {
let mut j = i + 1;
while j < end && buffer.info[j].is_unicode_mark() {
j += 1;
}
position_around_base(plan, face, buffer, i, j, adjust_offsets_when_zeroing);
i = j - 1;
}
i += 1;
}
}
fn position_around_base(
plan: &ShapePlan,
face: &Face,
buffer: &mut Buffer,
base: usize,
end: usize,
adjust_offsets_when_zeroing: bool,
) {
let mut horizontal_dir = Direction::Invalid;
buffer.unsafe_to_break(base, end);
let base_info = &buffer.info[base];
let base_pos = &buffer.pos[base];
let base_glyph = base_info.as_glyph();
let mut base_extents = match face.glyph_extents(base_glyph) {
Some(extents) => extents,
None => {
zero_mark_advances(buffer, base + 1, end, adjust_offsets_when_zeroing);
return;
}
};
base_extents.y_bearing += base_pos.y_offset;
base_extents.x_bearing = 0;
base_extents.width = face.glyph_h_advance(base_glyph) as i32;
let lig_id = base_info.lig_id() as u32;
let num_lig_components = base_info.lig_num_comps() as i32;
let mut x_offset = 0;
let mut y_offset = 0;
if buffer.direction.is_forward() {
x_offset -= base_pos.x_advance;
y_offset -= base_pos.y_advance;
}
let mut last_lig_component: i32 = -1;
let mut last_combining_class: u8 = 255;
let mut component_extents = base_extents;
let mut cluster_extents = base_extents;
for (info, pos) in buffer.info[base+1..end].iter().zip(&mut buffer.pos[base+1..end]) {
if info.modified_combining_class() != 0 {
if num_lig_components > 1 {
let this_lig_id = info.lig_id() as u32;
let mut this_lig_component = info.lig_comp() as i32 - 1;
if lig_id == 0 || lig_id != this_lig_id || this_lig_component >= num_lig_components {
this_lig_component = num_lig_components - 1;
}
if last_lig_component != this_lig_component {
last_lig_component = this_lig_component;
last_combining_class = 255;
component_extents = base_extents;
if horizontal_dir == Direction::Invalid {
horizontal_dir = if plan.direction.is_horizontal() {
plan.direction
} else {
plan.script
.and_then(Direction::from_script)
.unwrap_or(Direction::LeftToRight)
};
}
component_extents.x_bearing +=
(if horizontal_dir == Direction::LeftToRight {
this_lig_component
} else {
num_lig_components - 1 - this_lig_component
} * component_extents.width) / num_lig_components;
component_extents.width /= num_lig_components;
}
}
let this_combining_class = info.modified_combining_class();
if last_combining_class != this_combining_class {
last_combining_class = this_combining_class;
cluster_extents = component_extents;
}
position_mark(
plan,
face,
buffer.direction,
info.as_glyph(),
pos,
&mut cluster_extents,
conv_combining_class(this_combining_class),
);
pos.x_advance = 0;
pos.y_advance = 0;
pos.x_offset += x_offset;
pos.y_offset += y_offset;
} else {
if buffer.direction.is_forward() {
x_offset -= pos.x_advance;
y_offset -= pos.y_advance;
} else {
x_offset += pos.x_advance;
y_offset += pos.y_advance;
}
}
}
}
fn zero_mark_advances(
buffer: &mut Buffer,
start: usize,
end: usize,
adjust_offsets_when_zeroing: bool,
) {
for (info, pos) in buffer.info[start..end].iter().zip(&mut buffer.pos[start..end]) {
if info.general_category() == GeneralCategory::NonspacingMark {
if adjust_offsets_when_zeroing {
pos.x_offset -= pos.x_advance;
pos.y_offset -= pos.y_advance;
}
pos.x_advance = 0;
pos.y_advance = 0;
}
}
}
fn position_mark(
_: &ShapePlan,
face: &Face,
direction: Direction,
glyph: GlyphId,
pos: &mut GlyphPosition,
base_extents: &mut GlyphExtents,
combining_class: CanonicalCombiningClass,
) {
use CanonicalCombiningClass as Class;
let mark_extents = match face.glyph_extents(glyph) {
Some(extents) => extents,
None => return,
};
let y_gap = face.units_per_em as i32 / 16;
pos.x_offset = 0;
pos.y_offset = 0;
match combining_class {
Class::DoubleBelow |
Class::DoubleAbove if direction.is_horizontal() => {
pos.x_offset += base_extents.x_bearing
+ if direction.is_forward() { base_extents.width } else { 0 }
- mark_extents.width / 2 - mark_extents.x_bearing;
}
Class::AttachedBelowLeft |
Class::BelowLeft |
Class::AboveLeft => {
pos.x_offset += base_extents.x_bearing - mark_extents.x_bearing;
}
Class::AttachedAboveRight |
Class::BelowRight |
Class::AboveRight => {
pos.x_offset += base_extents.x_bearing + base_extents.width
- mark_extents.width - mark_extents.x_bearing;
}
Class::AttachedBelow |
Class::AttachedAbove |
Class::Below |
Class::Above |
_ => {
pos.x_offset += base_extents.x_bearing
+ (base_extents.width - mark_extents.width) / 2
- mark_extents.x_bearing;
}
}
let is_attached = matches!(
combining_class,
Class::AttachedBelowLeft |
Class::AttachedBelow |
Class::AttachedAbove |
Class::AttachedAboveRight
);
match combining_class {
Class::DoubleBelow |
Class::BelowLeft |
Class::Below |
Class::BelowRight |
Class::AttachedBelowLeft |
Class::AttachedBelow => {
if !is_attached {
base_extents.height -= y_gap;
}
pos.y_offset = base_extents.y_bearing + base_extents.height
- mark_extents.y_bearing;
if (y_gap > 0) == (pos.y_offset > 0) {
base_extents.height -= pos.y_offset;
pos.y_offset = 0;
}
base_extents.height += mark_extents.height;
}
Class::DoubleAbove |
Class::AboveLeft |
Class::Above |
Class::AboveRight |
Class::AttachedAbove |
Class::AttachedAboveRight => {
if !is_attached {
base_extents.y_bearing += y_gap;
base_extents.height -= y_gap;
}
pos.y_offset = base_extents.y_bearing
- (mark_extents.y_bearing + mark_extents.height);
if (y_gap > 0) != (pos.y_offset > 0) {
let correction = -pos.y_offset / 2;
base_extents.y_bearing += correction;
base_extents.height -= correction;
pos.y_offset += correction;
}
base_extents.y_bearing -= mark_extents.height;
base_extents.height += mark_extents.height;
}
_ => {}
}
}
pub fn adjust_spaces(_: &ShapePlan, face: &Face, buffer: &mut Buffer) {
let len = buffer.len;
let horizontal = buffer.direction.is_horizontal();
for (info, pos) in buffer.info[..len].iter().zip(&mut buffer.pos[..len]) {
let space_type = match info.space_fallback() {
Some(fallback) if !info.is_ligated() => fallback,
_ => continue,
};
match space_type {
space::SPACE_EM |
space::SPACE_EM_2 |
space::SPACE_EM_3 |
space::SPACE_EM_4 |
space::SPACE_EM_5 |
space::SPACE_EM_6 |
space::SPACE_EM_16 => {
let length = (face.units_per_em as i32 + (space_type as i32) / 2) / space_type as i32;
if horizontal {
pos.x_advance = length;
} else {
pos.y_advance = -length;
}
}
space::SPACE_4_EM_18 => {
let length = ((face.units_per_em as i64) * 4 / 18) as i32;
if horizontal {
pos.x_advance = length
} else {
pos.y_advance = -length;
}
}
space::SPACE_FIGURE => {
for u in '0'..='9' {
if let Some(glyph) = face.glyph_index(u as u32) {
if horizontal {
pos.x_advance = face.glyph_h_advance(glyph) as i32;
} else {
pos.y_advance = face.glyph_v_advance(glyph);
}
break;
}
}
}
space::SPACE_PUNCTUATION => {
let punct = face
.glyph_index('.' as u32)
.or_else(|| face.glyph_index(',' as u32));
if let Some(glyph) = punct {
if horizontal {
pos.x_advance = face.glyph_h_advance(glyph) as i32;
} else {
pos.y_advance = face.glyph_v_advance(glyph);
}
}
}
space::SPACE_NARROW => {
if horizontal {
pos.x_advance /= 2;
} else {
pos.y_advance /= 2;
}
}
_ => {}
}
}
}
fn conv_combining_class(n: u8) -> CanonicalCombiningClass {
use CanonicalCombiningClass as Class;
match n {
1 => Class::Overlay,
6 => Class::HanReading,
7 => Class::Nukta,
8 => Class::KanaVoicing,
9 => Class::Virama,
10 => Class::CCC10,
11 => Class::CCC11,
12 => Class::CCC12,
13 => Class::CCC13,
14 => Class::CCC14,
15 => Class::CCC15,
16 => Class::CCC16,
17 => Class::CCC17,
18 => Class::CCC18,
19 => Class::CCC19,
20 => Class::CCC20,
21 => Class::CCC21,
22 => Class::CCC22,
23 => Class::CCC23,
24 => Class::CCC24,
25 => Class::CCC25,
26 => Class::CCC26,
27 => Class::CCC27,
28 => Class::CCC28,
29 => Class::CCC29,
30 => Class::CCC30,
31 => Class::CCC31,
32 => Class::CCC32,
33 => Class::CCC33,
34 => Class::CCC34,
35 => Class::CCC35,
36 => Class::CCC36,
84 => Class::CCC84,
91 => Class::CCC91,
103 => Class::CCC103,
107 => Class::CCC107,
118 => Class::CCC118,
122 => Class::CCC122,
129 => Class::CCC129,
130 => Class::CCC130,
132 => Class::CCC132,
200 => Class::AttachedBelowLeft,
202 => Class::AttachedBelow,
214 => Class::AttachedAbove,
216 => Class::AttachedAboveRight,
218 => Class::BelowLeft,
220 => Class::Below,
222 => Class::BelowRight,
224 => Class::Left,
226 => Class::Right,
228 => Class::AboveLeft,
230 => Class::Above,
232 => Class::AboveRight,
233 => Class::DoubleBelow,
234 => Class::DoubleAbove,
240 => Class::IotaSubscript,
_ => Class::NotReordered,
}
}