use core::convert::TryFrom;
use crate::ttf_parser::{apple_layout, ankr, kerx, GlyphId, FromData};
use crate::Face;
use crate::buffer::{BufferScratchFlags, Buffer};
use crate::ot::{attach_type, lookup_flags, ApplyContext, TableIndex};
use crate::plan::ShapePlan;
use crate::ot::matching::SkippyIter;
trait ExtendedStateTableExt<T: FromData + Copy> {
fn class(&self, glyph_id: GlyphId) -> Option<u16>;
fn entry(&self, state: u16, class: u16) -> Option<apple_layout::GenericStateEntry<T>>;
}
impl ExtendedStateTableExt<kerx::EntryData> for kerx::Subtable1<'_> {
fn class(&self, glyph_id: GlyphId) -> Option<u16> {
self.state_table.class(glyph_id)
}
fn entry(&self, state: u16, class: u16) -> Option<apple_layout::GenericStateEntry<kerx::EntryData>> {
self.state_table.entry(state, class)
}
}
impl ExtendedStateTableExt<kerx::EntryData> for kerx::Subtable4<'_> {
fn class(&self, glyph_id: GlyphId) -> Option<u16> {
self.state_table.class(glyph_id)
}
fn entry(&self, state: u16, class: u16) -> Option<apple_layout::GenericStateEntry<kerx::EntryData>> {
self.state_table.entry(state, class)
}
}
pub(crate) fn apply(plan: &ShapePlan, face: &Face, buffer: &mut Buffer) -> Option<()> {
let mut seen_cross_stream = false;
for subtable in face.tables().kerx?.subtables {
if subtable.variable {
continue;
}
if buffer.direction.is_horizontal() != subtable.horizontal {
continue;
}
let reverse = buffer.direction.is_backward();
if !seen_cross_stream && subtable.has_cross_stream {
seen_cross_stream = true;
for pos in &mut buffer.pos {
pos.set_attach_type(attach_type::CURSIVE);
pos.set_attach_chain(if buffer.direction.is_forward() { -1 } else { 1 });
}
}
if reverse {
buffer.reverse();
}
match subtable.format {
kerx::Format::Format0(_) => {
if !plan.requested_kerning {
continue;
}
apply_simple_kerning(&subtable, plan, face, buffer);
}
kerx::Format::Format1(ref sub) => {
let mut driver = Driver1 {
stack: [0; 8],
depth: 0,
};
apply_state_machine_kerning(&subtable, sub, &mut driver, plan, buffer);
}
kerx::Format::Format2(_) => {
if !plan.requested_kerning {
continue;
}
apply_simple_kerning(&subtable, plan, face, buffer);
}
kerx::Format::Format4(ref sub) => {
let mut driver = Driver4 {
mark_set: false,
mark: 0,
ankr_table: face.tables().ankr.clone(),
};
apply_state_machine_kerning(&subtable, sub, &mut driver, plan, buffer);
}
kerx::Format::Format6(_) => {
if !plan.requested_kerning {
continue;
}
apply_simple_kerning(&subtable, plan, face, buffer);
}
}
if reverse {
buffer.reverse();
}
}
Some(())
}
fn apply_simple_kerning(
subtable: &kerx::Subtable,
plan: &ShapePlan,
face: &Face,
buffer: &mut Buffer,
) {
let mut ctx = ApplyContext::new(TableIndex::GPOS, face, buffer);
ctx.lookup_mask = plan.kern_mask;
ctx.lookup_props = u32::from(lookup_flags::IGNORE_FLAGS);
let horizontal = ctx.buffer.direction.is_horizontal();
let mut i = 0;
while i < ctx.buffer.len {
if (ctx.buffer.info[i].mask & plan.kern_mask) == 0 {
i += 1;
continue;
}
let mut iter = SkippyIter::new(&ctx, i, 1, false);
if !iter.next() {
i += 1;
continue;
}
let j = iter.index();
let info = &ctx.buffer.info;
let kern = subtable.glyphs_kerning(info[i].as_glyph(), info[j].as_glyph()).unwrap_or(0);
let kern = i32::from(kern);
let pos = &mut ctx.buffer.pos;
if kern != 0 {
if horizontal {
if subtable.has_cross_stream {
pos[j].y_offset = kern;
ctx.buffer.scratch_flags |= BufferScratchFlags::HAS_GPOS_ATTACHMENT;
} else {
let kern1 = kern >> 1;
let kern2 = kern - kern1;
pos[i].x_advance += kern1;
pos[j].x_advance += kern2;
pos[j].x_offset += kern2;
}
} else {
if subtable.has_cross_stream {
pos[j].x_offset = kern;
ctx.buffer.scratch_flags |= BufferScratchFlags::HAS_GPOS_ATTACHMENT;
} else {
let kern1 = kern >> 1;
let kern2 = kern - kern1;
pos[i].y_advance += kern1;
pos[j].y_advance += kern2;
pos[j].y_offset += kern2;
}
}
ctx.buffer.unsafe_to_break(i, j + 1)
}
i = j;
}
}
const START_OF_TEXT: u16 = 0;
trait KerxEntryDataExt {
fn action_index(self) -> u16;
fn is_actionable(&self) -> bool;
}
impl KerxEntryDataExt for apple_layout::GenericStateEntry<kerx::EntryData> {
fn action_index(self) -> u16 { self.extra.action_index }
fn is_actionable(&self) -> bool { self.extra.action_index != 0xFFFF }
}
fn apply_state_machine_kerning<T, E>(
subtable: &kerx::Subtable,
state_table: &T,
driver: &mut dyn StateTableDriver<T, E>,
plan: &ShapePlan,
buffer: &mut Buffer,
)
where T: ExtendedStateTableExt<E>,
E: FromData + Copy,
apple_layout::GenericStateEntry<E>: KerxEntryDataExt
{
let mut state = START_OF_TEXT;
buffer.idx = 0;
loop {
let class = if buffer.idx < buffer.len {
state_table.class(buffer.info[buffer.idx].as_glyph()).unwrap_or(1)
} else {
u16::from(apple_layout::class::END_OF_TEXT)
};
let entry: apple_layout::GenericStateEntry<E> = match state_table.entry(state, class) {
Some(v) => v,
None => break,
};
if state != START_OF_TEXT &&
buffer.backtrack_len() != 0 &&
buffer.idx < buffer.len
{
if entry.is_actionable() ||
!(entry.new_state == START_OF_TEXT && !entry.has_advance())
{
buffer.unsafe_to_break_from_outbuffer(buffer.backtrack_len() - 1, buffer.idx + 1);
}
}
if buffer.idx + 2 <= buffer.len {
let end_entry: Option<apple_layout::GenericStateEntry<E>>
= state_table.entry(state, u16::from(apple_layout::class::END_OF_TEXT));
let end_entry = match end_entry {
Some(v) => v,
None => break,
};
if end_entry.is_actionable() {
buffer.unsafe_to_break(buffer.idx, buffer.idx + 2);
}
}
let _ = driver.transition(state_table, entry, subtable.has_cross_stream, subtable.tuple_count, plan, buffer);
state = entry.new_state;
if buffer.idx >= buffer.len {
break;
}
if entry.has_advance() || buffer.max_ops <= 0 {
buffer.next_glyph();
}
buffer.max_ops -= 1;
}
}
trait StateTableDriver<Table, E: FromData> {
fn transition(&mut self, aat: &Table, entry: apple_layout::GenericStateEntry<E>,
has_cross_stream: bool, tuple_count: u32, plan: &ShapePlan, buffer: &mut Buffer) -> Option<()>;
}
struct Driver1 {
stack: [usize; 8],
depth: usize,
}
impl StateTableDriver<kerx::Subtable1<'_>, kerx::EntryData> for Driver1 {
fn transition(
&mut self,
aat: &kerx::Subtable1,
entry: apple_layout::GenericStateEntry<kerx::EntryData>,
has_cross_stream: bool,
tuple_count: u32,
plan: &ShapePlan,
buffer: &mut Buffer,
) -> Option<()> {
if entry.has_reset() {
self.depth = 0;
}
if entry.has_push() {
if self.depth < self.stack.len() {
self.stack[self.depth] = buffer.idx;
self.depth += 1;
} else {
self.depth = 0; }
}
if entry.is_actionable() && self.depth != 0 {
let tuple_count = u16::try_from(tuple_count.max(1)).ok()?;
let mut action_index = entry.action_index();
let mut last = false;
while !last && self.depth != 0 {
self.depth -= 1;
let idx = self.stack[self.depth];
let mut v = aat.glyphs_kerning(action_index)? as i32;
action_index = action_index.checked_add(tuple_count)?;
if idx >= buffer.len {
continue;
}
last = v & 1 != 0;
v &= !1;
let mut has_gpos_attachment = false;
let glyph_mask = buffer.info[idx].mask;
let pos = &mut buffer.pos[idx];
if buffer.direction.is_horizontal() {
if has_cross_stream {
if v == -0x8000 {
pos.set_attach_type(0);
pos.set_attach_chain(0);
pos.y_offset = 0;
} else if pos.attach_type() != 0 {
pos.y_offset += v;
has_gpos_attachment = true;
}
} else if glyph_mask & plan.kern_mask != 0 {
pos.x_advance += v;
pos.x_offset += v;
}
} else {
if has_cross_stream {
if v == -0x8000 {
pos.set_attach_type(0);
pos.set_attach_chain(0);
pos.x_offset = 0;
} else if pos.attach_type() != 0 {
pos.x_offset += v;
has_gpos_attachment = true;
}
} else if glyph_mask & plan.kern_mask != 0 {
if pos.y_offset == 0 {
pos.y_advance += v;
pos.y_offset += v;
}
}
}
if has_gpos_attachment {
buffer.scratch_flags |= BufferScratchFlags::HAS_GPOS_ATTACHMENT;
}
}
}
Some(())
}
}
struct Driver4<'a> {
mark_set: bool,
mark: usize,
ankr_table: Option<ankr::Table<'a>>,
}
impl StateTableDriver<kerx::Subtable4<'_>, kerx::EntryData> for Driver4<'_> {
fn transition(
&mut self,
aat: &kerx::Subtable4,
entry: apple_layout::GenericStateEntry<kerx::EntryData>,
_has_cross_stream: bool,
_tuple_count: u32,
_opt: &ShapePlan,
buffer: &mut Buffer,
) -> Option<()> {
if self.mark_set && entry.is_actionable() && buffer.idx < buffer.len {
if let Some(ref ankr_table) = self.ankr_table {
let point = aat.anchor_points.get(entry.action_index())?;
let mark_idx = buffer.info[self.mark].as_glyph();
let mark_anchor = ankr_table
.points(mark_idx)
.and_then(|list| list.get(u32::from(point.0)))
.unwrap_or_default();
let curr_idx = buffer.cur(0).as_glyph();
let curr_anchor = ankr_table
.points(curr_idx)
.and_then(|list| list.get(u32::from(point.1)))
.unwrap_or_default();
let pos = buffer.cur_pos_mut();
pos.x_offset = i32::from(mark_anchor.x - curr_anchor.x);
pos.y_offset = i32::from(mark_anchor.y - curr_anchor.y);
}
buffer.cur_pos_mut().set_attach_type(attach_type::MARK);
let idx = buffer.idx;
buffer.cur_pos_mut().set_attach_chain(self.mark as i16 - idx as i16);
buffer.scratch_flags |= BufferScratchFlags::HAS_GPOS_ATTACHMENT;
}
if entry.has_mark() {
self.mark_set = true;
self.mark = buffer.idx;
}
Some(())
}
}