#[cfg(not(feature = "std"))]
use core_maths::CoreFloat;
use crate::hb::paint_extents::hb_paint_extents_context_t;
use ttf_parser::gdef::GlyphClass;
use ttf_parser::opentype_layout::LayoutTable;
use ttf_parser::{GlyphId, RgbaColor};
use super::buffer::GlyphPropsFlags;
use super::ot_layout::TableIndex;
use super::ot_layout_common::{PositioningTable, SubstitutionTable};
use crate::Variation;
const WINDOWS_SYMBOL_ENCODING: u16 = 0;
const WINDOWS_UNICODE_BMP_ENCODING: u16 = 1;
const WINDOWS_UNICODE_FULL_ENCODING: u16 = 10;
const UNICODE_1_0_ENCODING: u16 = 0;
const UNICODE_1_1_ENCODING: u16 = 1;
const UNICODE_ISO_ENCODING: u16 = 2;
const UNICODE_2_0_BMP_ENCODING: u16 = 3;
const UNICODE_2_0_FULL_ENCODING: u16 = 4;
const UNICODE_FULL_ENCODING: u16 = 6;
#[derive(Clone)]
pub struct hb_font_t<'a> {
pub(crate) ttfp_face: ttf_parser::Face<'a>,
pub(crate) units_per_em: u16,
pixels_per_em: Option<(u16, u16)>,
pub(crate) points_per_em: Option<f32>,
prefered_cmap_encoding_subtable: Option<u16>,
pub(crate) gsub: Option<SubstitutionTable<'a>>,
pub(crate) gpos: Option<PositioningTable<'a>>,
}
impl<'a> AsRef<ttf_parser::Face<'a>> for hb_font_t<'a> {
#[inline]
fn as_ref(&self) -> &ttf_parser::Face<'a> {
&self.ttfp_face
}
}
impl<'a> AsMut<ttf_parser::Face<'a>> for hb_font_t<'a> {
#[inline]
fn as_mut(&mut self) -> &mut ttf_parser::Face<'a> {
&mut self.ttfp_face
}
}
impl<'a> core::ops::Deref for hb_font_t<'a> {
type Target = ttf_parser::Face<'a>;
#[inline]
fn deref(&self) -> &Self::Target {
&self.ttfp_face
}
}
impl<'a> core::ops::DerefMut for hb_font_t<'a> {
#[inline]
fn deref_mut(&mut self) -> &mut Self::Target {
&mut self.ttfp_face
}
}
impl<'a> hb_font_t<'a> {
pub fn from_slice(data: &'a [u8], face_index: u32) -> Option<Self> {
let face = ttf_parser::Face::parse(data, face_index).ok()?;
Some(Self::from_face(face))
}
pub fn from_face(face: ttf_parser::Face<'a>) -> Self {
hb_font_t {
units_per_em: face.units_per_em(),
pixels_per_em: None,
points_per_em: None,
prefered_cmap_encoding_subtable: find_best_cmap_subtable(&face),
gsub: face.tables().gsub.map(SubstitutionTable::new),
gpos: face.tables().gpos.map(PositioningTable::new),
ttfp_face: face,
}
}
#[inline]
pub fn units_per_em(&self) -> i32 {
self.units_per_em as i32
}
#[inline]
pub(crate) fn pixels_per_em(&self) -> Option<(u16, u16)> {
self.pixels_per_em
}
#[inline]
pub fn set_pixels_per_em(&mut self, ppem: Option<(u16, u16)>) {
self.pixels_per_em = ppem;
}
#[inline]
pub fn set_points_per_em(&mut self, ptem: Option<f32>) {
self.points_per_em = ptem;
}
pub fn set_variations(&mut self, variations: &[Variation]) {
for variation in variations {
self.set_variation(variation.tag, variation.value);
}
}
pub(crate) fn has_glyph(&self, c: u32) -> bool {
self.get_nominal_glyph(c).is_some()
}
pub(crate) fn get_nominal_glyph(&self, mut c: u32) -> Option<GlyphId> {
let subtable_idx = self.prefered_cmap_encoding_subtable?;
let subtable = self.tables().cmap?.subtables.get(subtable_idx)?;
if subtable.platform_id == ttf_parser::PlatformId::Macintosh && c > 0x7F {
c = unicode_to_macroman(c);
}
match subtable.glyph_index(c) {
Some(gid) => Some(gid),
None => {
if subtable.platform_id == ttf_parser::PlatformId::Windows
&& subtable.encoding_id == WINDOWS_SYMBOL_ENCODING
{
if c <= 0x00FF {
return self.get_nominal_glyph(0xF000 + c);
}
}
None
}
}
}
pub(crate) fn glyph_h_advance(&self, glyph: GlyphId) -> i32 {
self.glyph_advance(glyph, false) as i32
}
pub(crate) fn glyph_v_advance(&self, glyph: GlyphId) -> i32 {
-(self.glyph_advance(glyph, true) as i32)
}
fn glyph_advance(&self, glyph: GlyphId, is_vertical: bool) -> u32 {
let face = &self.ttfp_face;
if face.is_variable()
&& face.has_non_default_variation_coordinates()
&& face.tables().hvar.is_none()
&& face.tables().vvar.is_none()
&& face.glyph_phantom_points(glyph).is_none()
{
return match face.glyph_bounding_box(glyph) {
Some(bbox) => {
(if is_vertical {
bbox.y_max + bbox.y_min
} else {
bbox.x_max + bbox.x_min
}) as u32
}
None => 0,
};
}
if is_vertical {
if face.tables().vmtx.is_some() {
face.glyph_ver_advance(glyph).unwrap_or(0) as u32
} else {
(face.ascender() - face.descender()) as u32
}
} else if !is_vertical && face.tables().hmtx.is_some() {
face.glyph_hor_advance(glyph).unwrap_or(0) as u32
} else {
face.units_per_em() as u32
}
}
pub(crate) fn glyph_h_origin(&self, glyph: GlyphId) -> i32 {
self.glyph_h_advance(glyph) / 2
}
pub(crate) fn glyph_v_origin(&self, glyph: GlyphId) -> i32 {
match self.ttfp_face.glyph_y_origin(glyph) {
Some(y) => i32::from(y),
None => {
let mut extents = hb_glyph_extents_t::default();
if self.glyph_extents(glyph, &mut extents) {
if self.ttfp_face.tables().vmtx.is_some() {
extents.y_bearing + self.glyph_side_bearing(glyph, true)
} else {
let advance = self.ttfp_face.ascender() - self.ttfp_face.descender();
let diff = advance as i32 - -extents.height;
extents.y_bearing + (diff >> 1)
}
} else {
self.ttfp_face.ascender() as i32
}
}
}
}
pub(crate) fn glyph_side_bearing(&self, glyph: GlyphId, is_vertical: bool) -> i32 {
let face = &self.ttfp_face;
if face.is_variable() && face.tables().hvar.is_none() && face.tables().vvar.is_none() {
return match face.glyph_bounding_box(glyph) {
Some(bbox) => (if is_vertical { bbox.x_min } else { bbox.y_min }) as i32,
None => 0,
};
}
if is_vertical {
face.glyph_ver_side_bearing(glyph).unwrap_or(0) as i32
} else {
face.glyph_hor_side_bearing(glyph).unwrap_or(0) as i32
}
}
pub(crate) fn glyph_extents(
&self,
glyph: GlyphId,
glyph_extents: &mut hb_glyph_extents_t,
) -> bool {
let pixels_per_em = self.pixels_per_em.map_or(u16::MAX, |ppem| ppem.0);
if let Some(img) = self.ttfp_face.glyph_raster_image(glyph, pixels_per_em) {
if img.format == ttf_parser::RasterImageFormat::PNG {
let scale = self.units_per_em as f32 / img.pixels_per_em as f32;
glyph_extents.x_bearing = (f32::from(img.x) * scale).round() as i32;
glyph_extents.y_bearing =
((f32::from(img.y) + f32::from(img.height)) * scale).round() as i32;
glyph_extents.width = (f32::from(img.width) * scale).round() as i32;
glyph_extents.height = (-f32::from(img.height) * scale).round() as i32;
return true;
}
} else if let Some(colr) = self.ttfp_face.tables().colr {
if colr.is_simple() {
return false;
}
if let Some(clip_box) = colr.clip_box(glyph, self.variation_coordinates()) {
glyph_extents.x_bearing = (clip_box.x_min).round() as i32;
glyph_extents.y_bearing = (clip_box.y_max).round() as i32;
glyph_extents.width = (clip_box.x_max - clip_box.x_min).round() as i32;
glyph_extents.height = (clip_box.y_min - clip_box.y_max).round() as i32;
return true;
}
let mut extents_data = hb_paint_extents_context_t::new(&self.ttfp_face);
let ret = colr
.paint(
glyph,
0,
&mut extents_data,
self.variation_coordinates(),
RgbaColor::new(0, 0, 0, 0),
)
.is_some();
let e = extents_data.get_extents();
if e.is_void() {
glyph_extents.x_bearing = 0;
glyph_extents.y_bearing = 0;
glyph_extents.width = 0;
glyph_extents.height = 0;
} else {
glyph_extents.x_bearing = e.x_min as i32;
glyph_extents.y_bearing = e.y_max as i32;
glyph_extents.width = (e.x_max - e.x_min) as i32;
glyph_extents.height = (e.y_min - e.y_max) as i32;
}
return ret;
}
let mut bbox = None;
if let Some(glyf) = self.ttfp_face.tables().glyf {
bbox = glyf.bbox(glyph);
}
if self.ttfp_face.tables().glyf.is_some() && bbox.is_none() {
return true;
}
let Some(bbox) = bbox else {
return false;
};
glyph_extents.x_bearing = i32::from(bbox.x_min);
glyph_extents.y_bearing = i32::from(bbox.y_max);
glyph_extents.width = i32::from(bbox.width());
glyph_extents.height = i32::from(bbox.y_min - bbox.y_max);
true
}
pub(crate) fn glyph_name(&self, glyph: GlyphId) -> Option<&str> {
self.ttfp_face.glyph_name(glyph)
}
pub(crate) fn glyph_props(&self, glyph: GlyphId) -> u16 {
let table = match self.tables().gdef {
Some(v) => v,
None => return 0,
};
match table.glyph_class(glyph) {
Some(GlyphClass::Base) => GlyphPropsFlags::BASE_GLYPH.bits(),
Some(GlyphClass::Ligature) => GlyphPropsFlags::LIGATURE.bits(),
Some(GlyphClass::Mark) => {
let class = table.glyph_mark_attachment_class(glyph);
(class << 8) | GlyphPropsFlags::MARK.bits()
}
_ => 0,
}
}
pub(crate) fn layout_table(&self, table_index: TableIndex) -> Option<&LayoutTable<'a>> {
match table_index {
TableIndex::GSUB => self.gsub.as_ref().map(|table| &table.inner),
TableIndex::GPOS => self.gpos.as_ref().map(|table| &table.inner),
}
}
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)]
#[repr(C)]
pub struct hb_glyph_extents_t {
pub x_bearing: i32,
pub y_bearing: i32,
pub width: i32,
pub height: i32,
}
unsafe impl bytemuck::Zeroable for hb_glyph_extents_t {}
unsafe impl bytemuck::Pod for hb_glyph_extents_t {}
fn find_best_cmap_subtable(face: &ttf_parser::Face) -> Option<u16> {
use ttf_parser::PlatformId;
find_cmap_subtable(face, PlatformId::Windows, WINDOWS_SYMBOL_ENCODING)
.or_else(|| find_cmap_subtable(face, PlatformId::Windows, WINDOWS_UNICODE_FULL_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Unicode, UNICODE_FULL_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Unicode, UNICODE_2_0_FULL_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Windows, WINDOWS_UNICODE_BMP_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Unicode, UNICODE_2_0_BMP_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Unicode, UNICODE_ISO_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Unicode, UNICODE_1_1_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Unicode, UNICODE_1_0_ENCODING))
.or_else(|| find_cmap_subtable(face, PlatformId::Macintosh, 0))
}
fn find_cmap_subtable(
face: &ttf_parser::Face,
platform_id: ttf_parser::PlatformId,
encoding_id: u16,
) -> Option<u16> {
for (i, subtable) in face.tables().cmap?.subtables.into_iter().enumerate() {
if subtable.platform_id == platform_id && subtable.encoding_id == encoding_id {
return Some(i as u16);
}
}
None
}
#[rustfmt::skip]
static UNICODE_TO_MACROMAN: &[u16] = &[
0x00C4, 0x00C5, 0x00C7, 0x00C9, 0x00D1, 0x00D6, 0x00DC, 0x00E1,
0x00E0, 0x00E2, 0x00E4, 0x00E3, 0x00E5, 0x00E7, 0x00E9, 0x00E8,
0x00EA, 0x00EB, 0x00ED, 0x00EC, 0x00EE, 0x00EF, 0x00F1, 0x00F3,
0x00F2, 0x00F4, 0x00F6, 0x00F5, 0x00FA, 0x00F9, 0x00FB, 0x00FC,
0x2020, 0x00B0, 0x00A2, 0x00A3, 0x00A7, 0x2022, 0x00B6, 0x00DF,
0x00AE, 0x00A9, 0x2122, 0x00B4, 0x00A8, 0x2260, 0x00C6, 0x00D8,
0x221E, 0x00B1, 0x2264, 0x2265, 0x00A5, 0x00B5, 0x2202, 0x2211,
0x220F, 0x03C0, 0x222B, 0x00AA, 0x00BA, 0x03A9, 0x00E6, 0x00F8,
0x00BF, 0x00A1, 0x00AC, 0x221A, 0x0192, 0x2248, 0x2206, 0x00AB,
0x00BB, 0x2026, 0x00A0, 0x00C0, 0x00C3, 0x00D5, 0x0152, 0x0153,
0x2013, 0x2014, 0x201C, 0x201D, 0x2018, 0x2019, 0x00F7, 0x25CA,
0x00FF, 0x0178, 0x2044, 0x20AC, 0x2039, 0x203A, 0xFB01, 0xFB02,
0x2021, 0x00B7, 0x201A, 0x201E, 0x2030, 0x00C2, 0x00CA, 0x00C1,
0x00CB, 0x00C8, 0x00CD, 0x00CE, 0x00CF, 0x00CC, 0x00D3, 0x00D4,
0xF8FF, 0x00D2, 0x00DA, 0x00DB, 0x00D9, 0x0131, 0x02C6, 0x02DC,
0x00AF, 0x02D8, 0x02D9, 0x02DA, 0x00B8, 0x02DD, 0x02DB, 0x02C7,
];
fn unicode_to_macroman(c: u32) -> u32 {
let u = c as u16;
let Some(index) = UNICODE_TO_MACROMAN.iter().position(|m| *m == u) else {
return 0;
};
(0x80 + index) as u32
}