Cache font matches, use usize for line index, use font system for swash

This commit is contained in:
Jeremy Soller 2022-10-26 12:23:03 -06:00
parent 94576fb682
commit 119a570ee9
No known key found for this signature in database
GPG key ID: 87F211AF2BE4C2FE
8 changed files with 105 additions and 120 deletions

View file

@ -64,7 +64,7 @@ pub struct Window {
theme: Theme,
path_opt: Option<PathBuf>,
buffer: Mutex<TextBuffer<'static>>,
cache: Mutex<SwashCache>,
cache: Mutex<SwashCache<'static>>,
bold: bool,
italic: bool,
monospaced: bool,
@ -117,11 +117,13 @@ impl Application for Window {
FONT_SIZES[font_size_i],
);
let cache = SwashCache::new(&FONT_SYSTEM);
let mut window = Window {
theme: Theme::Dark,
path_opt: None,
buffer: Mutex::new(buffer),
cache: Mutex::new(SwashCache::new()),
cache: Mutex::new(cache),
bold: false,
italic: false,
monospaced: true,
@ -178,7 +180,7 @@ impl Application for Window {
} else {
cosmic_text::Weight::NORMAL
});
buffer.set_attrs(&FONT_SYSTEM, attrs);
buffer.set_attrs(attrs);
},
Message::Italic(italic) => {
self.italic = italic;
@ -189,7 +191,7 @@ impl Application for Window {
} else {
cosmic_text::Style::Normal
});
buffer.set_attrs(&FONT_SYSTEM, attrs);
buffer.set_attrs(attrs);
},
Message::Monospaced(monospaced) => {
self.monospaced = monospaced;
@ -202,7 +204,7 @@ impl Application for Window {
cosmic_text::Family::SansSerif
})
.monospaced(monospaced);
buffer.set_attrs(&FONT_SYSTEM, attrs);
buffer.set_attrs(attrs);
},
Message::MetricsChanged(metrics) => {
let mut buffer = self.buffer.lock().unwrap();

View file

@ -63,16 +63,16 @@ impl StyleSheet for Theme {
pub struct TextBox<'a> {
buffer: &'a Mutex<TextBuffer<'static>>,
cache: &'a Mutex<SwashCache>,
cache: &'a Mutex<SwashCache<'static>>,
}
impl<'a> TextBox<'a> {
pub fn new(buffer: &'a Mutex<TextBuffer<'static>>, cache: &'a Mutex<SwashCache>) -> Self {
pub fn new(buffer: &'a Mutex<TextBuffer<'static>>, cache: &'a Mutex<SwashCache<'static>>) -> Self {
Self { buffer, cache }
}
}
pub fn text_box<'a>(buffer: &'a Mutex<TextBuffer<'static>>, cache: &'a Mutex<SwashCache>) -> TextBox<'a> {
pub fn text_box<'a>(buffer: &'a Mutex<TextBuffer<'static>>, cache: &'a Mutex<SwashCache<'static>>) -> TextBox<'a> {
TextBox::new(buffer, cache)
}

View file

@ -53,7 +53,7 @@ fn main() {
default_text.to_string()
};
let mut swash_cache = SwashCache::new();
let mut swash_cache = SwashCache::new(&font_system);
let line_x = 8 * display_scale;
let attrs = cosmic_text::Attrs::new().monospaced(cfg!(feature = "mono"));

View file

@ -71,7 +71,7 @@ fn main() {
window.height() as i32
);
let mut swash_cache = SwashCache::new();
let mut swash_cache = SwashCache::new(&font_system);
let text = if let Some(arg) = env::args().nth(1) {
fs::read_to_string(&arg).expect("failed to open file")

View file

@ -2,7 +2,7 @@
pub use fontdb::{Family, Stretch, Style, Weight};
#[derive(Clone, Debug, Eq, PartialEq)]
#[derive(Clone, Copy, Debug, Eq, Hash, PartialEq)]
pub struct Attrs<'a> {
pub family: Family<'a>,
pub monospaced: bool,

View file

@ -3,6 +3,7 @@
use std::{
cmp,
fmt,
sync::Arc,
time::Instant,
};
use unicode_segmentation::UnicodeSegmentation;
@ -52,32 +53,32 @@ pub enum TextAction {
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
pub struct TextCursor {
/// Text line the cursor is on
pub line: TextLineIndex,
pub line: usize,
/// Index of glyph at cursor (will insert behind this glyph)
pub index: usize,
}
impl TextCursor {
pub const fn new(line: TextLineIndex, index: usize) -> Self {
pub const fn new(line: usize, index: usize) -> Self {
Self { line, index }
}
}
struct TextLayoutCursor {
line: TextLineIndex,
line: usize,
layout: usize,
glyph: usize,
}
impl TextLayoutCursor {
fn new(line: TextLineIndex, layout: usize, glyph: usize) -> Self {
fn new(line: usize, layout: usize, glyph: usize) -> Self {
Self { line, layout, glyph }
}
}
pub struct TextLayoutRun<'a> {
/// The index of the original text line
pub line_i: TextLineIndex,
pub line_i: usize,
/// The original text line
pub text: &'a str,
/// True if the original paragraph direction is RTL
@ -130,7 +131,7 @@ impl<'a, 'b> Iterator for TextLayoutRunIter<'a, 'b> {
}
return Some(TextLayoutRun {
line_i: TextLineIndex::new(self.line_i),
line_i: self.line_i,
text: line.text.as_str(),
rtl: shape.rtl,
glyphs: &layout_line.glyphs,
@ -145,20 +146,6 @@ impl<'a, 'b> Iterator for TextLayoutRunIter<'a, 'b> {
}
}
/// Index of a text line
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq, Ord, PartialOrd)]
pub struct TextLineIndex(usize);
impl TextLineIndex {
pub fn new(index: usize) -> Self {
Self(index)
}
pub fn get(&self) -> usize {
self.0
}
}
/// Metrics of text
#[derive(Clone, Copy, Debug, Default, Eq, PartialEq)]
pub struct TextMetrics {
@ -237,8 +224,10 @@ impl TextBufferLine {
/// A buffer of text that is shaped and laid out
pub struct TextBuffer<'a> {
font_matches: FontMatches<'a>,
font_system: &'a FontSystem<'a>,
font_matches: Arc<FontMatches<'a>>,
attrs: Attrs<'a>,
attr_spans: Vec<(TextCursor, usize, Attrs<'a>)>,
lines: Vec<TextBufferLine>,
metrics: TextMetrics,
width: i32,
@ -256,10 +245,12 @@ impl<'a> TextBuffer<'a> {
attrs: Attrs<'a>,
metrics: TextMetrics,
) -> Self {
let font_matches = font_system.matches_attrs(&attrs);
let font_matches = font_system.get_font_matches(attrs);
let mut buffer = Self {
font_system,
font_matches,
attrs,
attr_spans: Vec::new(),
lines: Vec::new(),
metrics,
width: 0,
@ -312,7 +303,7 @@ impl<'a> TextBuffer<'a> {
let mut reshaped = 0;
let mut layout_i = 0;
for (line_i, line) in self.lines.iter_mut().enumerate() {
if line_i > self.cursor.line.get() {
if line_i > self.cursor.line {
break;
}
@ -324,7 +315,7 @@ impl<'a> TextBuffer<'a> {
self.metrics.font_size,
self.width
);
if line_i == self.cursor.line.get() {
if line_i == self.cursor.line {
let layout_cursor = self.layout_cursor(&self.cursor);
layout_i += layout_cursor.layout as i32;
break;
@ -385,7 +376,7 @@ impl<'a> TextBuffer<'a> {
}
fn layout_cursor(&self, cursor: &TextCursor) -> TextLayoutCursor {
let line = &self.lines[cursor.line.get()];
let line = &self.lines[cursor.line];
let layout = line.layout_opt.as_ref().unwrap(); //TODO: ensure layout is done?
for (layout_i, layout_line) in layout.iter().enumerate() {
@ -428,7 +419,7 @@ impl<'a> TextBuffer<'a> {
}
fn set_layout_cursor(&mut self, cursor: TextLayoutCursor) {
let line = &mut self.lines[cursor.line.get()];
let line = &mut self.lines[cursor.line];
let layout = line.layout(
&self.font_matches,
self.metrics.font_size,
@ -516,9 +507,9 @@ impl<'a> TextBuffer<'a> {
}
/// Set attributes
pub fn set_attrs(&mut self, font_system: &'a FontSystem<'a>, attrs: Attrs<'a>) {
pub fn set_attrs(&mut self, attrs: Attrs<'a>) {
if attrs != self.attrs {
self.font_matches = font_system.matches_attrs(&attrs);
self.font_matches = self.font_system.get_font_matches(attrs);
self.attrs = attrs;
for line in self.lines.iter_mut() {
@ -529,6 +520,10 @@ impl<'a> TextBuffer<'a> {
}
}
pub fn add_attr_span(&mut self, cursor: TextCursor, len: usize, attrs: Attrs<'a>) {
self.attr_spans.push((cursor, len, attrs));
}
/// Set text of buffer
pub fn set_text(&mut self, text: &str) {
self.lines.clear();
@ -558,7 +553,7 @@ impl<'a> TextBuffer<'a> {
match action {
TextAction::Previous => {
let line = &mut self.lines[self.cursor.line.get()];
let line = &mut self.lines[self.cursor.line];
if self.cursor.index > 0 {
// Find previous character index
@ -573,14 +568,14 @@ impl<'a> TextBuffer<'a> {
self.cursor.index = prev_index;
self.redraw = true;
} else if self.cursor.line.get() > 0 {
self.cursor.line = TextLineIndex::new(self.cursor.line.get() - 1);
self.cursor.index = self.lines[self.cursor.line.get()].text.len();
} else if self.cursor.line > 0 {
self.cursor.line -= 1;
self.cursor.index = self.lines[self.cursor.line].text.len();
self.redraw = true;
}
},
TextAction::Next => {
let line = &mut self.lines[self.cursor.line.get()];
let line = &mut self.lines[self.cursor.line];
if self.cursor.index < line.text.len() {
for (i, c) in line.text.grapheme_indices(true) {
@ -590,14 +585,14 @@ impl<'a> TextBuffer<'a> {
break;
}
}
} else if self.cursor.line.get() + 1 < self.lines.len() {
self.cursor.line = TextLineIndex::new(self.cursor.line.get() + 1);
} else if self.cursor.line + 1 < self.lines.len() {
self.cursor.line += 1;
self.cursor.index = 0;
self.redraw = true;
}
},
TextAction::Left => {
let rtl_opt = self.lines[self.cursor.line.get()].shape_opt.as_ref().map(|shape| shape.rtl);
let rtl_opt = self.lines[self.cursor.line].shape_opt.as_ref().map(|shape| shape.rtl);
if let Some(rtl) = rtl_opt {
if rtl {
self.action(TextAction::Next);
@ -607,7 +602,7 @@ impl<'a> TextBuffer<'a> {
}
},
TextAction::Right => {
let rtl_opt = self.lines[self.cursor.line.get()].shape_opt.as_ref().map(|shape| shape.rtl);
let rtl_opt = self.lines[self.cursor.line].shape_opt.as_ref().map(|shape| shape.rtl);
if let Some(rtl) = rtl_opt {
if rtl {
self.action(TextAction::Previous);
@ -621,8 +616,8 @@ impl<'a> TextBuffer<'a> {
let mut cursor = self.layout_cursor(&self.cursor);
if cursor.layout > 0 {
cursor.layout -= 1;
} else if cursor.line.get() > 0 {
cursor.line = TextLineIndex::new(cursor.line.get() - 1);
} else if cursor.line > 0 {
cursor.line -= 1;
cursor.layout = usize::max_value();
}
self.set_layout_cursor(cursor);
@ -631,7 +626,7 @@ impl<'a> TextBuffer<'a> {
//TODO: make this preserve X as best as possible!
let mut cursor = self.layout_cursor(&self.cursor);
let layout_len = {
let line = &mut self.lines[cursor.line.get()];
let line = &mut self.lines[cursor.line];
let layout = line.layout(
&self.font_matches,
self.metrics.font_size,
@ -641,8 +636,8 @@ impl<'a> TextBuffer<'a> {
};
if cursor.layout + 1 < layout_len {
cursor.layout += 1;
} else if cursor.line.get() + 1 < self.lines.len() {
cursor.line = TextLineIndex::new(cursor.line.get() + 1);
} else if cursor.line + 1 < self.lines.len() {
cursor.line += 1;
cursor.layout = 0;
}
self.set_layout_cursor(cursor);
@ -678,7 +673,7 @@ impl<'a> TextBuffer<'a> {
// Filter out special chars (except for tab), use TextAction instead
log::debug!("Refusing to insert control character {:?}", character);
} else {
let line = &mut self.lines[self.cursor.line.get()];
let line = &mut self.lines[self.cursor.line];
line.reset();
line.text.insert(self.cursor.index, character);
@ -687,20 +682,20 @@ impl<'a> TextBuffer<'a> {
},
TextAction::Enter => {
let new_line = {
let line = &mut self.lines[self.cursor.line.get()];
let line = &mut self.lines[self.cursor.line];
line.reset();
line.text.split_off(self.cursor.index)
};
let next_line = self.cursor.line.get() + 1;
let next_line = self.cursor.line + 1;
self.lines.insert(next_line, TextBufferLine::new(new_line));
self.cursor.line = TextLineIndex::new(next_line);
self.cursor.line = next_line;
self.cursor.index = 0;
},
TextAction::Backspace => {
if self.cursor.index > 0 {
let line = &mut self.lines[self.cursor.line.get()];
let line = &mut self.lines[self.cursor.line];
line.reset();
// Find previous character index
@ -716,23 +711,23 @@ impl<'a> TextBuffer<'a> {
self.cursor.index = prev_index;
line.text.remove(self.cursor.index);
} else if self.cursor.line.get() > 0 {
let mut line_index = self.cursor.line.get();
} else if self.cursor.line > 0 {
let mut line_index = self.cursor.line;
let old_line = self.lines.remove(line_index);
line_index -= 1;
let line = &mut self.lines[line_index];
line.reset();
self.cursor.line = TextLineIndex::new(line_index);
self.cursor.line = line_index;
self.cursor.index = line.text.len();
line.text.push_str(&old_line.text);
}
},
TextAction::Delete => {
if self.cursor.index < self.lines[self.cursor.line.get()].text.len() {
let line = &mut self.lines[self.cursor.line.get()];
if self.cursor.index < self.lines[self.cursor.line].text.len() {
let line = &mut self.lines[self.cursor.line];
line.reset();
if let Some((i, c)) = line
@ -744,10 +739,10 @@ impl<'a> TextBuffer<'a> {
line.text.replace_range(i..(i + c.len()), "");
self.cursor.index = i;
}
} else if self.cursor.line.get() + 1 < self.lines.len() {
let old_line = self.lines.remove(self.cursor.line.get() + 1);
} else if self.cursor.line + 1 < self.lines.len() {
let old_line = self.lines.remove(self.cursor.line + 1);
let line = &mut self.lines[self.cursor.line.get()];
let line = &mut self.lines[self.cursor.line];
line.reset();
line.text.push_str(&old_line.text);
@ -860,7 +855,7 @@ impl<'a> TextBuffer<'a> {
let text_glyph = &run.text[glyph.start..glyph.end];
log::debug!(
"{}, {}: '{}' ('{}'): '{}' ({:?})",
self.cursor.line.get(),
self.cursor.line,
self.cursor.index,
font_opt.as_ref().map_or("?", |font| font.info.family.as_str()),
font_opt.as_ref().map_or("?", |font| font.info.post_script_name.as_str()),
@ -1045,7 +1040,7 @@ impl<'a> TextBuffer<'a> {
for glyph in run.glyphs.iter() {
let (cache_key, x_int, y_int) = (glyph.cache_key, glyph.x_int, glyph.y_int);
cache.with_pixels(&self.font_matches, cache_key, color, |x, y, color| {
cache.with_pixels(cache_key, color, |x, y, color| {
f(x_int + x, line_y + y_int + y, 1, 1, color)
});
}

View file

@ -5,13 +5,14 @@ use std::{
sync::{Arc, Mutex},
};
use crate::{Attrs, Family, Font, FontMatches};
use crate::{Attrs, Font, FontMatches};
/// Access system fonts
pub struct FontSystem<'a> {
pub locale: String,
pub db: fontdb::Database,
pub font_cache: Mutex<HashMap<fontdb::ID, Option<Arc<Font<'a>>>>>,
pub font_matches_cache: Mutex<HashMap<Attrs<'a>, Arc<FontMatches<'a>>>>,
}
impl<'a> FontSystem<'a> {
@ -60,6 +61,7 @@ impl<'a> FontSystem<'a> {
locale,
db,
font_cache: Mutex::new(HashMap::new()),
font_matches_cache: Mutex::new(HashMap::new()),
}
}
@ -77,48 +79,33 @@ impl<'a> FontSystem<'a> {
}).clone()
}
pub fn matches<F: Fn(&fontdb::FaceInfo) -> bool>(
&'a self,
default_family: &Family,
f: F,
) -> FontMatches<'_> {
let mut fonts = Vec::new();
for face in self.db.faces() {
if !f(face) {
continue;
pub fn get_font_matches(&'a self, attrs: Attrs<'a>) -> Arc<FontMatches<'a>> {
let mut font_matches_cache = self.font_matches_cache.lock().unwrap();
font_matches_cache.entry(attrs).or_insert_with(|| {
let now = std::time::Instant::now();
let mut fonts = Vec::new();
for face in self.db.faces() {
if !attrs.matches(face) {
continue;
}
match self.get_font(face.id) {
Some(font) => fonts.push(font),
None => (),
}
}
match self.get_font(face.id) {
Some(font) => fonts.push(font),
None => (),
}
}
let font_matches = Arc::new(FontMatches {
locale: &self.locale,
default_family: self.db.family_name(&attrs.family).to_string(),
fonts
});
FontMatches {
locale: &self.locale,
default_family: self.db.family_name(default_family).to_string(),
fonts
}
}
let elapsed = now.elapsed();
log::debug!("font matches for {:?} in {:?}", attrs, elapsed);
pub fn matches_attrs(&'a self, attrs: &Attrs) -> FontMatches<'_> {
self.matches(&attrs.family, |face| {
let matched = attrs.matches(face);
if matched {
log::debug!(
"{:?}: family '{}' postscript name '{}' style {:?} weight {:?} stretch {:?} monospaced {:?}",
face.id,
face.family,
face.post_script_name,
face.style,
face.weight,
face.stretch,
face.monospaced
);
}
matched
})
font_matches
}).clone()
}
}

View file

@ -5,12 +5,12 @@ use swash::scale::{ScaleContext, image::Content};
use swash::scale::{Render, Source, StrikeWith};
use swash::zeno::{Format, Vector};
use crate::{CacheKey, FontMatches};
use crate::{CacheKey, FontSystem};
pub use swash::scale::image::{Content as SwashContent, Image as SwashImage};
fn swash_image(context: &mut ScaleContext, matches: &FontMatches, cache_key: CacheKey) -> Option<SwashImage> {
let font = match matches.get_font(&cache_key.font_id) {
fn swash_image<'a>(font_system: &'a FontSystem<'a>, context: &mut ScaleContext, cache_key: CacheKey) -> Option<SwashImage> {
let font = match font_system.get_font(cache_key.font_id) {
Some(some) => some,
None => {
log::warn!("did not find font {:?}", cache_key.font_id);
@ -47,41 +47,42 @@ fn swash_image(context: &mut ScaleContext, matches: &FontMatches, cache_key: Cac
.render(&mut scaler, cache_key.glyph_id)
}
pub struct SwashCache {
pub struct SwashCache<'a> {
font_system: &'a FontSystem<'a>,
context: ScaleContext,
pub image_cache: HashMap<CacheKey, Option<SwashImage>>,
}
impl SwashCache {
impl<'a> SwashCache<'a> {
/// Create a new swash cache
pub fn new() -> Self {
pub fn new(font_system: &'a FontSystem<'a>) -> Self {
Self {
font_system: font_system,
context: ScaleContext::new(),
image_cache: HashMap::new()
}
}
/// Create a swash Image from a cache key, without caching results
pub fn get_image_uncached(&mut self, matches: &FontMatches, cache_key: CacheKey) -> Option<SwashImage> {
swash_image(&mut self.context, matches, cache_key)
pub fn get_image_uncached(&mut self, cache_key: CacheKey) -> Option<SwashImage> {
swash_image(self.font_system, &mut self.context, cache_key)
}
/// Create a swash Image from a cache key, caching results
pub fn get_image(&mut self, matches: &FontMatches, cache_key: CacheKey) -> &Option<SwashImage> {
pub fn get_image(&mut self, cache_key: CacheKey) -> &Option<SwashImage> {
self.image_cache.entry(cache_key).or_insert_with(|| {
swash_image(&mut self.context, matches, cache_key)
swash_image(self.font_system, &mut self.context, cache_key)
})
}
/// Enumerate pixels in an Image, use `with_image` for better performance
pub fn with_pixels<F: FnMut(i32, i32, u32)>(
&mut self,
matches: &FontMatches<'_>,
cache_key: CacheKey,
base: u32,
mut f: F
) {
if let Some(image) = self.get_image(matches, cache_key) {
if let Some(image) = self.get_image(cache_key) {
let x = image.placement.left;
let y = -image.placement.top;