use crate::krilla::{FrameContext, GlobalContext}; use crate::paint; use crate::util::{display_font, AbsExt, TransformExt}; use bytemuck::TransparentWrapper; use krilla::font::{GlyphId, GlyphUnits}; use krilla::surface::Surface; use std::ops::Range; use std::sync::Arc; use typst_library::diag::{bail, SourceResult}; use typst_library::layout::Size; use typst_library::text::{Font, Glyph, TextItem}; use typst_library::visualize::FillRule; use typst_syntax::Span; pub(crate) fn handle_text( fc: &mut FrameContext, t: &TextItem, surface: &mut Surface, gc: &mut GlobalContext, ) -> SourceResult<()> { *gc.languages.entry(t.lang).or_insert(0) += t.glyphs.len(); let font = convert_font(gc, t.font.clone())?; let fill = paint::convert_fill( gc, &t.fill, FillRule::NonZero, true, surface, fc.state().transforms(Size::zero()), )?; let text = t.text.as_str(); let size = t.size; let glyphs: &[PdfGlyph] = TransparentWrapper::wrap_slice(t.glyphs.as_slice()); surface.push_transform(&fc.state().transform.to_krilla()); surface.fill_glyphs( krilla::geom::Point::from_xy(0.0, 0.0), fill, &glyphs, font.clone(), text, size.to_f32(), GlyphUnits::Normalized, false, ); if let Some(stroke) = t .stroke .as_ref() .map(|s| paint::convert_stroke(gc, s, true, surface, fc.state().transforms(Size::zero()))) { let stroke = stroke?; surface.stroke_glyphs( krilla::geom::Point::from_xy(0.0, 0.0), stroke, &glyphs, font, text, size.to_f32(), GlyphUnits::Normalized, true, ); } surface.pop(); Ok(()) } fn convert_font( gc: &mut GlobalContext, typst_font: Font, ) -> SourceResult { if let Some(font) = gc.fonts_forward.get(&typst_font) { Ok(font.clone()) } else { let font = match krilla::font::Font::new( Arc::new(typst_font.data().clone()), typst_font.index(), true, ) { None => { let font_str = display_font(&typst_font); bail!(Span::detached(), "failed to process font {font_str}"); } Some(f) => f, }; gc.fonts_forward.insert(typst_font.clone(), font.clone()); gc.fonts_backward.insert(font.clone(), typst_font.clone()); Ok(font) } } #[derive(TransparentWrapper)] #[repr(transparent)] struct PdfGlyph(Glyph); impl krilla::font::Glyph for PdfGlyph { fn glyph_id(&self) -> GlyphId { GlyphId::new(self.0.id as u32) } fn text_range(&self) -> Range { self.0.range.start as usize..self.0.range.end as usize } fn x_advance(&self) -> f32 { self.0.x_advance.get() as f32 } fn x_offset(&self) -> f32 { self.0.x_offset.get() as f32 } fn y_offset(&self) -> f32 { 0.0 } fn y_advance(&self) -> f32 { 0.0 } }