cache rendered chars
This commit is contained in:
parent
4eed089a06
commit
42042ec502
4
Cargo.lock
generated
4
Cargo.lock
generated
|
@ -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",
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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},
|
||||||
|
@ -46,11 +47,11 @@ impl CommandExecute for BitmapCommand {
|
||||||
fn execute(&self, context: &CommandExecutionContext) -> ExecutionResult {
|
fn execute(&self, context: &CommandExecutionContext) -> ExecutionResult {
|
||||||
let Self {
|
let Self {
|
||||||
origin:
|
origin:
|
||||||
Origin {
|
Origin {
|
||||||
x: offset_x,
|
x: offset_x,
|
||||||
y: offset_y,
|
y: offset_y,
|
||||||
..
|
..
|
||||||
},
|
},
|
||||||
bitmap: pixels,
|
bitmap: pixels,
|
||||||
..
|
..
|
||||||
} = self;
|
} = self;
|
||||||
|
@ -143,7 +144,7 @@ impl CommandExecute for Cp437GridCommand {
|
||||||
bitmap: context.cp437_font[char_code].clone(),
|
bitmap: context.cp437_font[char_code].clone(),
|
||||||
compression: CompressionCode::default(),
|
compression: CompressionCode::default(),
|
||||||
}
|
}
|
||||||
.execute(context);
|
.execute(context);
|
||||||
match execute_result {
|
match execute_result {
|
||||||
Success => {}
|
Success => {}
|
||||||
Failure => {
|
Failure => {
|
||||||
|
@ -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}"
|
||||||
|
|
|
@ -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));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue