use std::mem;
use crossfont::{GlyphKey, Rasterize, RasterizedGlyph};
use tiny_skia::{Color, Pixmap, PixmapPaint, PixmapRef, Transform};
use crate::title::config;
pub struct CrossfontTitleText {
title: String,
font_desc: crossfont::FontDesc,
font_key: crossfont::FontKey,
size: crossfont::Size,
scale: u32,
metrics: crossfont::Metrics,
rasterizer: crossfont::Rasterizer,
color: Color,
pixmap: Option<Pixmap>,
}
impl std::fmt::Debug for CrossfontTitleText {
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
f.debug_struct("TitleText")
.field("title", &self.title)
.field("font_desc", &self.font_desc)
.field("font_key", &self.font_key)
.field("size", &self.size)
.field("scale", &self.scale)
.field("pixmap", &self.pixmap)
.finish()
}
}
impl CrossfontTitleText {
pub fn new(color: Color) -> Result<Self, crossfont::Error> {
let title = "".into();
let font_pref = config::titlebar_font().unwrap_or_default();
let font_style = font_pref
.style
.map(crossfont::Style::Specific)
.unwrap_or_else(|| crossfont::Style::Description {
slant: crossfont::Slant::Normal,
weight: crossfont::Weight::Normal,
});
let font_desc = crossfont::FontDesc::new(&font_pref.name, font_style);
let mut rasterizer = crossfont::Rasterizer::new()?;
let size = crossfont::Size::new(font_pref.pt_size);
let font_key = rasterizer.load_font(&font_desc, size)?;
rasterizer.get_glyph(GlyphKey {
font_key,
character: 'm',
size,
})?;
let metrics = rasterizer.metrics(font_key, size)?;
let mut this = Self {
pixmap: None,
rasterizer,
font_desc,
font_key,
scale: 1,
metrics,
title,
color,
size,
};
this.rerender();
Ok(this)
}
fn update_metrics(&mut self) -> Result<(), crossfont::Error> {
self.rasterizer.get_glyph(GlyphKey {
font_key: self.font_key,
character: 'm',
size: self.size,
})?;
self.metrics = self.rasterizer.metrics(self.font_key, self.size)?;
Ok(())
}
pub fn update_scale(&mut self, scale: u32) {
let old_scale = mem::replace(&mut self.scale, scale);
if old_scale != self.scale {
self.size = self.size.scale(self.scale as f32 / old_scale as f32);
self.update_metrics().ok();
self.rerender();
}
}
pub fn update_title<S: Into<String>>(&mut self, title: S) {
let title = title.into();
if self.title != title {
self.title = title;
self.rerender();
}
}
pub fn update_color(&mut self, color: Color) {
if self.color != color {
self.color = color;
self.rerender();
}
}
fn rerender(&mut self) {
let glyphs: Vec<_> = self
.title
.chars()
.filter_map(|character| {
let key = GlyphKey {
character,
font_key: self.font_key,
size: self.size,
};
self.rasterizer
.get_glyph(key)
.map(|glyph| (key, glyph))
.ok()
})
.collect();
if glyphs.is_empty() {
self.pixmap = None;
return;
}
let width = self.calc_width(&glyphs);
let height = self.metrics.line_height.round() as i32;
let mut pixmap = if let Some(p) = Pixmap::new(width as u32, height as u32) {
p
} else {
self.pixmap = None;
return;
};
let mut caret = 0;
let mut last_glyph = None;
for (key, glyph) in glyphs {
let mut buffer = Vec::with_capacity(glyph.width as usize * 4);
let glyph_buffer = match &glyph.buffer {
crossfont::BitmapBuffer::Rgb(v) => v.chunks(3),
crossfont::BitmapBuffer::Rgba(v) => v.chunks(4),
};
for px in glyph_buffer {
let alpha = if let Some(alpha) = px.get(3) {
*alpha as f32 / 255.0
} else {
let r = px[0] as f32 / 255.0;
let g = px[1] as f32 / 255.0;
let b = px[2] as f32 / 255.0;
(r + g + b) / 3.0
};
let mut color = self.color;
color.set_alpha(alpha);
let color = color.premultiply().to_color_u8();
buffer.push(color.red());
buffer.push(color.green());
buffer.push(color.blue());
buffer.push(color.alpha());
}
if let Some(last) = last_glyph {
let (x, _) = self.rasterizer.kerning(last, key);
caret += x as i32;
}
if let Some(pixmap_glyph) =
PixmapRef::from_bytes(&buffer, glyph.width as _, glyph.height as _)
{
pixmap.draw_pixmap(
glyph.left + caret,
height - glyph.top + self.metrics.descent.round() as i32,
pixmap_glyph,
&PixmapPaint::default(),
Transform::identity(),
None,
);
}
caret += glyph.advance.0;
last_glyph = Some(key);
}
self.pixmap = Some(pixmap);
}
pub fn pixmap(&self) -> Option<&Pixmap> {
self.pixmap.as_ref()
}
fn calc_width(&mut self, glyphs: &[(GlyphKey, RasterizedGlyph)]) -> i32 {
let mut caret = 0;
let mut last_glyph: Option<&GlyphKey> = None;
for (key, glyph) in glyphs.iter() {
if let Some(last) = last_glyph {
let (x, _) = self.rasterizer.kerning(*last, *key);
caret += x as i32;
}
caret += glyph.advance.0;
last_glyph = Some(key);
}
caret
}
}