cache rendered chars

This commit is contained in:
Vinzenz Schroeter 2025-07-06 20:42:21 +02:00
parent 4eed089a06
commit 42042ec502
5 changed files with 46 additions and 28 deletions

4
Cargo.lock generated
View file

@ -1521,9 +1521,7 @@ dependencies = [
[[package]] [[package]]
name = "servicepoint" name = "servicepoint"
version = "0.15.2" version = "0.16.0"
source = "registry+https://github.com/rust-lang/crates.io-index"
checksum = "7d1e88713031e003dc3ee708dbb282e36714eee466a12d311d0e2e24c61c7118"
dependencies = [ dependencies = [
"bitvec", "bitvec",
"bzip2", "bzip2",

View file

@ -19,9 +19,6 @@ env_logger = "0.11"
clap = { version = "4.5", features = ["derive"] } clap = { version = "4.5", features = ["derive"] }
thiserror = "2.0" thiserror = "2.0"
# package parsing
servicepoint = { features = ["all_compressions"], version = "0.15.2" }
# font rendering # font rendering
font-kit = "0.14.2" font-kit = "0.14.2"
# I should not need this as a direct dependency, but then I cannot spell the types needed to use font-kit... # I should not need this as a direct dependency, but then I cannot spell the types needed to use font-kit...
@ -32,6 +29,11 @@ winit = "0.30"
# for drawing pixels onto the surface of the window # for drawing pixels onto the surface of the window
softbuffer = "0.4.6" softbuffer = "0.4.6"
[dependencies.servicepoint]
version = "0.16.0"
features = ["all_compressions"]
path = "../servicepoint-uniffi3"
[profile.release] [profile.release]
lto = true # Enable link-time optimization lto = true # Enable link-time optimization
codegen-units = 1 # Reduce number of codegen units to increase optimizations codegen-units = 1 # Reduce number of codegen units to increase optimizations

View file

@ -5,10 +5,11 @@ use crate::{
}; };
use log::{debug, error, info, trace, warn}; use log::{debug, error, info, trace, warn};
use servicepoint::{ use servicepoint::{
BinaryOperation, BitVecCommand, Bitmap, BitmapCommand, GlobalBrightnessCommand, BinaryOperation, BitVecCommand, Bitmap, BitmapCommand, BrightnessGrid,
BrightnessGrid, BrightnessGridCommand, CharGridCommand, ClearCommand, BrightnessGridCommand, CharGridCommand, ClearCommand, CompressionCode,
CompressionCode, Cp437GridCommand, FadeOutCommand, Grid, HardResetCommand, Cp437GridCommand, FadeOutCommand, GlobalBrightnessCommand,
Origin, TypedCommand, PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE, HardResetCommand, Origin, TypedCommand, PIXEL_COUNT, PIXEL_WIDTH,
TILE_SIZE,
}; };
use std::{ use std::{
ops::{BitAnd, BitOr, BitXor}, ops::{BitAnd, BitOr, BitXor},
@ -199,8 +200,14 @@ impl CommandExecute for CharGridCommand {
if let Err(e) = context.font_renderer.render( if let Err(e) = context.font_renderer.render(
char, char,
&mut display, &mut display
Origin::new(tile_x * TILE_SIZE, tile_y * TILE_SIZE), .window_mut(
tile_x * TILE_SIZE,
tile_y * TILE_SIZE,
TILE_SIZE,
TILE_SIZE,
)
.unwrap(),
) { ) {
error!( error!(
"stopping drawing text because char draw failed: {e}" "stopping drawing text because char draw failed: {e}"

View file

@ -12,8 +12,11 @@ use pathfinder_geometry::{
transform2d::Transform2F, transform2d::Transform2F,
vector::{vec2f, vec2i}, vector::{vec2f, vec2i},
}; };
use servicepoint::{Bitmap, Grid, Origin, Pixels, TILE_SIZE}; use servicepoint::{Bitmap, GridMut, WindowMut, TILE_SIZE};
use std::sync::{Mutex, MutexGuard}; use std::{
collections::HashMap,
sync::{Mutex, MutexGuard},
};
#[derive(Debug)] #[derive(Debug)]
struct SendFont(Font); struct SendFont(Font);
@ -32,6 +35,7 @@ pub struct FontRenderer8x8 {
font: SendFont, font: SendFont,
canvas: Mutex<Canvas>, canvas: Mutex<Canvas>,
fallback_char: Option<u32>, fallback_char: Option<u32>,
cache: Mutex<HashMap<char, Bitmap>>,
} }
#[derive(Debug, thiserror::Error)] #[derive(Debug, thiserror::Error)]
@ -56,6 +60,7 @@ impl FontRenderer8x8 {
font: SendFont(font), font: SendFont(font),
fallback_char, fallback_char,
canvas: Mutex::new(canvas), canvas: Mutex::new(canvas),
cache: Mutex::new(HashMap::new()),
} }
} }
@ -74,9 +79,13 @@ impl FontRenderer8x8 {
pub fn render( pub fn render(
&self, &self,
char: char, char: char,
bitmap: &mut Bitmap, target: &mut WindowMut<bool, Bitmap>,
offset: Origin<Pixels>,
) -> Result<(), RenderError> { ) -> Result<(), RenderError> {
let cache = &mut *self.cache.lock().unwrap();
if let Some(drawn_char) = cache.get(&char) {
target.deref_assign(drawn_char);
}
let glyph_id = self.get_glyph(char)?; let glyph_id = self.get_glyph(char)?;
let mut canvas = self.canvas.lock().unwrap(); let mut canvas = self.canvas.lock().unwrap();
@ -91,20 +100,21 @@ impl FontRenderer8x8 {
RasterizationOptions::Bilevel, RasterizationOptions::Bilevel,
)?; )?;
Self::copy_to_bitmap(canvas, bitmap, offset) let mut bitmap = Bitmap::new(TILE_SIZE, TILE_SIZE).unwrap();
Self::copy_to_bitmap(canvas, &mut bitmap)?;
target.deref_assign(&bitmap);
cache.insert(char, bitmap);
Ok(())
} }
fn copy_to_bitmap( fn copy_to_bitmap(
canvas: MutexGuard<Canvas>, canvas: MutexGuard<Canvas>,
bitmap: &mut Bitmap, bitmap: &mut Bitmap,
offset: Origin<Pixels>,
) -> Result<(), RenderError> { ) -> Result<(), RenderError> {
for y in 0..TILE_SIZE { for y in 0..TILE_SIZE {
for x in 0..TILE_SIZE { for x in 0..TILE_SIZE {
let canvas_val = canvas.pixels[x + y * TILE_SIZE] != 0; let canvas_val = canvas.pixels[x + y * TILE_SIZE] != 0;
let bitmap_x = offset.x + x; if !bitmap.set_optional(x, y, canvas_val) {
let bitmap_y = offset.y + y;
if !bitmap.set_optional(bitmap_x, bitmap_y, canvas_val) {
return Err(OutOfBounds(x, y)); return Err(OutOfBounds(x, y));
} }
} }

View file

@ -24,6 +24,7 @@ const PIXEL_HEIGHT_WITH_SPACERS: usize =
PIXEL_HEIGHT + NUM_SPACERS * SPACER_HEIGHT; PIXEL_HEIGHT + NUM_SPACERS * SPACER_HEIGHT;
const OFF_COLOR: u32 = u32::from_ne_bytes([0u8, 0, 0, 0]); const OFF_COLOR: u32 = u32::from_ne_bytes([0u8, 0, 0, 0]);
const SPACER_COLOR: u32 = u32::from_ne_bytes([100u8, 100, 100, 0]);
#[derive(Debug)] #[derive(Debug)]
pub enum AppEvents { pub enum AppEvents {
@ -61,7 +62,7 @@ impl<'t> Gui<'t> {
if self.options.spacers && tile_y != 0 { if self.options.spacers && tile_y != 0 {
// cannot just frame.skip(PIXEL_WIDTH as usize * SPACER_HEIGHT as usize) because of typing // cannot just frame.skip(PIXEL_WIDTH as usize * SPACER_HEIGHT as usize) because of typing
for _ in 0..PIXEL_WIDTH * SPACER_HEIGHT { for _ in 0..PIXEL_WIDTH * SPACER_HEIGHT {
frame.next().unwrap(); *frame.next().unwrap() = SPACER_COLOR;
} }
} }