use std::fmt::{self, Debug, Formatter};
use az::SaturatingAs;
use comemo::Tracked;
use rustybuzz::{BufferFlags, UnicodeBuffer};
use ttf_parser::GlyphId;
use ttf_parser::math::{GlyphAssembly, GlyphConstruction, GlyphPart};
use typst_library::World;
use typst_library::diag::{At, HintedStrResult, SourceResult, bail, warning};
use typst_library::foundations::{Repr, StyleChain};
use typst_library::introspection::Tag;
use typst_library::layout::{
Abs, Axes, Axis, Corner, Em, Frame, FrameItem, Point, Size, VAlignment,
};
use typst_library::math::{EquationElem, MathSize};
use typst_library::text::{
Font, FontFamily, FontVariant, Glyph, TextElem, TextItem, features, language, variant,
};
use typst_library::visualize::Paint;
use typst_syntax::Span;
use typst_utils::{Get, default_math_class};
use unicode_math_class::MathClass;
use unicode_segmentation::UnicodeSegmentation;
use super::{MathContext, families};
use crate::inline::create_shape_plan;
use crate::modifiers::{FrameModifiers, FrameModify};
const MAX_REPEATS: usize = 1024;
#[allow(clippy::large_enum_variant)]
#[derive(Debug, Clone)]
pub enum MathFragment {
Glyph(GlyphFragment),
Frame(FrameFragment),
Spacing(Abs, bool),
Space(Abs),
Linebreak,
Align,
Tag(Tag),
}
impl MathFragment {
pub fn size(&self) -> Size {
match self {
Self::Glyph(glyph) => glyph.size,
Self::Frame(fragment) => fragment.frame.size(),
Self::Spacing(amount, _) => Size::with_x(*amount),
Self::Space(amount) => Size::with_x(*amount),
_ => Size::zero(),
}
}
pub fn width(&self) -> Abs {
match self {
Self::Glyph(glyph) => glyph.size.x,
Self::Frame(fragment) => fragment.frame.width(),
Self::Spacing(amount, _) => *amount,
Self::Space(amount) => *amount,
_ => Abs::zero(),
}
}
pub fn height(&self) -> Abs {
match self {
Self::Glyph(glyph) => glyph.size.y,
Self::Frame(fragment) => fragment.frame.height(),
_ => Abs::zero(),
}
}
pub fn ascent(&self) -> Abs {
match self {
Self::Glyph(glyph) => glyph.ascent(),
Self::Frame(fragment) => fragment.frame.ascent(),
_ => Abs::zero(),
}
}
pub fn descent(&self) -> Abs {
match self {
Self::Glyph(glyph) => glyph.descent(),
Self::Frame(fragment) => fragment.frame.descent(),
_ => Abs::zero(),
}
}
pub fn is_ignorant(&self) -> bool {
match self {
Self::Frame(fragment) => fragment.ignorant,
Self::Tag(_) => true,
_ => false,
}
}
pub fn class(&self) -> MathClass {
match self {
Self::Glyph(glyph) => glyph.class,
Self::Frame(fragment) => fragment.class,
Self::Spacing(_, _) => MathClass::Space,
Self::Space(_) => MathClass::Space,
Self::Linebreak => MathClass::Space,
Self::Align => MathClass::Special,
Self::Tag(_) => MathClass::Special,
}
}
pub fn math_size(&self) -> Option<MathSize> {
match self {
Self::Glyph(glyph) => Some(glyph.math_size),
Self::Frame(fragment) => Some(fragment.math_size),
_ => None,
}
}
#[inline]
pub fn font(&self, ctx: &MathContext, styles: StyleChain) -> (Font, Abs) {
(
match self {
Self::Glyph(glyph) => glyph.item.font.clone(),
_ => ctx.font().clone(),
},
self.font_size().unwrap_or_else(|| styles.resolve(TextElem::size)),
)
}
pub fn font_size(&self) -> Option<Abs> {
match self {
Self::Glyph(glyph) => Some(glyph.item.size),
Self::Frame(fragment) => Some(fragment.font_size),
_ => None,
}
}
pub fn set_class(&mut self, class: MathClass) {
match self {
Self::Glyph(glyph) => glyph.class = class,
Self::Frame(fragment) => fragment.class = class,
_ => {}
}
}
pub fn set_limits(&mut self, limits: Limits) {
match self {
Self::Glyph(glyph) => glyph.limits = limits,
Self::Frame(fragment) => fragment.limits = limits,
_ => {}
}
}
pub fn is_spaced(&self) -> bool {
if self.class() == MathClass::Fence {
return true;
}
matches!(
self,
MathFragment::Frame(FrameFragment {
spaced: true,
class: MathClass::Normal | MathClass::Alphabetic,
..
})
)
}
pub fn is_text_like(&self) -> bool {
match self {
Self::Glyph(glyph) => !glyph.extended_shape,
MathFragment::Frame(frame) => frame.text_like,
_ => false,
}
}
pub fn italics_correction(&self) -> Abs {
match self {
Self::Glyph(glyph) => glyph.italics_correction,
Self::Frame(fragment) => fragment.italics_correction,
_ => Abs::zero(),
}
}
pub fn accent_attach(&self) -> (Abs, Abs) {
match self {
Self::Glyph(glyph) => glyph.accent_attach,
Self::Frame(fragment) => fragment.accent_attach,
_ => (self.width() / 2.0, self.width() / 2.0),
}
}
pub fn into_frame(self) -> Frame {
match self {
Self::Glyph(glyph) => glyph.into_frame(),
Self::Frame(fragment) => fragment.frame,
Self::Tag(tag) => {
let mut frame = Frame::soft(Size::zero());
frame.push(Point::zero(), FrameItem::Tag(tag));
frame
}
_ => Frame::soft(self.size()),
}
}
pub fn limits(&self) -> Limits {
match self {
MathFragment::Glyph(glyph) => glyph.limits,
MathFragment::Frame(fragment) => fragment.limits,
_ => Limits::Never,
}
}
pub fn fill(&self) -> Option<Paint> {
match self {
Self::Glyph(glyph) => Some(glyph.item.fill.clone()),
_ => None,
}
}
pub fn stretch_vertical(
&mut self,
ctx: &mut MathContext,
height: Abs,
short_fall: Abs,
) {
if let Self::Glyph(glyph) = self {
glyph.stretch_vertical(ctx, height, short_fall)
}
}
pub fn stretch_horizontal(
&mut self,
ctx: &mut MathContext,
width: Abs,
short_fall: Abs,
) {
if let Self::Glyph(glyph) = self {
glyph.stretch_horizontal(ctx, width, short_fall)
}
}
pub fn center_on_axis(&mut self) {
if let Self::Glyph(glyph) = self {
glyph.center_on_axis()
}
}
pub fn kern_at_height(&self, corner: Corner, height: Abs) -> Abs {
match self {
Self::Glyph(glyph) => {
let is_vertical =
glyph.item.glyphs.iter().all(|glyph| glyph.y_advance != Em::zero());
let glyph_index = match (is_vertical, corner) {
(true, Corner::TopLeft | Corner::TopRight) => {
glyph.item.glyphs.len() - 1
}
(false, Corner::TopRight | Corner::BottomRight) => {
glyph.item.glyphs.len() - 1
}
_ => 0,
};
kern_at_height(
&glyph.item.font,
GlyphId(glyph.item.glyphs[glyph_index].id),
corner,
Em::from_abs(height, glyph.item.size),
)
.unwrap_or_default()
.at(glyph.item.size)
}
_ => Abs::zero(),
}
}
}
impl From<GlyphFragment> for MathFragment {
fn from(glyph: GlyphFragment) -> Self {
Self::Glyph(glyph)
}
}
impl From<FrameFragment> for MathFragment {
fn from(fragment: FrameFragment) -> Self {
Self::Frame(fragment)
}
}
#[derive(Clone)]
pub struct GlyphFragment {
pub item: TextItem,
pub base_glyph: Glyph,
pub size: Size,
pub baseline: Option<Abs>,
pub italics_correction: Abs,
pub accent_attach: (Abs, Abs),
pub math_size: MathSize,
pub class: MathClass,
pub limits: Limits,
pub extended_shape: bool,
pub mid_stretched: Option<bool>,
pub modifiers: FrameModifiers,
pub shift: Abs,
pub align: Abs,
}
impl GlyphFragment {
pub fn new_char(
ctx: &MathContext,
styles: StyleChain,
c: char,
span: Span,
) -> SourceResult<Option<Self>> {
Self::new(ctx.engine.world, styles, c.encode_utf8(&mut [0; 4]), span)
}
#[comemo::memoize]
pub fn new(
world: Tracked<dyn World + '_>,
styles: StyleChain,
text: &str,
span: Span,
) -> SourceResult<Option<GlyphFragment>> {
assert!(text.graphemes(true).count() == 1);
let Some((c, font, mut glyph)) = shape(
world,
variant(styles),
features(styles),
language(styles),
styles.get(TextElem::fallback),
text,
families(styles).collect(),
)
.at(span)?
else {
return Ok(None);
};
glyph.span.0 = span;
let limits = Limits::for_char(c);
let class = styles
.get(EquationElem::class)
.or_else(|| default_math_class(c))
.unwrap_or(MathClass::Normal);
let item = TextItem {
font,
size: styles.resolve(TextElem::size),
fill: styles.get_ref(TextElem::fill).as_decoration(),
stroke: styles.resolve(TextElem::stroke).map(|s| s.unwrap_or_default()),
lang: styles.get(TextElem::lang),
region: styles.get(TextElem::region),
text: text.into(),
glyphs: vec![glyph.clone()],
};
let mut fragment = Self {
item,
base_glyph: glyph,
math_size: styles.get(EquationElem::size),
class,
limits,
mid_stretched: None,
extended_shape: false,
italics_correction: Abs::zero(),
accent_attach: (Abs::zero(), Abs::zero()),
size: Size::zero(),
baseline: None,
align: Abs::zero(),
shift: styles.resolve(TextElem::baseline),
modifiers: FrameModifiers::get_in(styles),
};
fragment.update_glyph();
Ok(Some(fragment))
}
pub fn update_glyph(&mut self) {
let id = GlyphId(self.item.glyphs[0].id);
let extended_shape = is_extended_shape(&self.item.font, id);
let italics = italics_correction(&self.item.font, id).unwrap_or_default();
let width = self.item.width();
if !extended_shape {
self.item.glyphs[0].x_advance += italics;
}
let italics = italics.at(self.item.size);
let (ascent, descent) =
ascent_descent(&self.item.font, id).unwrap_or((Em::zero(), Em::zero()));
let top_accent_attach = accent_attach(&self.item.font, id)
.map(|x| x.at(self.item.size))
.unwrap_or((width + italics) / 2.0);
let bottom_accent_attach = (width - italics) / 2.0;
self.baseline = Some(ascent.at(self.item.size));
self.size = Size::new(
self.item.width(),
ascent.at(self.item.size) + descent.at(self.item.size),
);
self.italics_correction = italics;
self.accent_attach = (top_accent_attach, bottom_accent_attach);
self.extended_shape = extended_shape;
}
pub fn reset_glyph(&mut self) {
self.align = Abs::zero();
self.item.glyphs = vec![self.base_glyph.clone()];
self.update_glyph();
}
pub fn baseline(&self) -> Abs {
self.ascent()
}
pub fn ascent(&self) -> Abs {
self.baseline.unwrap_or(self.size.y)
}
pub fn descent(&self) -> Abs {
self.size.y - self.ascent()
}
pub fn into_frame(self) -> Frame {
let mut frame = Frame::soft(self.size);
frame.set_baseline(self.baseline());
frame.push(
Point::with_y(self.ascent() + self.shift + self.align),
FrameItem::Text(self.item),
);
frame.modify(&self.modifiers);
frame
}
pub fn stretch_vertical(
&mut self,
ctx: &mut MathContext,
height: Abs,
short_fall: Abs,
) {
self.stretch(ctx, height, short_fall, Axis::Y)
}
pub fn stretch_horizontal(
&mut self,
ctx: &mut MathContext,
width: Abs,
short_fall: Abs,
) {
self.stretch(ctx, width, short_fall, Axis::X)
}
pub fn stretch(
&mut self,
ctx: &mut MathContext,
target: Abs,
short_fall: Abs,
axis: Axis,
) {
self.reset_glyph();
let mut advance = self.size.get(axis);
if axis == Axis::X && !self.extended_shape {
advance -= self.italics_correction;
}
let short_target = target - short_fall;
if short_target <= advance {
return;
}
let id = GlyphId(self.item.glyphs[0].id);
let font = self.item.font.clone();
let Some(construction) = glyph_construction(&font, id, axis) else { return };
let mut best_id = id;
let mut best_advance = advance;
for variant in construction.variants {
best_id = variant.variant_glyph;
best_advance =
self.item.font.to_em(variant.advance_measurement).at(self.item.size);
if short_target <= best_advance {
break;
}
}
if short_target <= best_advance || construction.assembly.is_none() {
self.item.glyphs[0].id = best_id.0;
self.item.glyphs[0].x_advance =
self.item.font.x_advance(best_id.0).unwrap_or_default();
self.item.glyphs[0].x_offset = Em::zero();
self.item.glyphs[0].y_advance =
self.item.font.y_advance(best_id.0).unwrap_or_default();
self.item.glyphs[0].y_offset = Em::zero();
self.update_glyph();
return;
}
let assembly = construction.assembly.unwrap();
let min_overlap = min_connector_overlap(&self.item.font)
.unwrap_or_default()
.at(self.item.size);
assemble(ctx, self, assembly, min_overlap, target, axis);
}
pub fn center_on_axis(&mut self) {
self.align_on_axis(VAlignment::Horizon);
}
pub fn align_on_axis(&mut self, align: VAlignment) {
let h = self.size.y;
let axis = self.item.font.math().axis_height.at(self.item.size);
self.align += self.baseline();
self.baseline = Some(align.inv().position(h + axis * 2.0));
self.align -= self.baseline();
}
}
impl Debug for GlyphFragment {
fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
write!(f, "GlyphFragment({:?})", self.item.text)
}
}
#[derive(Debug, Clone)]
pub struct FrameFragment {
pub frame: Frame,
pub font_size: Abs,
pub class: MathClass,
pub math_size: MathSize,
pub limits: Limits,
pub spaced: bool,
pub base_ascent: Abs,
pub base_descent: Abs,
pub italics_correction: Abs,
pub accent_attach: (Abs, Abs),
pub text_like: bool,
pub ignorant: bool,
}
impl FrameFragment {
pub fn new(styles: StyleChain, frame: Frame) -> Self {
let base_ascent = frame.ascent();
let base_descent = frame.descent();
let accent_attach = frame.width() / 2.0;
Self {
frame: frame.modified(&FrameModifiers::get_in(styles)),
font_size: styles.resolve(TextElem::size),
class: styles.get(EquationElem::class).unwrap_or(MathClass::Normal),
math_size: styles.get(EquationElem::size),
limits: Limits::Never,
spaced: false,
base_ascent,
base_descent,
italics_correction: Abs::zero(),
accent_attach: (accent_attach, accent_attach),
text_like: false,
ignorant: false,
}
}
pub fn with_class(self, class: MathClass) -> Self {
Self { class, ..self }
}
pub fn with_limits(self, limits: Limits) -> Self {
Self { limits, ..self }
}
pub fn with_spaced(self, spaced: bool) -> Self {
Self { spaced, ..self }
}
pub fn with_base_ascent(self, base_ascent: Abs) -> Self {
Self { base_ascent, ..self }
}
pub fn with_base_descent(self, base_descent: Abs) -> Self {
Self { base_descent, ..self }
}
pub fn with_italics_correction(self, italics_correction: Abs) -> Self {
Self { italics_correction, ..self }
}
pub fn with_accent_attach(self, accent_attach: (Abs, Abs)) -> Self {
Self { accent_attach, ..self }
}
pub fn with_text_like(self, text_like: bool) -> Self {
Self { text_like, ..self }
}
pub fn with_ignorant(self, ignorant: bool) -> Self {
Self { ignorant, ..self }
}
}
fn ascent_descent(font: &Font, id: GlyphId) -> Option<(Em, Em)> {
let bbox = font.ttf().glyph_bounding_box(id)?;
Some((font.to_em(bbox.y_max), -font.to_em(bbox.y_min)))
}
fn italics_correction(font: &Font, id: GlyphId) -> Option<Em> {
font.ttf()
.tables()
.math?
.glyph_info?
.italic_corrections?
.get(id)
.map(|value| font.to_em(value.value))
}
fn accent_attach(font: &Font, id: GlyphId) -> Option<Em> {
font.ttf()
.tables()
.math?
.glyph_info?
.top_accent_attachments?
.get(id)
.map(|value| font.to_em(value.value))
}
fn is_extended_shape(font: &Font, id: GlyphId) -> bool {
font.ttf()
.tables()
.math
.and_then(|math| math.glyph_info)
.and_then(|glyph_info| glyph_info.extended_shapes)
.and_then(|coverage| coverage.get(id))
.is_some()
}
fn kern_at_height(font: &Font, id: GlyphId, corner: Corner, height: Em) -> Option<Em> {
let kerns = font.ttf().tables().math?.glyph_info?.kern_infos?.get(id)?;
let kern = match corner {
Corner::TopLeft => kerns.top_left,
Corner::TopRight => kerns.top_right,
Corner::BottomRight => kerns.bottom_right,
Corner::BottomLeft => kerns.bottom_left,
}?;
let mut i = 0;
while i < kern.count() && height > font.to_em(kern.height(i)?.value) {
i += 1;
}
Some(font.to_em(kern.kern(i)?.value))
}
pub fn stretch_axes(font: &Font, id: u16) -> Axes<bool> {
let id = GlyphId(id);
let horizontal = font
.ttf()
.tables()
.math
.and_then(|math| math.variants)
.and_then(|variants| variants.horizontal_constructions.get(id))
.is_some();
let vertical = font
.ttf()
.tables()
.math
.and_then(|math| math.variants)
.and_then(|variants| variants.vertical_constructions.get(id))
.is_some();
Axes::new(horizontal, vertical)
}
fn min_connector_overlap(font: &Font) -> Option<Em> {
font.ttf()
.tables()
.math?
.variants
.map(|variants| font.to_em(variants.min_connector_overlap))
}
fn glyph_construction(
font: &Font,
id: GlyphId,
axis: Axis,
) -> Option<GlyphConstruction<'_>> {
font.ttf()
.tables()
.math?
.variants
.map(|variants| match axis {
Axis::X => variants.horizontal_constructions,
Axis::Y => variants.vertical_constructions,
})?
.get(id)
}
fn assemble(
ctx: &mut MathContext,
base: &mut GlyphFragment,
assembly: GlyphAssembly,
min_overlap: Abs,
target: Abs,
axis: Axis,
) {
let mut full;
let mut ratio;
let mut repeat = 0;
loop {
full = Abs::zero();
ratio = 0.0;
let mut parts = parts(assembly, repeat).peekable();
let mut growable = Abs::zero();
while let Some(part) = parts.next() {
let mut advance = base.item.font.to_em(part.full_advance).at(base.item.size);
if let Some(next) = parts.peek() {
let max_overlap = base
.item
.font
.to_em(part.end_connector_length.min(next.start_connector_length))
.at(base.item.size);
if max_overlap < min_overlap {
ctx.engine.sink.warn(warning!(
base.item.glyphs[0].span.0,
"glyph has assembly parts with overlap less than minConnectorOverlap";
hint: "its rendering may appear broken - this is probably a font bug";
hint: "please file an issue at https://github.com/typst/typst/issues"
));
}
advance -= max_overlap;
growable += (max_overlap - min_overlap).max(Abs::zero());
}
full += advance;
}
if full < target {
let delta = target - full;
ratio = (delta / growable).min(1.0);
full += ratio * growable;
}
if target <= full || repeat >= MAX_REPEATS {
break;
}
repeat += 1;
}
let mut glyphs = vec![];
let mut parts = parts(assembly, repeat).peekable();
while let Some(part) = parts.next() {
let mut advance = base.item.font.to_em(part.full_advance).at(base.item.size);
if let Some(next) = parts.peek() {
let max_overlap = base
.item
.font
.to_em(part.end_connector_length.min(next.start_connector_length))
.at(base.item.size);
advance -= max_overlap;
advance += ratio * (max_overlap - min_overlap);
}
let (x_advance, y_advance, y_offset) = match axis {
Axis::X => (Em::from_abs(advance, base.item.size), Em::zero(), Em::zero()),
Axis::Y => (
Em::zero(),
Em::from_abs(advance, base.item.size),
ascent_descent(&base.item.font, part.glyph_id)
.map(|x| x.1)
.unwrap_or_default(),
),
};
glyphs.push(Glyph {
id: part.glyph_id.0,
x_advance,
x_offset: Em::zero(),
y_advance,
y_offset,
..base.item.glyphs[0].clone()
});
}
match axis {
Axis::X => {
base.size.x = full;
let (ascent, descent) = glyphs
.iter()
.filter_map(|glyph| ascent_descent(&base.item.font, GlyphId(glyph.id)))
.reduce(|(ma, md), (a, d)| (ma.max(a), md.max(d)))
.unwrap_or((Em::zero(), Em::zero()));
base.baseline = Some(ascent.at(base.item.size));
base.size.y = (ascent + descent).at(base.item.size);
}
Axis::Y => {
base.baseline = None;
base.size.y = full;
base.size.x = glyphs
.iter()
.map(|glyph| base.item.font.x_advance(glyph.id).unwrap_or_default())
.max()
.unwrap_or_default()
.at(base.item.size);
}
}
base.item.glyphs = glyphs;
base.italics_correction = base
.item
.font
.to_em(assembly.italics_correction.value)
.at(base.item.size);
if axis == Axis::X {
base.accent_attach = (full / 2.0, full / 2.0);
}
base.mid_stretched = None;
base.extended_shape = true;
}
fn parts(
assembly: GlyphAssembly<'_>,
repeat: usize,
) -> impl Iterator<Item = GlyphPart> + '_ {
assembly.parts.into_iter().flat_map(move |part| {
let count = if part.part_flags.extender() { repeat } else { 1 };
std::iter::repeat_n(part, count)
})
}
pub fn has_dtls_feat(font: &Font) -> bool {
font.ttf()
.tables()
.gsub
.and_then(|gsub| gsub.features.index(ttf_parser::Tag::from_bytes(b"dtls")))
.is_some()
}
#[comemo::memoize]
fn shape(
world: Tracked<dyn World + '_>,
variant: FontVariant,
features: Vec<rustybuzz::Feature>,
language: rustybuzz::Language,
fallback: bool,
text: &str,
families: Vec<&FontFamily>,
) -> HintedStrResult<Option<(char, Font, Glyph)>> {
let mut used = vec![];
let buffer = UnicodeBuffer::new();
shape_glyph(
world,
&mut used,
buffer,
variant,
features,
language,
fallback,
text,
families.into_iter(),
)
}
#[allow(clippy::too_many_arguments)]
fn shape_glyph<'a>(
world: Tracked<'a, dyn World + 'a>,
used: &mut Vec<Font>,
mut buffer: rustybuzz::UnicodeBuffer,
variant: FontVariant,
features: Vec<rustybuzz::Feature>,
language: rustybuzz::Language,
fallback: bool,
text: &str,
mut families: impl Iterator<Item = &'a FontFamily> + Clone,
) -> HintedStrResult<Option<(char, Font, Glyph)>> {
let book = world.book();
let mut selection = None;
let mut covers = None;
for family in families.by_ref() {
selection = book
.select(family.as_str(), variant)
.and_then(|id| world.font(id))
.filter(|font| !used.contains(font));
if selection.is_some() {
covers = family.covers();
break;
}
}
if selection.is_none() && fallback {
let first = used.first().map(Font::info);
selection = book
.select_fallback(first, variant, text)
.and_then(|id| world.font(id))
.filter(|font| !used.contains(font))
}
let Some(font) = selection else {
if let Some(font) = used.first().cloned() {
let glyph = Glyph {
id: 0,
x_advance: font.x_advance(0).unwrap_or_default(),
x_offset: Em::zero(),
y_advance: Em::zero(),
y_offset: Em::zero(),
range: 0..text.len().saturating_as(),
span: (Span::detached(), 0),
};
let c = text.chars().next().unwrap();
return Ok(Some((c, font, glyph)));
}
return Ok(None);
};
if covers.is_none() {
used.push(font.clone());
}
buffer.push_str(text);
buffer.set_language(language.clone());
buffer.set_script(
rustybuzz::Script::from_iso15924_tag(ttf_parser::Tag::from_bytes(b"math"))
.unwrap(),
);
buffer.set_direction(rustybuzz::Direction::LeftToRight);
buffer.set_flags(BufferFlags::REMOVE_DEFAULT_IGNORABLES);
let plan = create_shape_plan(
&font,
buffer.direction(),
buffer.script(),
buffer.language().as_ref(),
&features,
);
let buffer = rustybuzz::shape_with_plan(font.rusty(), &plan, buffer);
match buffer.len() {
0 => return Ok(None),
1 => {}
_ => bail!(
"shaping the text `{}` yielded more than one glyph", text.repr();
hint: "please report this as a bug",
),
}
let info = buffer.glyph_infos()[0];
let pos = buffer.glyph_positions()[0];
let cluster = info.cluster as usize;
let end = text[cluster..]
.char_indices()
.nth(1)
.map(|(i, _)| i)
.unwrap_or(text.len());
if info.glyph_id != 0 && covers.is_none_or(|cov| cov.is_match(&text[cluster..end])) {
let glyph = Glyph {
id: info.glyph_id as u16,
x_advance: font.to_em(pos.x_advance),
x_offset: font.to_em(pos.x_offset),
y_advance: font.to_em(pos.y_advance),
y_offset: font.to_em(pos.y_offset),
range: 0..text.len().saturating_as(),
span: (Span::detached(), 0),
};
let c = text[cluster..].chars().next().unwrap();
Ok(Some((c, font, glyph)))
} else {
shape_glyph(
world,
used,
buffer.clear(),
variant,
features,
language,
fallback,
text,
families,
)
}
}
#[derive(Debug, Copy, Clone)]
pub enum Limits {
Never,
Display,
Always,
}
impl Limits {
pub fn for_char(c: char) -> Self {
match default_math_class(c) {
Some(MathClass::Large) => {
if is_integral_char(c) {
Limits::Never
} else {
Limits::Display
}
}
Some(MathClass::Relation) => Limits::Always,
_ => Limits::Never,
}
}
pub fn for_class(class: MathClass) -> Self {
match class {
MathClass::Large => Self::Display,
MathClass::Relation => Self::Always,
_ => Self::Never,
}
}
pub fn active(&self, styles: StyleChain) -> bool {
match self {
Self::Always => true,
Self::Display => styles.get(EquationElem::size) == MathSize::Display,
Self::Never => false,
}
}
}
fn is_integral_char(c: char) -> bool {
('∫'..='∳').contains(&c) || ('⨋'..='⨜').contains(&c)
}