2022-01-18 18:41:28 -08:00
|
|
|
// 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.
|
|
|
|
|
2022-01-19 11:58:01 -08:00
|
|
|
use piet::{kurbo::Affine, RenderContext};
|
|
|
|
use swash::{scale::ScaleContext, CacheKey, FontDataRef, FontRef};
|
2022-01-18 18:41:28 -08:00
|
|
|
|
|
|
|
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 {
|
2022-01-19 11:58:01 -08:00
|
|
|
let render_ctx = PietGpuRenderContext::new();
|
|
|
|
let scale_context = ScaleContext::new();
|
2022-01-18 18:41:28 -08:00
|
|
|
GlyphRenderer {
|
2022-01-19 11:58:01 -08:00
|
|
|
render_ctx,
|
|
|
|
scale_context,
|
2022-01-18 18:41:28 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-19 11:58:01 -08:00
|
|
|
pub unsafe fn add_glyph(
|
|
|
|
&mut self,
|
|
|
|
font_data: &[u8],
|
|
|
|
font_id: u64,
|
|
|
|
glyph_id: u16,
|
|
|
|
transform: [f32; 6],
|
|
|
|
) {
|
2022-01-18 18:41:28 -08:00
|
|
|
// 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);
|
2022-01-19 11:58:01 -08:00
|
|
|
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);
|
2022-01-18 18:41:28 -08:00
|
|
|
self.render_ctx.encode_glyph(&encoder);
|
|
|
|
// TODO: don't fill glyph if RGBA
|
|
|
|
self.render_ctx.fill_glyph(0xff_ff_ff_ff);
|
2022-01-19 11:58:01 -08:00
|
|
|
self.render_ctx.transform(affine.inverse());
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn reset(&mut self) {
|
|
|
|
self.render_ctx = PietGpuRenderContext::new();
|
2022-01-18 18:41:28 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
fn make_glyph(&mut self, font_data: &[u8], font_id: FontId, glyph_id: u16) -> GlyphEncoder {
|
|
|
|
let mut encoder = GlyphEncoder::default();
|
2022-01-19 11:58:01 -08:00
|
|
|
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;
|
2022-01-18 18:41:28 -08:00
|
|
|
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());
|
2022-01-19 11:58:01 -08:00
|
|
|
} else {
|
|
|
|
println!("failed to scale");
|
2022-01-18 18:41:28 -08:00
|
|
|
}
|
|
|
|
encoder
|
|
|
|
}
|
|
|
|
}
|