2021-03-18 19:13:31 +00:00
|
|
|
use crate::{
|
2021-08-03 19:48:58 +00:00
|
|
|
color::Color,
|
2021-03-25 09:42:46 +00:00
|
|
|
fonts::{FontId, GlyphId},
|
2021-03-25 16:21:26 +00:00
|
|
|
geometry::{
|
|
|
|
rect::RectF,
|
|
|
|
vector::{vec2f, Vector2F},
|
|
|
|
},
|
2021-03-24 15:51:28 +00:00
|
|
|
platform, scene, PaintContext,
|
2021-03-18 19:13:31 +00:00
|
|
|
};
|
|
|
|
use ordered_float::OrderedFloat;
|
|
|
|
use parking_lot::{Mutex, RwLock, RwLockUpgradableReadGuard};
|
|
|
|
use smallvec::SmallVec;
|
|
|
|
use std::{
|
|
|
|
borrow::Borrow,
|
|
|
|
collections::HashMap,
|
|
|
|
hash::{Hash, Hasher},
|
|
|
|
sync::Arc,
|
|
|
|
};
|
|
|
|
|
|
|
|
pub struct TextLayoutCache {
|
2021-05-21 20:10:38 +00:00
|
|
|
prev_frame: Mutex<HashMap<CacheKeyValue, Arc<LineLayout>>>,
|
|
|
|
curr_frame: RwLock<HashMap<CacheKeyValue, Arc<LineLayout>>>,
|
2021-03-24 15:51:28 +00:00
|
|
|
fonts: Arc<dyn platform::FontSystem>,
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl TextLayoutCache {
|
2021-03-24 15:51:28 +00:00
|
|
|
pub fn new(fonts: Arc<dyn platform::FontSystem>) -> Self {
|
2021-03-18 19:13:31 +00:00
|
|
|
Self {
|
|
|
|
prev_frame: Mutex::new(HashMap::new()),
|
|
|
|
curr_frame: RwLock::new(HashMap::new()),
|
2021-03-24 15:51:28 +00:00
|
|
|
fonts,
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn finish_frame(&self) {
|
|
|
|
let mut prev_frame = self.prev_frame.lock();
|
|
|
|
let mut curr_frame = self.curr_frame.write();
|
|
|
|
std::mem::swap(&mut *prev_frame, &mut *curr_frame);
|
|
|
|
curr_frame.clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn layout_str<'a>(
|
|
|
|
&'a self,
|
|
|
|
text: &'a str,
|
|
|
|
font_size: f32,
|
2021-08-03 19:48:58 +00:00
|
|
|
runs: &'a [(usize, FontId, Color)],
|
2021-05-21 20:10:38 +00:00
|
|
|
) -> Line {
|
2021-03-18 19:13:31 +00:00
|
|
|
let key = &CacheKeyRef {
|
|
|
|
text,
|
|
|
|
font_size: OrderedFloat(font_size),
|
|
|
|
runs,
|
|
|
|
} as &dyn CacheKey;
|
|
|
|
let curr_frame = self.curr_frame.upgradable_read();
|
2021-05-21 20:10:38 +00:00
|
|
|
if let Some(layout) = curr_frame.get(key) {
|
|
|
|
return Line::new(layout.clone(), runs);
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
let mut curr_frame = RwLockUpgradableReadGuard::upgrade(curr_frame);
|
2021-05-21 20:10:38 +00:00
|
|
|
if let Some((key, layout)) = self.prev_frame.lock().remove_entry(key) {
|
|
|
|
curr_frame.insert(key, layout.clone());
|
|
|
|
Line::new(layout.clone(), runs)
|
2021-03-18 19:13:31 +00:00
|
|
|
} else {
|
2021-07-16 16:43:50 +00:00
|
|
|
let layout = Arc::new(self.fonts.layout_line(text, font_size, runs));
|
2021-03-18 19:13:31 +00:00
|
|
|
let key = CacheKeyValue {
|
|
|
|
text: text.into(),
|
|
|
|
font_size: OrderedFloat(font_size),
|
|
|
|
runs: SmallVec::from(runs),
|
|
|
|
};
|
2021-05-21 20:10:38 +00:00
|
|
|
curr_frame.insert(key, layout.clone());
|
|
|
|
Line::new(layout, runs)
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
trait CacheKey {
|
|
|
|
fn key<'a>(&'a self) -> CacheKeyRef<'a>;
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> PartialEq for (dyn CacheKey + 'a) {
|
|
|
|
fn eq(&self, other: &dyn CacheKey) -> bool {
|
|
|
|
self.key() == other.key()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Eq for (dyn CacheKey + 'a) {}
|
|
|
|
|
|
|
|
impl<'a> Hash for (dyn CacheKey + 'a) {
|
|
|
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
|
|
|
self.key().hash(state)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Eq, PartialEq)]
|
|
|
|
struct CacheKeyValue {
|
|
|
|
text: String,
|
|
|
|
font_size: OrderedFloat<f32>,
|
2021-08-03 19:48:58 +00:00
|
|
|
runs: SmallVec<[(usize, FontId, Color); 1]>,
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl CacheKey for CacheKeyValue {
|
|
|
|
fn key<'a>(&'a self) -> CacheKeyRef<'a> {
|
|
|
|
CacheKeyRef {
|
|
|
|
text: &self.text.as_str(),
|
|
|
|
font_size: self.font_size,
|
|
|
|
runs: self.runs.as_slice(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Hash for CacheKeyValue {
|
|
|
|
fn hash<H: Hasher>(&self, state: &mut H) {
|
|
|
|
self.key().hash(state);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> Borrow<dyn CacheKey + 'a> for CacheKeyValue {
|
|
|
|
fn borrow(&self) -> &(dyn CacheKey + 'a) {
|
|
|
|
self as &dyn CacheKey
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq, Hash)]
|
|
|
|
struct CacheKeyRef<'a> {
|
|
|
|
text: &'a str,
|
|
|
|
font_size: OrderedFloat<f32>,
|
2021-08-03 19:48:58 +00:00
|
|
|
runs: &'a [(usize, FontId, Color)],
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a> CacheKey for CacheKeyRef<'a> {
|
|
|
|
fn key<'b>(&'b self) -> CacheKeyRef<'b> {
|
|
|
|
*self
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-24 15:51:28 +00:00
|
|
|
#[derive(Default, Debug)]
|
2021-03-18 19:13:31 +00:00
|
|
|
pub struct Line {
|
2021-05-21 20:10:38 +00:00
|
|
|
layout: Arc<LineLayout>,
|
2021-08-03 19:48:58 +00:00
|
|
|
color_runs: SmallVec<[(u32, Color); 32]>,
|
2021-05-21 20:10:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Default, Debug)]
|
|
|
|
pub struct LineLayout {
|
2021-03-18 19:13:31 +00:00
|
|
|
pub width: f32,
|
2021-04-06 15:14:55 +00:00
|
|
|
pub ascent: f32,
|
|
|
|
pub descent: f32,
|
2021-03-18 19:13:31 +00:00
|
|
|
pub runs: Vec<Run>,
|
|
|
|
pub len: usize,
|
2021-03-24 15:51:28 +00:00
|
|
|
pub font_size: f32,
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Run {
|
|
|
|
pub font_id: FontId,
|
|
|
|
pub glyphs: Vec<Glyph>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Glyph {
|
|
|
|
pub id: GlyphId,
|
|
|
|
pub position: Vector2F,
|
|
|
|
pub index: usize,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Line {
|
2021-08-03 19:48:58 +00:00
|
|
|
fn new(layout: Arc<LineLayout>, runs: &[(usize, FontId, Color)]) -> Self {
|
2021-05-21 20:10:38 +00:00
|
|
|
let mut color_runs = SmallVec::new();
|
|
|
|
for (len, _, color) in runs {
|
|
|
|
color_runs.push((*len as u32, *color));
|
|
|
|
}
|
|
|
|
Self { layout, color_runs }
|
|
|
|
}
|
|
|
|
|
2021-08-25 01:09:16 +00:00
|
|
|
pub fn runs(&self) -> &[Run] {
|
|
|
|
&self.layout.runs
|
|
|
|
}
|
|
|
|
|
2021-05-21 20:10:38 +00:00
|
|
|
pub fn width(&self) -> f32 {
|
|
|
|
self.layout.width
|
|
|
|
}
|
|
|
|
|
2021-03-18 19:13:31 +00:00
|
|
|
pub fn x_for_index(&self, index: usize) -> f32 {
|
2021-05-21 20:10:38 +00:00
|
|
|
for run in &self.layout.runs {
|
2021-03-18 19:13:31 +00:00
|
|
|
for glyph in &run.glyphs {
|
|
|
|
if glyph.index == index {
|
|
|
|
return glyph.position.x();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-05-21 20:10:38 +00:00
|
|
|
self.layout.width
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn index_for_x(&self, x: f32) -> Option<usize> {
|
2021-05-21 20:10:38 +00:00
|
|
|
if x >= self.layout.width {
|
2021-03-18 19:13:31 +00:00
|
|
|
None
|
|
|
|
} else {
|
2021-05-21 20:10:38 +00:00
|
|
|
for run in self.layout.runs.iter().rev() {
|
2021-03-18 19:13:31 +00:00
|
|
|
for glyph in run.glyphs.iter().rev() {
|
|
|
|
if glyph.position.x() <= x {
|
|
|
|
return Some(glyph.index);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Some(0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-05-28 22:25:15 +00:00
|
|
|
pub fn paint(&self, origin: Vector2F, visible_bounds: RectF, cx: &mut PaintContext) {
|
2021-05-21 20:14:05 +00:00
|
|
|
let padding_top = (visible_bounds.height() - self.layout.ascent - self.layout.descent) / 2.;
|
2021-05-21 20:10:38 +00:00
|
|
|
let baseline_origin = vec2f(0., padding_top + self.layout.ascent);
|
2021-03-18 19:13:31 +00:00
|
|
|
|
2021-05-21 20:10:38 +00:00
|
|
|
let mut color_runs = self.color_runs.iter();
|
|
|
|
let mut color_end = 0;
|
2021-08-03 19:48:58 +00:00
|
|
|
let mut color = Color::black();
|
2021-04-06 15:14:55 +00:00
|
|
|
|
2021-05-21 20:10:38 +00:00
|
|
|
for run in &self.layout.runs {
|
2021-05-28 22:25:15 +00:00
|
|
|
let max_glyph_width = cx
|
2021-05-21 20:10:38 +00:00
|
|
|
.font_cache
|
|
|
|
.bounding_box(run.font_id, self.layout.font_size)
|
|
|
|
.x();
|
2021-04-06 15:14:55 +00:00
|
|
|
|
2021-03-23 14:15:41 +00:00
|
|
|
for glyph in &run.glyphs {
|
2021-04-06 15:14:55 +00:00
|
|
|
let glyph_origin = baseline_origin + glyph.position;
|
|
|
|
|
2021-05-21 20:14:05 +00:00
|
|
|
if glyph_origin.x() + max_glyph_width < visible_bounds.origin().x() {
|
2021-03-23 14:15:41 +00:00
|
|
|
continue;
|
|
|
|
}
|
2021-05-21 20:14:05 +00:00
|
|
|
if glyph_origin.x() > visible_bounds.upper_right().x() {
|
2021-03-23 14:15:41 +00:00
|
|
|
break;
|
|
|
|
}
|
2021-03-18 19:13:31 +00:00
|
|
|
|
2021-05-21 20:10:38 +00:00
|
|
|
if glyph.index >= color_end {
|
|
|
|
if let Some(next_run) = color_runs.next() {
|
|
|
|
color_end += next_run.0 as usize;
|
|
|
|
color = next_run.1;
|
2021-03-23 14:15:41 +00:00
|
|
|
} else {
|
2021-05-21 20:10:38 +00:00
|
|
|
color_end = self.layout.len;
|
2021-08-03 19:48:58 +00:00
|
|
|
color = Color::black();
|
2021-03-23 14:15:41 +00:00
|
|
|
}
|
|
|
|
}
|
2021-03-18 19:13:31 +00:00
|
|
|
|
2021-05-28 22:25:15 +00:00
|
|
|
cx.scene.push_glyph(scene::Glyph {
|
2021-03-23 14:15:41 +00:00
|
|
|
font_id: run.font_id,
|
2021-05-21 20:10:38 +00:00
|
|
|
font_size: self.layout.font_size,
|
2021-03-23 18:11:56 +00:00
|
|
|
id: glyph.id,
|
2021-04-06 15:14:55 +00:00
|
|
|
origin: origin + glyph_origin,
|
2021-03-23 14:15:41 +00:00
|
|
|
color,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2021-03-18 19:13:31 +00:00
|
|
|
}
|
|
|
|
}
|
2021-08-25 01:09:16 +00:00
|
|
|
|
|
|
|
impl Run {
|
|
|
|
pub fn glyphs(&self) -> &[Glyph] {
|
|
|
|
&self.glyphs
|
|
|
|
}
|
|
|
|
}
|