| // Copyright 2022 The piet-gpu authors. |
| // |
| // Licensed under the Apache License, Version 2.0 (the "License"); |
| // you may not use this file except in compliance with the License. |
| // You may obtain a copy of the License at |
| // |
| // https://www.apache.org/licenses/LICENSE-2.0 |
| // |
| // Unless required by applicable law or agreed to in writing, software |
| // distributed under the License is distributed on an "AS IS" BASIS, |
| // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| // See the License for the specific language governing permissions and |
| // limitations under the License. |
| // |
| // Also licensed under MIT license, at your choice. |
| |
| //! An experimental API for glyph rendering. |
| |
| use piet::{kurbo::Affine, RenderContext}; |
| use swash::{scale::ScaleContext, CacheKey, FontDataRef}; |
| |
| use crate::{encoder::GlyphEncoder, PietGpuRenderContext}; |
| |
| pub struct GlyphRenderer { |
| pub render_ctx: PietGpuRenderContext, |
| scale_context: ScaleContext, |
| } |
| |
| #[repr(transparent)] |
| pub struct FontId(CacheKey); |
| |
| impl GlyphRenderer { |
| pub fn new() -> GlyphRenderer { |
| let render_ctx = PietGpuRenderContext::new(); |
| let scale_context = ScaleContext::new(); |
| GlyphRenderer { |
| render_ctx, |
| scale_context, |
| } |
| } |
| |
| pub unsafe fn add_glyph( |
| &mut self, |
| font_data: &[u8], |
| font_id: u64, |
| glyph_id: u16, |
| transform: [f32; 6], |
| ) { |
| // This transmute is dodgy because the definition in swash isn't repr(transparent). |
| // I think the best solution is to have a from_u64 method, but we'll work that out |
| // later. |
| let font_id = FontId(std::mem::transmute(font_id)); |
| let encoder = self.make_glyph(font_data, font_id, glyph_id); |
| const DEFAULT_UPEM: u16 = 2048; |
| let affine = Affine::new([ |
| transform[0] as f64, |
| transform[1] as f64, |
| transform[2] as f64, |
| transform[3] as f64, |
| transform[4] as f64, |
| transform[5] as f64, |
| ]) * Affine::scale(1.0 / DEFAULT_UPEM as f64); |
| self.render_ctx.transform(affine); |
| self.render_ctx.encode_glyph(&encoder); |
| // TODO: don't fill glyph if RGBA |
| self.render_ctx.fill_glyph(0xff_ff_ff_ff); |
| self.render_ctx.transform(affine.inverse()); |
| } |
| |
| pub fn reset(&mut self) { |
| self.render_ctx = PietGpuRenderContext::new(); |
| } |
| |
| fn make_glyph(&mut self, font_data: &[u8], font_id: FontId, glyph_id: u16) -> GlyphEncoder { |
| let mut encoder = GlyphEncoder::default(); |
| let font_data = FontDataRef::new(font_data).expect("invalid font"); |
| let mut font_ref = font_data.get(0).expect("invalid font index"); |
| font_ref.key = font_id.0; |
| let mut scaler = self.scale_context.builder(font_ref).size(2048.).build(); |
| if let Some(outline) = scaler.scale_outline(glyph_id) { |
| crate::text::append_outline(&mut encoder, outline.verbs(), outline.points()); |
| } else { |
| println!("failed to scale"); |
| } |
| encoder |
| } |
| } |