text: migrate to ab_glyph. this should give rendering consistency across platforms
This commit is contained in:
parent
21a79c56a7
commit
4d8a567b36
@ -10,7 +10,5 @@ edition = "2018"
|
|||||||
bevy_app = { path = "../bevy_app" }
|
bevy_app = { path = "../bevy_app" }
|
||||||
bevy_asset = { path = "../bevy_asset" }
|
bevy_asset = { path = "../bevy_asset" }
|
||||||
bevy_render = { path = "../bevy_render" }
|
bevy_render = { path = "../bevy_render" }
|
||||||
skribo = "0.1.0"
|
ab_glyph = "0.2.2"
|
||||||
font-kit = "0.6"
|
|
||||||
pathfinder_geometry = "0.5"
|
|
||||||
anyhow = "1.0"
|
anyhow = "1.0"
|
@ -1,30 +1,117 @@
|
|||||||
use crate::render::render_text;
|
use ab_glyph::{FontVec, Glyph, InvalidFont, Point, PxScale, ScaleFont};
|
||||||
use bevy_render::{texture::Texture, Color};
|
use bevy_render::{
|
||||||
use font_kit::{error::FontLoadingError, metrics::Metrics};
|
texture::{Texture, TextureType},
|
||||||
use skribo::{FontCollection, FontFamily};
|
Color,
|
||||||
use std::sync::Arc;
|
};
|
||||||
|
|
||||||
pub struct Font {
|
pub struct Font {
|
||||||
pub collection: FontCollection,
|
pub font: FontVec,
|
||||||
pub metrics: Metrics,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
unsafe impl Send for Font {}
|
unsafe impl Send for Font {}
|
||||||
unsafe impl Sync for Font {}
|
unsafe impl Sync for Font {}
|
||||||
|
|
||||||
impl Font {
|
impl Font {
|
||||||
pub fn try_from_bytes(font_data: Vec<u8>) -> Result<Self, FontLoadingError> {
|
pub fn try_from_bytes(font_data: Vec<u8>) -> Result<Self, InvalidFont> {
|
||||||
let font = font_kit::font::Font::from_bytes(Arc::new(font_data), 0)?;
|
let font = FontVec::try_from_vec(font_data)?;
|
||||||
let metrics = font.metrics();
|
Ok(Font { font })
|
||||||
let mut collection = FontCollection::new();
|
|
||||||
collection.add_family(FontFamily::new_from_font(font));
|
|
||||||
Ok(Font {
|
|
||||||
collection,
|
|
||||||
metrics,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn render_text(&self, text: &str, color: Color, width: usize, height: usize) -> Texture {
|
// adapted from ab_glyph example: https://github.com/alexheretic/ab-glyph/blob/master/dev/examples/image.rs
|
||||||
render_text(self, text, color, width, height)
|
pub fn render_text(&self, text: &str, color: Color, font_size: f32, width: usize, height: usize) -> Texture {
|
||||||
|
let scale = PxScale::from(font_size);
|
||||||
|
|
||||||
|
let scaled_font = ab_glyph::Font::as_scaled(&self.font, scale);
|
||||||
|
|
||||||
|
let mut glyphs = Vec::new();
|
||||||
|
layout_paragraph(
|
||||||
|
scaled_font,
|
||||||
|
ab_glyph::point(0.0, 0.0),
|
||||||
|
width as f32,
|
||||||
|
text,
|
||||||
|
&mut glyphs,
|
||||||
|
);
|
||||||
|
|
||||||
|
let color_u8 = [
|
||||||
|
(color.r * 255.0) as u8,
|
||||||
|
(color.g * 255.0) as u8,
|
||||||
|
(color.b * 255.0) as u8,
|
||||||
|
];
|
||||||
|
|
||||||
|
// TODO: this offset is a bit hackey
|
||||||
|
let mut alpha = vec![0.0; width * height];
|
||||||
|
for glyph in glyphs {
|
||||||
|
if let Some(outlined) = scaled_font.outline_glyph(glyph) {
|
||||||
|
let bounds = outlined.px_bounds();
|
||||||
|
// Draw the glyph into the image per-pixel by using the draw closure
|
||||||
|
outlined.draw(|x, y, v| {
|
||||||
|
// Offset the position by the glyph bounding box
|
||||||
|
// Turn the coverage into an alpha value (blended with any previous)
|
||||||
|
let offset_x = x as usize + bounds.min.x as usize;
|
||||||
|
let offset_y = y as usize + bounds.min.y as usize;
|
||||||
|
if offset_x >= width || offset_y >= height {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
alpha[offset_y * width + offset_x] = v;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Texture::load(TextureType::Data(
|
||||||
|
alpha
|
||||||
|
.iter()
|
||||||
|
.map(|a| {
|
||||||
|
vec![
|
||||||
|
color_u8[0],
|
||||||
|
color_u8[1],
|
||||||
|
color_u8[2],
|
||||||
|
(color.a * a * 255.0) as u8,
|
||||||
|
]
|
||||||
|
})
|
||||||
|
.flatten()
|
||||||
|
.collect::<Vec<u8>>(),
|
||||||
|
width,
|
||||||
|
height,
|
||||||
|
))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn layout_paragraph<F, SF>(
|
||||||
|
font: SF,
|
||||||
|
position: Point,
|
||||||
|
max_width: f32,
|
||||||
|
text: &str,
|
||||||
|
target: &mut Vec<Glyph>,
|
||||||
|
) where
|
||||||
|
F: ab_glyph::Font,
|
||||||
|
SF: ScaleFont<F>,
|
||||||
|
{
|
||||||
|
let v_advance = font.height() + font.line_gap();
|
||||||
|
let mut caret = position + ab_glyph::point(0.0, font.ascent());
|
||||||
|
let mut last_glyph: Option<Glyph> = None;
|
||||||
|
for c in text.chars() {
|
||||||
|
if c.is_control() {
|
||||||
|
if c == '\n' {
|
||||||
|
caret = ab_glyph::point(position.x, caret.y + v_advance);
|
||||||
|
last_glyph = None;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
let mut glyph = font.scaled_glyph(c);
|
||||||
|
if let Some(previous) = last_glyph.take() {
|
||||||
|
caret.x += font.kern(previous.id, glyph.id);
|
||||||
|
}
|
||||||
|
glyph.position = caret;
|
||||||
|
|
||||||
|
last_glyph = Some(glyph.clone());
|
||||||
|
caret.x += font.h_advance(glyph.id);
|
||||||
|
|
||||||
|
if !c.is_whitespace() && caret.x > position.x + max_width {
|
||||||
|
caret = ab_glyph::point(position.x, caret.y + v_advance);
|
||||||
|
glyph.position = caret;
|
||||||
|
last_glyph = None;
|
||||||
|
}
|
||||||
|
|
||||||
|
target.push(glyph);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
mod font;
|
mod font;
|
||||||
mod font_loader;
|
mod font_loader;
|
||||||
mod render;
|
|
||||||
|
|
||||||
pub use font::*;
|
pub use font::*;
|
||||||
pub use font_loader::*;
|
pub use font_loader::*;
|
||||||
|
@ -1,141 +0,0 @@
|
|||||||
use crate::Font;
|
|
||||||
use bevy_render::{
|
|
||||||
texture::{Texture, TextureType},
|
|
||||||
Color,
|
|
||||||
};
|
|
||||||
use font_kit::{
|
|
||||||
canvas::{Canvas, Format, RasterizationOptions},
|
|
||||||
hinting::HintingOptions,
|
|
||||||
};
|
|
||||||
use pathfinder_geometry::transform2d::Transform2F;
|
|
||||||
use skribo::{LayoutSession, TextStyle};
|
|
||||||
use std::ops::Range;
|
|
||||||
|
|
||||||
struct TextSurface {
|
|
||||||
width: usize,
|
|
||||||
height: usize,
|
|
||||||
pixels: Vec<u8>,
|
|
||||||
}
|
|
||||||
|
|
||||||
fn composite(a: u8, b: u8) -> u8 {
|
|
||||||
let y = ((255 - a) as u16) * ((255 - b) as u16);
|
|
||||||
let y = (y + (y >> 8) + 0x80) >> 8; // fast approx to round(y / 255)
|
|
||||||
255 - (y as u8)
|
|
||||||
}
|
|
||||||
|
|
||||||
impl TextSurface {
|
|
||||||
fn new(width: usize, height: usize) -> TextSurface {
|
|
||||||
let pixels = vec![0; width * height];
|
|
||||||
TextSurface {
|
|
||||||
width,
|
|
||||||
height,
|
|
||||||
pixels,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn paint_from_canvas(&mut self, canvas: &Canvas, x: i32, y: i32) {
|
|
||||||
let (cw, ch) = (canvas.size.x(), canvas.size.y());
|
|
||||||
let (w, h) = (self.width as i32, self.height as i32);
|
|
||||||
let y = y - ch;
|
|
||||||
let xmin = 0.max(-x);
|
|
||||||
let xmax = cw.min(w - x);
|
|
||||||
let ymin = 0.max(-y);
|
|
||||||
let ymax = ch.min(h - y);
|
|
||||||
for yy in ymin..(ymax.max(ymin)) {
|
|
||||||
for xx in xmin..(xmax.max(xmin)) {
|
|
||||||
let pix = canvas.pixels[(cw * yy + xx) as usize];
|
|
||||||
let dst_ix = ((y + yy) * w + x + xx) as usize;
|
|
||||||
self.pixels[dst_ix] = composite(self.pixels[dst_ix], pix);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
fn paint_layout_session<S: AsRef<str>>(
|
|
||||||
&mut self,
|
|
||||||
layout: &mut LayoutSession<S>,
|
|
||||||
x: i32,
|
|
||||||
y: i32,
|
|
||||||
size: f32,
|
|
||||||
range: Range<usize>,
|
|
||||||
) {
|
|
||||||
for run in layout.iter_substr(range) {
|
|
||||||
let font = run.font();
|
|
||||||
for glyph in run.glyphs() {
|
|
||||||
let glyph_id = glyph.glyph_id;
|
|
||||||
let glyph_x = (glyph.offset.x() as i32) + x;
|
|
||||||
let glyph_y = (glyph.offset.y() as i32) + y;
|
|
||||||
let bounds = font
|
|
||||||
.font
|
|
||||||
.raster_bounds(
|
|
||||||
glyph_id,
|
|
||||||
size,
|
|
||||||
Transform2F::default(),
|
|
||||||
HintingOptions::None,
|
|
||||||
RasterizationOptions::GrayscaleAa,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
if bounds.width() > 0 && bounds.height() > 0 {
|
|
||||||
let origin_adj = bounds.origin().to_f32();
|
|
||||||
let neg_origin = -origin_adj;
|
|
||||||
let mut canvas = Canvas::new(bounds.size(), Format::A8);
|
|
||||||
font.font
|
|
||||||
.rasterize_glyph(
|
|
||||||
&mut canvas,
|
|
||||||
glyph_id,
|
|
||||||
size,
|
|
||||||
Transform2F::from_translation(neg_origin),
|
|
||||||
HintingOptions::None,
|
|
||||||
RasterizationOptions::GrayscaleAa,
|
|
||||||
)
|
|
||||||
.unwrap();
|
|
||||||
self.paint_from_canvas(
|
|
||||||
&canvas,
|
|
||||||
glyph_x + bounds.origin_x(),
|
|
||||||
glyph_y - bounds.origin_y(),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn render_text(font: &Font, text: &str, color: Color, width: usize, height: usize) -> Texture {
|
|
||||||
let mut surface = TextSurface::new(width, height);
|
|
||||||
let style = TextStyle {
|
|
||||||
size: height as f32,
|
|
||||||
};
|
|
||||||
let offset = style.size * (font.metrics.ascent - font.metrics.cap_height)
|
|
||||||
/ font.metrics.units_per_em as f32;
|
|
||||||
|
|
||||||
let mut layout = LayoutSession::create(&text, &style, &font.collection);
|
|
||||||
surface.paint_layout_session(
|
|
||||||
&mut layout,
|
|
||||||
0,
|
|
||||||
style.size as i32 - offset as i32,
|
|
||||||
style.size,
|
|
||||||
0..text.len(),
|
|
||||||
);
|
|
||||||
let color_u8 = [
|
|
||||||
(color.r * 255.0) as u8,
|
|
||||||
(color.g * 255.0) as u8,
|
|
||||||
(color.b * 255.0) as u8,
|
|
||||||
];
|
|
||||||
|
|
||||||
Texture::load(TextureType::Data(
|
|
||||||
surface
|
|
||||||
.pixels
|
|
||||||
.iter()
|
|
||||||
.map(|p| {
|
|
||||||
vec![
|
|
||||||
color_u8[0],
|
|
||||||
color_u8[1],
|
|
||||||
color_u8[2],
|
|
||||||
(color.a * *p as f32) as u8,
|
|
||||||
]
|
|
||||||
})
|
|
||||||
.flatten()
|
|
||||||
.collect::<Vec<u8>>(),
|
|
||||||
surface.width,
|
|
||||||
surface.height,
|
|
||||||
))
|
|
||||||
}
|
|
@ -38,7 +38,7 @@ impl Label {
|
|||||||
|
|
||||||
if let Some(font) = fonts.get(&label.font) {
|
if let Some(font) = fonts.get(&label.font) {
|
||||||
let texture =
|
let texture =
|
||||||
font.render_text(&label.text, label.color, width as usize, height as usize);
|
font.render_text(&label.text, label.color, label.font_size, width as usize, height as usize);
|
||||||
|
|
||||||
let material = color_materials.get_or_insert_with(*color_material_handle, || {
|
let material = color_materials.get_or_insert_with(*color_material_handle, || {
|
||||||
ColorMaterial::from(Handle::<Texture>::new())
|
ColorMaterial::from(Handle::<Texture>::new())
|
||||||
|
@ -15,7 +15,7 @@ fn main() {
|
|||||||
fn text_update_system(diagnostics: Res<Diagnostics>, mut label: ComMut<Label>) {
|
fn text_update_system(diagnostics: Res<Diagnostics>, mut label: ComMut<Label>) {
|
||||||
if let Some(fps) = diagnostics.get(FrameTimeDiagnosticsPlugin::FPS) {
|
if let Some(fps) = diagnostics.get(FrameTimeDiagnosticsPlugin::FPS) {
|
||||||
if let Some(average) = fps.average() {
|
if let Some(average) = fps.average() {
|
||||||
label.text = format!("FPS: {}", average.round());
|
label.text = format!("FPS: {}", average);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user