include font from the real display

This commit is contained in:
Vinzenz Schroeter 2024-10-13 13:11:12 +02:00
parent 828b486f03
commit 5d8a6004da
5 changed files with 308 additions and 20 deletions

View file

@ -1,6 +1,6 @@
use std::sync::{RwLock, RwLockWriteGuard}; use std::sync::{RwLock, RwLockWriteGuard};
use log::{debug, error, info, warn}; use log::{debug, error, info, trace, warn};
use servicepoint::{ use servicepoint::{
BrightnessGrid, Command, Cp437Grid, Grid, Origin, PixelGrid, Tiles, BrightnessGrid, Command, Cp437Grid, Grid, Origin, PixelGrid, Tiles,
PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE, PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE,
@ -126,6 +126,11 @@ fn print_cp437_data(
for char_y in 0usize..grid.height() { for char_y in 0usize..grid.height() {
for char_x in 0usize..grid.width() { for char_x in 0usize..grid.width() {
let char_code = grid.get(char_x, char_y); let char_code = grid.get(char_x, char_y);
trace!(
"drawing char_code {:#04x} (if this was UTF-8, it would be {})",
char_code,
char::from(char_code)
);
let tile_x = char_x + x; let tile_x = char_x + x;
let tile_y = char_y + y; let tile_y = char_y + y;

View file

@ -1,5 +1,4 @@
use std::sync::Arc; use crate::static_font;
use font_kit::canvas::{Canvas, Format, RasterizationOptions}; use font_kit::canvas::{Canvas, Format, RasterizationOptions};
use font_kit::font::Font; use font_kit::font::Font;
use font_kit::hinting::HintingOptions; use font_kit::hinting::HintingOptions;
@ -9,14 +8,24 @@ use servicepoint::{Grid, PixelGrid, TILE_SIZE};
const DEFAULT_FONT_FILE: &[u8] = include_bytes!("../Web437_IBM_BIOS.woff"); const DEFAULT_FONT_FILE: &[u8] = include_bytes!("../Web437_IBM_BIOS.woff");
const CHAR_COUNT: usize = u8::MAX as usize + 1;
pub struct BitmapFont { pub struct BitmapFont {
bitmaps: [PixelGrid; u8::MAX as usize + 1], bitmaps: [PixelGrid; CHAR_COUNT],
} }
impl BitmapFont { impl BitmapFont {
pub fn load(font: Font) -> BitmapFont { pub fn new(bitmaps: [PixelGrid; CHAR_COUNT]) -> Self {
Self { bitmaps }
}
pub fn load(font: Font, size: usize) -> BitmapFont {
let mut bitmaps = let mut bitmaps =
core::array::from_fn(|_| PixelGrid::new(TILE_SIZE, TILE_SIZE)); core::array::from_fn(|_| PixelGrid::new(TILE_SIZE, TILE_SIZE));
let mut canvas =
Canvas::new(vec2i(size as i32, size as i32), Format::A8);
let size_f = size as f32;
let transform = Transform2F::default();
for char_code in u8::MIN..=u8::MAX { for char_code in u8::MIN..=u8::MAX {
let char = char_code as char; let char = char_code as char;
@ -25,33 +34,31 @@ impl BitmapFont {
Some(val) => val, Some(val) => val,
}; };
let size = 8f32; canvas.pixels.fill(0);
let transform = Transform2F::default();
let mut canvas =
Canvas::new(vec2i(size as i32, size as i32), Format::A8);
font.rasterize_glyph( font.rasterize_glyph(
&mut canvas, &mut canvas,
glyph_id, glyph_id,
size, size_f,
Transform2F::from_translation(vec2f(0f32, size)) * transform, Transform2F::from_translation(vec2f(0f32, size_f)) * transform,
HintingOptions::None, HintingOptions::None,
RasterizationOptions::GrayscaleAa, RasterizationOptions::GrayscaleAa,
) )
.unwrap(); .unwrap();
assert_eq!(canvas.pixels.len(), 64); assert_eq!(canvas.pixels.len(), size * size);
assert_eq!(canvas.stride, 8); assert_eq!(canvas.stride, size);
let bitmap = &mut bitmaps[char_code as usize];
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 index = x + y * TILE_SIZE; let index = x + y * TILE_SIZE;
let canvas_val = canvas.pixels[index] != 0; let canvas_val = canvas.pixels[index] != 0;
bitmaps[char_code as usize].set(x, y, canvas_val); bitmap.set(x, y, canvas_val);
} }
} }
} }
BitmapFont { bitmaps } Self::new(bitmaps)
} }
pub fn get_bitmap(&self, char_code: u8) -> &PixelGrid { pub fn get_bitmap(&self, char_code: u8) -> &PixelGrid {
@ -61,8 +68,6 @@ impl BitmapFont {
impl Default for BitmapFont { impl Default for BitmapFont {
fn default() -> Self { fn default() -> Self {
let font = Font::from_bytes(Arc::new(DEFAULT_FONT_FILE.to_vec()), 0) static_font::load_static()
.expect("could not load included font");
Self::load(font)
} }
} }

View file

@ -67,7 +67,8 @@ impl<'t> App<'t> {
for x in 0..PIXEL_WIDTH { for x in 0..PIXEL_WIDTH {
let is_set = display.get(x, y); let is_set = display.get(x, y);
let brightness: u8 = luma.get(x / TILE_SIZE, y / TILE_SIZE).into(); let brightness: u8 =
luma.get(x / TILE_SIZE, y / TILE_SIZE).into();
let max_brightness: u8 = Brightness::MAX.into(); let max_brightness: u8 = Brightness::MAX.into();
let scale: f32 = (u8::MAX as f32) / (max_brightness as f32); let scale: f32 = (u8::MAX as f32) / (max_brightness as f32);

View file

@ -17,6 +17,7 @@ use crate::gui::{App, AppEvents};
mod execute_command; mod execute_command;
mod font; mod font;
mod gui; mod gui;
mod static_font;
#[derive(Parser, Debug)] #[derive(Parser, Debug)]
struct Cli { struct Cli {

276
src/static_font.rs Normal file
View file

@ -0,0 +1,276 @@
use crate::font::BitmapFont;
use servicepoint::{DataRef, PixelGrid, TILE_SIZE};
/// Font from the display firmware `cape-cccb-apd/cp437font_linear.h`
pub(crate) const CP437_FONT_LINEAR: [u64; 256] = [
0x0000000000000000, // 0x00
0x003854ba82aa4438, // 0x01
0x003844bafed67c38, // 0x02
0x0010387cfeee4400, // 0x03
0x0010387cfe7c3810, // 0x04
0x003810d6fefe3838, // 0x05
0x003810d6fe7c3810, // 0x06
0x0000387c7c7c3800, // 0x07
0x00fec6828282c6fe, // 0x08
0x0000384444443800, // 0x09
0x00fec6bababac6fe, // 0x0a
0x007088888a7a061e, // 0x0b
0x1038103844444438, // 0x0c
0x0030301010141418, // 0x0d
0xc0c64642724e720e, // 0x0e
0x00927c44c6447c92, // 0x0f
0x00c0f0fcfefcf0c0, // 0x10
0x00061e7efe7e1e06, // 0x11
0x1038541010543810, // 0x12
0x2800282828282828, // 0x13
0x000e0a0a7a8a8a7e, // 0x14
0x0038441c28704438, // 0x15
0x000000ffff000000, // 0x16
0xfe10385410543810, // 0x17
0x1010101010543810, // 0x18
0x1038541010101010, // 0x19
0x00000804fe040800, // 0x1a
0x00002040fe402000, // 0x1b
0xffffc0c0c0c0c0c0, // 0x1c
0x00002844fe442800, // 0x1d
0x00fefe7c7c383810, // 0x1e
0x001038387c7cfefe, // 0x1f
0x0000000000000000, // 0x20
0x1000101010101010, // 0x21
0x0000000000505028, // 0x22
0x00247e24247e2400, // 0x23
0x1038541830543810, // 0x24
0x00844a2a54a8a442, // 0x25
0x003a444a32484830, // 0x26
0x0000000000201010, // 0x27
0x0810202020201008, // 0x28
0x2010080808081020, // 0x29
0x0010543854100000, // 0x2a
0x0010107c10100000, // 0x2b
0x2010100000000000, // 0x2c
0x0000007c00000000, // 0x2d
0x0010000000000000, // 0x2e
0x4020201010080804, // 0x2f
0x0038444454444438, // 0x30
0x007c101010503010, // 0x31
0x007c201008044438, // 0x32
0x003844043810087c, // 0x33
0x0004047e24140c04, // 0x34
0x003844047840407c, // 0x35
0x003844447840201c, // 0x36
0x004020100804047c, // 0x37
0x0038444438444438, // 0x38
0x007008043c444438, // 0x39
0x0000100000100000, // 0x3a
0x2010100000100000, // 0x3b
0x0006186018060000, // 0x3c
0x00007c007c000000, // 0x3d
0x00c0300c30c00000, // 0x3e
0x1000384038044438, // 0x3f
0x001c2248564a221c, // 0x40
0x0042427e24241818, // 0x41
0x007c42427c444478, // 0x42
0x001c22404040221c, // 0x43
0x0078444242424478, // 0x44
0x007e40407e40407e, // 0x45
0x004040407e40407e, // 0x46
0x001c22424e40221c, // 0x47
0x004242427e424242, // 0x48
0x007c10101010107c, // 0x49
0x003844040404047c, // 0x4a
0x0042444870484442, // 0x4b
0x007e404040404040, // 0x4c
0x0082828292aac682, // 0x4d
0x004242464a526242, // 0x4e
0x0018244242422418, // 0x4f
0x004040407c42427c, // 0x50
0x001a244a42422418, // 0x51
0x004244487c42427c, // 0x52
0x003c42023c40423c, // 0x53
0x00101010101010fe, // 0x54
0x003c424242424242, // 0x55
0x0010282844448282, // 0x56
0x0044446caa929282, // 0x57
0x0082442810284482, // 0x58
0x0010101010284482, // 0x59
0x007e20100804027e, // 0x5a
0x3820202020202038, // 0x5b
0x0408081010202040, // 0x5c
0x3808080808080838, // 0x5d
0x0000000000442810, // 0x5e
0x007e000000000000, // 0x5f
0x0000000000080810, // 0x60
0x003c443c04380000, // 0x61
0x0038444444784040, // 0x62
0x0038444044380000, // 0x63
0x003c4444443c0404, // 0x64
0x003c407844380000, // 0x65
0x2020202078202418, // 0x66
0x78043c44443e0000, // 0x67
0x0044444464584040, // 0x68
0x001c101010700010, // 0x69
0x38440404047c0010, // 0x6a
0x0022243828242020, // 0x6b
0x0018242020202020, // 0x6c
0x0054545454780000, // 0x6d
0x0044444464580000, // 0x6e
0x0038444444380000, // 0x6f
0x4040784444780000, // 0x70
0x04043c44443c0000, // 0x71
0x0040404064580000, // 0x72
0x00384418201c0000, // 0x73
0x0018242020782020, // 0x74
0x0038444444440000, // 0x75
0x0010282844440000, // 0x76
0x0028285454440000, // 0x77
0x0044281028440000, // 0x78
0x38043c4444440000, // 0x79
0x007c2010087c0000, // 0x7a
0x0c0808083008080c, // 0x7b
0x1010101010101010, // 0x7c
0x301010100c101030, // 0x7d
0x0000004c32000000, // 0x7e
0xfe82828282442810, // 0x7f
0x18083c428080423c, // 0x80
0x0038444444440028, // 0x81
0x003c407844381008, // 0x82
0x003c443c04382810, // 0x83
0x003c443c04380044, // 0x84
0x003c443c04381020, // 0x85
0x003c443c04382838, // 0x86
0x1038444044380000, // 0x87
0x003c407844382810, // 0x88
0x003c407844380044, // 0x89
0x003c407844381020, // 0x8a
0x001c101010700028, // 0x8b
0x001c101010702810, // 0x8c
0x001c101010701020, // 0x8d
0x0082827c44282892, // 0x8e
0x0082827c44281038, // 0x8f
0x00fe80fe80fe1008, // 0x90
0x007e907c126c0000, // 0x91
0x008e88784e28281e, // 0x92
0x0038444438002810, // 0x93
0x0038444444380028, // 0x94
0x0038444438001020, // 0x95
0x0038444444002810, // 0x96
0x0038444444001020, // 0x97
0x38043c4444440028, // 0x98
0x0038448282443882, // 0x99
0x0038448282820082, // 0x9a
0x1038444044381010, // 0x9b
0x00fc4240f0404438, // 0x9c
0x107c107c10284482, // 0x9d
0x001c22f840f8221c, // 0x9e
0x205010103810120c, // 0x9f
0x003c443c04381008, // 0xa0
0x001c101010701008, // 0xa1
0x0038444438001008, // 0xa2
0x0038444444001008, // 0xa3
0x0044446458004834, // 0xa4
0x00464a5262424834, // 0xa5
0x0000007090701060, // 0xa6
0x0000007088888870, // 0xa7
0x3844403804380010, // 0xa8
0x0040407c00000000, // 0xa9
0x0004047c00000000, // 0xaa
0x0e84422c5048c442, // 0xab
0x049e542c5448c442, // 0xac
0x1010101010100010, // 0xad
0x0024489048240000, // 0xae
0x0048241224480000, // 0xaf
0x1122448811224488, // 0xb0
0x55aa55aa55aa55aa, // 0xb1
0xddbb77eeddbb77ee, // 0xb2
0x1010101010101010, // 0xb3
0x101020c020101010, // 0xb4
0x1020c000c0201010, // 0xb5
0x2828488848282828, // 0xb6
0x282850e000000000, // 0xb7
0x1030d020c0000000, // 0xb8
0x2848880888482828, // 0xb9
0x2828282828282828, // 0xba
0x2828c810e0000000, // 0xbb
0x000000e010c82828, // 0xbc
0x000000e050282828, // 0xbd
0x0000c020d0301010, // 0xbe
0x101020c000000000, // 0xbf
0x0000000708101010, // 0xc0
0x000000c728101010, // 0xc1
0x101028c700000000, // 0xc2
0x1010080708101010, // 0xc3
0x000000ff00000000, // 0xc4
0x101028c728101010, // 0xc5
0x1008070007081010, // 0xc6
0x2828242324282828, // 0xc7
0x00000f1027282828, // 0xc8
0x282827100f000000, // 0xc9
0x0000ff0083442828, // 0xca
0x28448300ff000000, // 0xcb
0x2824232023242828, // 0xcc
0x0000ff00ff000000, // 0xcd
0x2844932893442828, // 0xce
0x0000ff00c7281010, // 0xcf
0x0000008344282828, // 0xd0
0x1028c700ff000000, // 0xd1
0x2828448300000000, // 0xd2
0x0000000f14282828, // 0xd3
0x0000070817181010, // 0xd4
0x1018170807000000, // 0xd5
0x2828140f00000000, // 0xd6
0x2828448344282828, // 0xd7
0x1028c700c7281010, // 0xd8
0x000000c020101010, // 0xd9
0x1010080700000000, // 0xda
0xffffffffffffffff, // 0xdb
0xffffffff00000000, // 0xdc
0xf0f0f0f0f0f0f0f0, // 0xdd
0x0f0f0f0f0f0f0f0f, // 0xde
0x00000000ffffffff, // 0xdf
0x0076888888740200, // 0xe0
0x5844444458484830, // 0xe1
0x00e04040404242fe, // 0xe2
0x00242828a87c0000, // 0xe3
0x00fe8240204082fe, // 0xe4
0x00384444443e0000, // 0xe5
0x405a644444440000, // 0xe6
0x0010282020fc0000, // 0xe7
0xfe103854543810fe, // 0xe8
0x003844aabaaa4438, // 0xe9
0x00ee448282824438, // 0xea
0x007088887012221c, // 0xeb
0x00006c92926c0000, // 0xec
0x10107c92924c0000, // 0xed
0x0038403040380000, // 0xee
0x0082828282824438, // 0xef
0x00fe00fe00fe0000, // 0xf0
0x007c10107c101000, // 0xf1
0x007e006018061860, // 0xf2
0x007e000618601806, // 0xf3
0x101010101010120c, // 0xf4
0x6090101010101010, // 0xf5
0x0010007c00100000, // 0xf6
0x000c926c92600000, // 0xf7
0x0000000030484830, // 0xf8
0x0000103810000000, // 0xf9
0x0000001000000000, // 0xfa
0x10102828a4440202, // 0xfb
0x00000000484848b0, // 0xfc
0x0000000070201060, // 0xfd
0x00007c7c7c7c7c00, // 0xfe
0x0000000000000000, // 0xff
];
pub fn load_static() -> BitmapFont {
let mut bitmaps =
core::array::from_fn(|_| PixelGrid::new(TILE_SIZE, TILE_SIZE));
for (char_code, bitmap) in bitmaps.iter_mut().enumerate() {
let bits = CP437_FONT_LINEAR[char_code];
let mut bytes = bits.to_be_bytes();
bytes.reverse();
bitmap.data_ref_mut().copy_from_slice(bytes.as_slice());
}
BitmapFont::new(bitmaps)
}