use read_fonts::types::{F2Dot14, Fixed, GlyphId};
use read_fonts::{FontRef, TableProvider};
use smallvec::SmallVec;
#[cfg(not(feature = "std"))]
#[allow(unused_imports)]
use core_maths::CoreFloat as _;
use super::aat::AatTables;
use super::charmap::{cache_t as cmap_cache_t, Charmap};
use super::font_funcs::FontFuncsDispatch;
use super::glyph_metrics::GlyphMetrics;
use super::glyph_names::GlyphNames;
use super::ot::{LayoutTable, OtCache, OtTables};
use super::ot_layout::TableIndex;
use super::ot_shape::OtShapeContext;
use crate::hb::aat::AatCache;
use crate::hb::tables::TableRanges;
use crate::{script, Feature, GlyphBuffer, NormalizedCoord, ShapePlan, UnicodeBuffer, Variation};
pub use super::font_funcs::{AdvanceWidthBatch, BuiltinFontFuncs, FontFuncs, RawAdvanceWidthBatch};
pub struct ShaperData {
table_ranges: TableRanges,
ot_cache: OtCache,
aat_cache: AatCache,
cmap_cache: cmap_cache_t,
}
impl ShaperData {
pub fn new(font: &FontRef) -> Self {
let ot_cache = OtCache::new(font);
let aat_cache = AatCache::new(font);
let table_ranges = TableRanges::new(font);
let cmap_cache = cmap_cache_t::new();
Self {
table_ranges,
ot_cache,
aat_cache,
cmap_cache,
}
}
pub fn shaper<'a>(&'a self, font: &FontRef<'a>) -> ShaperBuilder<'a> {
ShaperBuilder {
data: self,
font: font.clone(),
instance: None,
}
}
}
const MAX_INLINE_COORDS: usize = 11;
#[derive(Clone, Default, Debug)]
pub struct ShaperInstance {
coords: SmallVec<[F2Dot14; MAX_INLINE_COORDS]>,
pub(crate) feature_variations: [Option<u32>; 2],
}
impl ShaperInstance {
pub fn from_variations<V>(font: &FontRef, variations: V) -> Self
where
V: IntoIterator,
V::Item: Into<Variation>,
{
let mut this = Self::default();
this.set_variations(font, variations);
this
}
pub fn from_coords(font: &FontRef, coords: impl IntoIterator<Item = NormalizedCoord>) -> Self {
let mut this = Self::default();
this.set_coords(font, coords);
this
}
pub fn from_named_instance(font: &FontRef, index: usize) -> Self {
let mut this = Self::default();
this.set_named_instance(font, index);
this
}
pub fn coords(&self) -> &[F2Dot14] {
&self.coords
}
pub fn set_variations<V>(&mut self, font: &FontRef, variations: V)
where
V: IntoIterator,
V::Item: Into<Variation>,
{
self.coords.clear();
if let Ok(fvar) = font.fvar() {
self.coords
.resize(fvar.axis_count() as usize, F2Dot14::ZERO);
fvar.user_to_normalized(
font.avar().ok().as_ref(),
variations
.into_iter()
.map(Into::into)
.map(|var| (var.tag, Fixed::from_f64(var.value as _))),
self.coords.as_mut_slice(),
);
self.check_default();
self.set_feature_variations(font);
}
}
pub fn set_coords(&mut self, font: &FontRef, coords: impl IntoIterator<Item = F2Dot14>) {
self.coords.clear();
if let Ok(fvar) = font.fvar() {
let count = fvar.axis_count() as usize;
self.coords.reserve(count);
self.coords.extend(coords.into_iter().take(count));
self.check_default();
self.set_feature_variations(font);
}
}
pub fn set_named_instance(&mut self, font: &FontRef, index: usize) {
self.coords.clear();
if let Ok(fvar) = font.fvar() {
if let Ok((axes, instance)) = fvar
.axis_instance_arrays()
.and_then(|arrays| Ok((arrays.axes(), arrays.instances().get(index)?)))
{
self.set_variations(
font,
axes.iter()
.zip(instance.coordinates)
.map(|(axis, coord)| (axis.axis_tag(), coord.get().to_f32())),
);
}
}
}
fn set_feature_variations(&mut self, font: &FontRef) {
self.feature_variations = [None; 2];
if self.coords.is_empty() {
return;
}
self.feature_variations[0] = font
.gsub()
.ok()
.and_then(|t| LayoutTable::Gsub(t).feature_variation_index(&self.coords));
self.feature_variations[1] = font
.gpos()
.ok()
.and_then(|t| LayoutTable::Gpos(t).feature_variation_index(&self.coords));
}
fn check_default(&mut self) {
if self.coords.iter().all(|coord| *coord == F2Dot14::ZERO) {
self.coords.clear();
}
}
}
pub struct ShaperBuilder<'a> {
data: &'a ShaperData,
font: FontRef<'a>,
instance: Option<&'a ShaperInstance>,
}
impl<'a> ShaperBuilder<'a> {
pub fn instance(mut self, instance: Option<&'a ShaperInstance>) -> Self {
self.instance = instance;
self
}
pub fn build(self) -> crate::Shaper<'a> {
let font = self.font;
let units_per_em = self.data.table_ranges.units_per_em;
let charmap = Charmap::new(&font, &self.data.table_ranges);
let glyph_metrics = GlyphMetrics::new(&font, &self.data.table_ranges);
let (coords, feature_variations) = self
.instance
.map(|instance| (instance.coords(), instance.feature_variations))
.unwrap_or_default();
let ot_tables = OtTables::new(
&font,
&self.data.ot_cache,
&self.data.table_ranges,
coords,
feature_variations,
);
let aat_tables = AatTables::new(&font, &self.data.aat_cache, &self.data.table_ranges);
hb_font_t {
font,
units_per_em,
charmap,
cmap_cache: &self.data.cmap_cache,
glyph_metrics,
ot_tables,
aat_tables,
}
}
}
#[derive(Default)]
pub struct ShapeOptions<'a> {
plan: Option<&'a ShapePlan>,
scale: Option<(i32, i32)>,
point_size: Option<f32>,
features: &'a [Feature],
font_funcs: Option<&'a mut (dyn FontFuncs + 'a)>,
}
impl<'a> ShapeOptions<'a> {
pub fn new() -> Self {
Self::default()
}
pub fn plan(mut self, plan: Option<&'a ShapePlan>) -> Self {
self.plan = plan;
self
}
pub fn scale(mut self, scale: Option<i32>) -> Self {
self.scale = scale.map(|s| (s, s));
self
}
pub fn scale_separate(mut self, scale: Option<(i32, i32)>) -> Self {
self.scale = scale;
self
}
pub fn point_size(mut self, point_size: Option<f32>) -> Self {
self.point_size = point_size;
self
}
pub fn features(mut self, features: &'a [Feature]) -> Self {
self.features = features;
self
}
pub fn font_funcs(mut self, funcs: Option<&'a mut (dyn FontFuncs + 'a)>) -> Self {
self.font_funcs = funcs;
self
}
}
#[derive(Copy, Clone)]
pub(crate) struct Scale {
x_mult: i64,
y_mult: i64,
x_multf: f32,
y_multf: f32,
}
impl Default for Scale {
fn default() -> Self {
Self {
x_mult: 1 << 16,
y_mult: 1 << 16,
x_multf: 1.0,
y_multf: 1.0,
}
}
}
#[allow(clippy::cast_precision_loss)]
impl Scale {
pub(crate) fn new(scale: Option<(i32, i32)>, upem: i32) -> Self {
let (Some((x_scale, y_scale)), true) = (scale, upem != 0) else {
return Self::default();
};
let [x_mult, y_mult] = [x_scale, y_scale].map(|s| Self::mult_from_scale(s, upem));
let upem = upem as f32;
Self {
x_mult,
y_mult,
x_multf: x_scale as f32 / upem,
y_multf: y_scale as f32 / upem,
}
}
#[inline(always)]
pub(crate) fn scale_x(&self, x: i32) -> i32 {
Self::scale_by_mult(x, self.x_mult)
}
#[inline(always)]
pub(crate) fn scale_y(&self, y: i32) -> i32 {
Self::scale_by_mult(y, self.y_mult)
}
pub(crate) fn scale_extents(&self, mut extents: GlyphExtents) -> GlyphExtents {
let x1 = extents.x_bearing as f32 * self.x_multf;
let y1 = extents.y_bearing as f32 * self.y_multf;
let x2 = (extents.x_bearing + extents.width) as f32 * self.x_multf;
let y2 = (extents.y_bearing + extents.height) as f32 * self.y_multf;
extents.x_bearing = x1.floor() as i32;
extents.y_bearing = y1.floor() as i32;
extents.width = x2.ceil() as i32 - extents.x_bearing;
extents.height = y2.ceil() as i32 - extents.y_bearing;
extents
}
#[inline(always)]
fn mult_from_scale(scale: i32, upem: i32) -> i64 {
if scale < 0 {
-((-(scale as i64)) << 16) / upem as i64
} else {
((scale as i64) << 16) / upem as i64
}
}
#[inline(always)]
fn scale_by_mult(value: i32, mult: i64) -> i32 {
(((value as i64) * mult + 32768) >> 16) as i32
}
}
#[derive(Clone)]
pub struct hb_font_t<'a> {
pub(crate) font: FontRef<'a>,
pub(crate) units_per_em: u16,
charmap: Charmap<'a>,
pub(crate) cmap_cache: &'a cmap_cache_t,
pub(crate) glyph_metrics: GlyphMetrics<'a>,
pub(crate) ot_tables: OtTables<'a>,
pub(crate) aat_tables: AatTables<'a>,
}
impl<'a> crate::Shaper<'a> {
#[inline]
pub fn units_per_em(&self) -> i32 {
self.units_per_em as i32
}
pub fn coords(&self) -> &'a [NormalizedCoord] {
self.ot_tables.coords
}
pub fn shape(&self, buffer: UnicodeBuffer, options: ShapeOptions<'_>) -> GlyphBuffer {
if let Some(plan) = options.plan {
self.shape_with_plan(plan, buffer, options)
} else {
let plan = ShapePlan::new(
self,
buffer.0.direction,
buffer.0.script,
buffer.0.language.as_ref(),
options.features,
);
self.shape_with_plan(&plan, buffer, options)
}
}
fn shape_with_plan(
&self,
plan: &ShapePlan,
buffer: UnicodeBuffer,
options: ShapeOptions<'_>,
) -> GlyphBuffer {
let mut buffer = buffer.0;
buffer.enter();
assert_eq!(
buffer.direction, plan.direction,
"Buffer direction does not match plan direction: {:?} != {:?}",
buffer.direction, plan.direction
);
assert_eq!(
buffer.script.unwrap_or(script::UNKNOWN),
plan.script.unwrap_or(script::UNKNOWN),
"Buffer script does not match plan script: {:?} != {:?}",
buffer.script.unwrap_or(script::UNKNOWN),
plan.script.unwrap_or(script::UNKNOWN)
);
if buffer.len > 0 {
let target_direction = buffer.direction;
let scale = Scale::new(options.scale, self.units_per_em as i32);
let mut font_funcs = FontFuncsDispatch::new(self, scale, options.font_funcs);
OtShapeContext {
plan,
face: self,
buffer: &mut buffer,
target_direction,
features: options.features,
point_size: options.point_size,
font_funcs: &mut font_funcs,
}
.shape_internal();
}
buffer.leave();
GlyphBuffer(buffer)
}
pub(crate) fn get_nominal_glyph(&self, c: u32) -> Option<GlyphId> {
self.charmap.map(c)
}
pub(crate) fn get_nominal_variant_glyph(&self, c: u32, vs: u32) -> Option<GlyphId> {
self.charmap.map_variant(c, vs)
}
pub(crate) fn glyph_h_advance(&self, glyph: GlyphId) -> i32 {
self.glyph_metrics
.advance_width(glyph, self.ot_tables.coords)
.unwrap_or_default()
}
pub(crate) fn glyph_v_advance(&self, glyph: GlyphId) -> i32 {
-self
.glyph_metrics
.advance_height(glyph, self.ot_tables.coords)
.unwrap_or(self.units_per_em as i32)
}
pub(crate) fn glyph_v_origin(&self, glyph: GlyphId) -> i32 {
self.glyph_metrics
.v_origin(glyph, self.ot_tables.coords)
.unwrap_or_default()
}
pub(crate) fn glyph_extents(&self, glyph: GlyphId, glyph_extents: &mut GlyphExtents) -> bool {
if let Some(extents) = self.glyph_metrics.extents(glyph, self.ot_tables.coords) {
glyph_extents.x_bearing = extents.x_min;
glyph_extents.y_bearing = extents.y_max;
glyph_extents.width = extents.x_max - extents.x_min;
glyph_extents.height = extents.y_min - extents.y_max;
true
} else {
false
}
}
pub(crate) fn glyph_names(&self) -> GlyphNames<'a> {
GlyphNames::new(&self.font)
}
pub(crate) fn layout_table(&self, table_index: TableIndex) -> Option<LayoutTable<'a>> {
match table_index {
TableIndex::GSUB => self
.ot_tables
.gsub
.as_ref()
.map(|table| LayoutTable::Gsub(table.table.clone())),
TableIndex::GPOS => self
.ot_tables
.gpos
.as_ref()
.map(|table| LayoutTable::Gpos(table.table.clone())),
}
}
pub(crate) fn layout_tables(&self) -> impl Iterator<Item = (TableIndex, LayoutTable<'a>)> + '_ {
TableIndex::iter().filter_map(move |idx| self.layout_table(idx).map(|table| (idx, table)))
}
}
#[derive(Clone, Copy, Default, bytemuck::Pod, bytemuck::Zeroable)]
#[repr(C)]
pub struct GlyphExtents {
pub x_bearing: i32,
pub y_bearing: i32,
pub width: i32,
pub height: i32,
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn extents_scale_from_corners_like_harfbuzz() {
let scale = Scale::new(Some((1500, 1500)), 1000);
let extents = GlyphExtents {
x_bearing: 1,
y_bearing: 4,
width: 3,
height: -2,
};
let scaled = scale.scale_extents(extents);
assert_eq!(scaled.x_bearing, 1);
assert_eq!(scaled.y_bearing, 6);
assert_eq!(scaled.width, 5);
assert_eq!(scaled.height, -3);
}
}