use crate::hb::common::{HB_FEATURE_GLOBAL_END, HB_FEATURE_GLOBAL_START};
use crate::Feature;
use alloc::vec;
use alloc::vec::Vec;
use core::cmp::Ordering;
use super::layout::*;
use crate::hb::{hb_font_t, hb_mask_t, hb_tag_t};
#[doc(alias = "hb_aat_map_t")]
#[derive(Default)]
pub struct AatMap {
pub chain_flags: Vec<Vec<RangeFlags>>,
}
#[derive(Copy, Clone)]
pub struct RangeFlags {
pub flags: hb_mask_t,
pub cluster_first: u32,
pub cluster_last: u32, }
#[doc(alias = "hb_aat_map_builder_t")]
pub struct AatMapBuilder {
pub current_features: Vec<FeatureInfo>,
pub features: Vec<FeatureRange>,
pub range_first: usize,
pub range_last: usize,
}
impl Default for AatMapBuilder {
fn default() -> Self {
Self {
range_first: HB_FEATURE_GLOBAL_START as usize,
range_last: HB_FEATURE_GLOBAL_END as usize,
current_features: Vec::default(),
features: Vec::default(),
}
}
}
impl AatMapBuilder {
pub fn add_feature(&mut self, face: &hb_font_t, feature: &Feature) -> Option<()> {
let feat = face.aat_tables.feat.as_ref()?;
if feature.tag == hb_tag_t::new(b"aalt") {
let exposes_feature = feat
.find(FEATURE_TYPE_CHARACTER_ALTERNATIVES as u16)
.is_some_and(|f| f.n_settings() != 0);
if !exposes_feature {
return Some(());
}
self.features.push(FeatureRange {
start: feature.start,
end: feature.end,
info: FeatureInfo {
kind: FEATURE_TYPE_CHARACTER_ALTERNATIVES as u16,
setting: u16::try_from(feature.value).unwrap(),
is_exclusive: true,
},
});
}
let idx = feature_mappings
.binary_search_by(|map| map.ot_feature_tag.cmp(&feature.tag))
.ok()?;
let mapping = &feature_mappings[idx];
let mut feature_name = feat.find(mapping.aat_feature_type as u16);
match feature_name {
Some(feature) if feature.n_settings() != 0 => {}
_ => {
if mapping.aat_feature_type == FEATURE_TYPE_LOWER_CASE
&& mapping.selector_to_enable == FEATURE_SELECTOR_LOWER_CASE_SMALL_CAPS
{
feature_name = feat.find(FEATURE_TYPE_LETTER_CASE as u16);
}
}
}
match feature_name {
Some(feature_name) if feature_name.n_settings() != 0 => {
let setting = if feature.value != 0 {
mapping.selector_to_enable
} else {
mapping.selector_to_disable
} as u16;
self.features.push(FeatureRange {
start: feature.start,
end: feature.end,
info: FeatureInfo {
kind: mapping.aat_feature_type as u16,
setting,
is_exclusive: feature_name.is_exclusive(),
},
});
}
_ => {}
}
Some(())
}
pub fn compile(&mut self, face: &hb_font_t, m: &mut AatMap) {
let mut feature_events = vec![];
for feature in &self.features {
if feature.start == feature.end {
continue;
}
feature_events.push(FeatureEvent {
index: feature.start as usize,
start: true,
feature: feature.info,
});
feature_events.push(FeatureEvent {
index: feature.end as usize,
start: false,
feature: feature.info,
});
}
feature_events.sort();
feature_events.push(FeatureEvent {
index: u32::MAX as usize,
start: false,
feature: FeatureInfo::default(),
});
let mut active_features = vec![];
let mut last_index = 0;
for event in &feature_events {
if event.index != last_index {
self.current_features.clone_from(&active_features);
self.range_first = last_index;
self.range_last = event.index.wrapping_sub(1);
if !self.current_features.is_empty() {
self.current_features.sort();
let mut j = 0;
for i in 1..self.current_features.len() {
let non_exclusive = !self.current_features[i].is_exclusive
&& (self.current_features[i].setting & !1)
!= (self.current_features[j].setting & !1);
if self.current_features[i].kind != self.current_features[j].kind
|| non_exclusive
{
j += 1;
self.current_features[j] = self.current_features[i];
}
}
self.current_features.truncate(j + 1);
}
super::layout_morx_table::compile_flags(face, self, m);
last_index = event.index;
}
if event.start {
active_features.push(event.feature);
} else {
if let Some(index) = active_features.iter().position(|&f| f == event.feature) {
active_features.remove(index);
}
}
}
for chain_flags in &mut m.chain_flags {
if let Some(last) = chain_flags.last_mut() {
last.cluster_last = HB_FEATURE_GLOBAL_END;
}
}
}
}
#[derive(Copy, Clone, PartialEq, Eq, Default)]
pub struct FeatureInfo {
pub kind: u16,
pub setting: u16,
pub is_exclusive: bool,
}
impl Ord for FeatureInfo {
fn cmp(&self, other: &Self) -> Ordering {
self.partial_cmp(other).unwrap()
}
}
impl PartialOrd for FeatureInfo {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
if self.kind != other.kind {
Some(self.kind.cmp(&other.kind))
} else if !self.is_exclusive && (self.setting & !1) != (other.setting & !1) {
Some(self.setting.cmp(&other.setting))
} else {
Some(Ordering::Equal)
}
}
}
#[derive(Copy, Clone, PartialEq, Eq)]
pub struct FeatureRange {
pub info: FeatureInfo,
pub start: u32,
pub end: u32,
}
#[derive(Copy, Clone, Eq, PartialEq)]
struct FeatureEvent {
pub index: usize,
pub start: bool,
pub feature: FeatureInfo,
}
impl Ord for FeatureEvent {
fn cmp(&self, other: &Self) -> Ordering {
self.partial_cmp(other).unwrap()
}
}
impl PartialOrd for FeatureEvent {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
if self.index != other.index {
Some(self.index.cmp(&other.index))
} else if self.start != other.start {
Some(self.start.cmp(&other.start))
} else {
Some(Ordering::Equal)
}
}
}