mirror of
https://github.com/kaesaecracker/servicepoint-simulator.git
synced 2025-01-18 02:20:14 +01:00
update dependencies, initial UTF-8 support
Some checks failed
Rust / build (push) Has been cancelled
Some checks failed
Rust / build (push) Has been cancelled
This commit is contained in:
parent
398cb8c165
commit
ad5f1e8abe
628
Cargo.lock
generated
628
Cargo.lock
generated
File diff suppressed because it is too large
Load diff
|
@ -11,13 +11,14 @@ log = "0.4"
|
|||
env_logger = "0.11"
|
||||
clap = { version = "4.5", features = ["derive"] }
|
||||
# for drawing pixels onto the surface of the window
|
||||
pixels = "0.14"
|
||||
pixels = "0.15"
|
||||
# I should not need this as a direct dependency, but then I cannot spell the types needed to use font-kit...
|
||||
pathfinder_geometry = "0.5.1"
|
||||
font-kit = "0.14.2"
|
||||
thiserror = "2.0"
|
||||
|
||||
[dependencies.servicepoint]
|
||||
version = "0.12.0"
|
||||
version = "0.13.0"
|
||||
features = ["all_compressions"]
|
||||
|
||||
[dependencies.winit]
|
||||
|
|
|
@ -139,7 +139,7 @@
|
|||
{
|
||||
default = pkgs.mkShell rec {
|
||||
inputsFrom = [ self.packages.${system}.default ];
|
||||
packages = [ rust-toolchain ];
|
||||
packages = [ rust-toolchain pkgs.gdb ];
|
||||
LD_LIBRARY_PATH = "${pkgs.lib.makeLibraryPath (builtins.concatMap (d: d.buildInputs) inputsFrom)}";
|
||||
RUST_SRC_PATH = "${pkgs.rust.packages.stable.rustPlatform.rustLibSrc}";
|
||||
};
|
||||
|
|
|
@ -1,16 +1,17 @@
|
|||
use std::sync::{RwLock, RwLockWriteGuard};
|
||||
|
||||
use log::{debug, error, info, trace, warn};
|
||||
use servicepoint::{
|
||||
Bitmap, BrightnessGrid, Command, Cp437Grid, Grid, Origin, Tiles,
|
||||
Bitmap, BrightnessGrid, CharGrid, Command, Cp437Grid, Grid, Origin, Tiles,
|
||||
PIXEL_COUNT, PIXEL_WIDTH, TILE_SIZE,
|
||||
};
|
||||
use std::sync::{RwLock, RwLockWriteGuard};
|
||||
|
||||
use crate::font::BitmapFont;
|
||||
use crate::font::Cp437Font;
|
||||
use crate::font_renderer::FontRenderer8x8;
|
||||
|
||||
pub(crate) fn execute_command(
|
||||
command: Command,
|
||||
font: &BitmapFont,
|
||||
cp436_font: &Cp437Font,
|
||||
utf8_font: &FontRenderer8x8,
|
||||
display_ref: &RwLock<Bitmap>,
|
||||
luma_ref: &RwLock<BrightnessGrid>,
|
||||
) -> bool {
|
||||
|
@ -30,7 +31,7 @@ pub(crate) fn execute_command(
|
|||
}
|
||||
Command::Cp437Data(origin, grid) => {
|
||||
let mut display = display_ref.write().unwrap();
|
||||
print_cp437_data(origin, &grid, font, &mut display);
|
||||
print_cp437_data(origin, &grid, cp436_font, &mut display);
|
||||
}
|
||||
#[allow(deprecated)]
|
||||
Command::BitmapLegacy => {
|
||||
|
@ -99,6 +100,10 @@ pub(crate) fn execute_command(
|
|||
Command::FadeOut => {
|
||||
error!("command not implemented: {command:?}")
|
||||
}
|
||||
Command::Utf8Data(origin, grid) => {
|
||||
let mut display = display_ref.write().unwrap();
|
||||
print_utf8_data(origin, &grid, utf8_font, &mut display);
|
||||
}
|
||||
};
|
||||
|
||||
true
|
||||
|
@ -116,7 +121,7 @@ fn check_bitmap_valid(offset: u16, payload_len: usize) -> bool {
|
|||
fn print_cp437_data(
|
||||
origin: Origin<Tiles>,
|
||||
grid: &Cp437Grid,
|
||||
font: &BitmapFont,
|
||||
font: &Cp437Font,
|
||||
display: &mut RwLockWriteGuard<Bitmap>,
|
||||
) {
|
||||
let Origin { x, y, .. } = origin;
|
||||
|
@ -145,6 +150,33 @@ fn print_cp437_data(
|
|||
}
|
||||
}
|
||||
|
||||
fn print_utf8_data(
|
||||
origin: Origin<Tiles>,
|
||||
grid: &CharGrid,
|
||||
font: &FontRenderer8x8,
|
||||
display: &mut RwLockWriteGuard<Bitmap>,
|
||||
) {
|
||||
let Origin { x, y, .. } = origin;
|
||||
for char_y in 0usize..grid.height() {
|
||||
for char_x in 0usize..grid.width() {
|
||||
let char = grid.get(char_x, char_y);
|
||||
trace!("drawing {char}");
|
||||
|
||||
let tile_x = char_x + x;
|
||||
let tile_y = char_y + y;
|
||||
|
||||
if let Err(e) = font.render(
|
||||
char,
|
||||
display,
|
||||
Origin::new(tile_x * TILE_SIZE, tile_y * TILE_SIZE),
|
||||
) {
|
||||
error!("stopping drawing text because char draw failed: {e}");
|
||||
return;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
fn print_pixel_grid(
|
||||
offset_x: usize,
|
||||
offset_y: usize,
|
||||
|
|
334
src/font.rs
334
src/font.rs
|
@ -1,73 +1,297 @@
|
|||
use crate::static_font;
|
||||
use font_kit::canvas::{Canvas, Format, RasterizationOptions};
|
||||
use font_kit::font::Font;
|
||||
use font_kit::hinting::HintingOptions;
|
||||
use pathfinder_geometry::transform2d::Transform2F;
|
||||
use pathfinder_geometry::vector::{vec2f, vec2i};
|
||||
use servicepoint::{Bitmap, Grid, TILE_SIZE};
|
||||
|
||||
const DEFAULT_FONT_FILE: &[u8] = include_bytes!("../Web437_IBM_BIOS.woff");
|
||||
use servicepoint::{Bitmap, DataRef, TILE_SIZE};
|
||||
|
||||
const CHAR_COUNT: usize = u8::MAX as usize + 1;
|
||||
|
||||
pub struct BitmapFont {
|
||||
pub struct Cp437Font {
|
||||
bitmaps: [Bitmap; CHAR_COUNT],
|
||||
}
|
||||
|
||||
impl BitmapFont {
|
||||
impl Cp437Font {
|
||||
pub fn new(bitmaps: [Bitmap; CHAR_COUNT]) -> Self {
|
||||
Self { bitmaps }
|
||||
}
|
||||
|
||||
pub fn load(font: Font, size: usize) -> BitmapFont {
|
||||
let mut bitmaps =
|
||||
core::array::from_fn(|_| Bitmap::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 {
|
||||
let char = char_code as char;
|
||||
let glyph_id = match font.glyph_for_char(char) {
|
||||
None => continue,
|
||||
Some(val) => val,
|
||||
};
|
||||
|
||||
canvas.pixels.fill(0);
|
||||
font.rasterize_glyph(
|
||||
&mut canvas,
|
||||
glyph_id,
|
||||
size_f,
|
||||
Transform2F::from_translation(vec2f(0f32, size_f)) * transform,
|
||||
HintingOptions::None,
|
||||
RasterizationOptions::GrayscaleAa,
|
||||
)
|
||||
.unwrap();
|
||||
|
||||
assert_eq!(canvas.pixels.len(), size * size);
|
||||
assert_eq!(canvas.stride, size);
|
||||
|
||||
let bitmap = &mut bitmaps[char_code as usize];
|
||||
for y in 0..TILE_SIZE {
|
||||
for x in 0..TILE_SIZE {
|
||||
let index = x + y * TILE_SIZE;
|
||||
let canvas_val = canvas.pixels[index] != 0;
|
||||
bitmap.set(x, y, canvas_val);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Self::new(bitmaps)
|
||||
}
|
||||
|
||||
pub fn get_bitmap(&self, char_code: u8) -> &Bitmap {
|
||||
&self.bitmaps[char_code as usize]
|
||||
}
|
||||
}
|
||||
|
||||
impl Default for BitmapFont {
|
||||
impl Default for Cp437Font {
|
||||
fn default() -> Self {
|
||||
static_font::load_static()
|
||||
load_static()
|
||||
}
|
||||
}
|
||||
|
||||
/// 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
|
||||
];
|
||||
|
||||
fn load_static() -> Cp437Font {
|
||||
let mut bitmaps =
|
||||
core::array::from_fn(|_| Bitmap::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());
|
||||
}
|
||||
|
||||
Cp437Font::new(bitmaps)
|
||||
}
|
||||
|
|
94
src/font_renderer.rs
Normal file
94
src/font_renderer.rs
Normal file
|
@ -0,0 +1,94 @@
|
|||
use crate::font_renderer::RenderError::{GlyphNotFound, OutOfBounds};
|
||||
use font_kit::canvas::{Canvas, Format, RasterizationOptions};
|
||||
use font_kit::error::GlyphLoadingError;
|
||||
use font_kit::family_name::FamilyName;
|
||||
use font_kit::font::Font;
|
||||
use font_kit::hinting::HintingOptions;
|
||||
use font_kit::properties::Properties;
|
||||
use font_kit::source::SystemSource;
|
||||
use pathfinder_geometry::transform2d::Transform2F;
|
||||
use pathfinder_geometry::vector::{vec2f, vec2i};
|
||||
use servicepoint::{Bitmap, Grid, Origin, Pixels, TILE_SIZE};
|
||||
use std::sync::Mutex;
|
||||
|
||||
pub struct FontRenderer8x8 {
|
||||
font: Font,
|
||||
canvas: Mutex<Canvas>,
|
||||
fallback_char: Option<u32>,
|
||||
}
|
||||
|
||||
#[derive(Debug, thiserror::Error)]
|
||||
pub enum RenderError {
|
||||
#[error("Glyph not found for '{0}'")]
|
||||
GlyphNotFound(char),
|
||||
#[error(transparent)]
|
||||
GlyphLoadingError(#[from] GlyphLoadingError),
|
||||
#[error("out of bounds at {0} {1}")]
|
||||
OutOfBounds(usize, usize),
|
||||
}
|
||||
|
||||
impl FontRenderer8x8 {
|
||||
pub fn new(font: Font, fallback_char: Option<char>) -> Self {
|
||||
let canvas =
|
||||
Canvas::new(vec2i(TILE_SIZE as i32, TILE_SIZE as i32), Format::A8);
|
||||
assert_eq!(canvas.pixels.len(), TILE_SIZE * TILE_SIZE);
|
||||
assert_eq!(canvas.stride, TILE_SIZE);
|
||||
let fallback_char = fallback_char.and_then(|c| font.glyph_for_char(c));
|
||||
let result = Self {
|
||||
font,
|
||||
fallback_char,
|
||||
canvas: Mutex::new(canvas),
|
||||
};
|
||||
result
|
||||
}
|
||||
|
||||
pub fn render(
|
||||
&self,
|
||||
char: char,
|
||||
bitmap: &mut Bitmap,
|
||||
offset: Origin<Pixels>,
|
||||
) -> Result<(), RenderError> {
|
||||
let mut canvas = self.canvas.lock().unwrap();
|
||||
let glyph_id = self.font.glyph_for_char(char).or(self.fallback_char);
|
||||
let glyph_id = match glyph_id {
|
||||
None => return Err(GlyphNotFound(char)),
|
||||
Some(val) => val,
|
||||
};
|
||||
|
||||
canvas.pixels.fill(0);
|
||||
self.font.rasterize_glyph(
|
||||
&mut canvas,
|
||||
glyph_id,
|
||||
TILE_SIZE as f32,
|
||||
Transform2F::from_translation(vec2f(0f32, TILE_SIZE as f32))
|
||||
* Transform2F::default(),
|
||||
HintingOptions::None,
|
||||
RasterizationOptions::Bilevel,
|
||||
)?;
|
||||
|
||||
for y in 0..TILE_SIZE {
|
||||
for x in 0..TILE_SIZE {
|
||||
let index = x + y * TILE_SIZE;
|
||||
let canvas_val = canvas.pixels[index] != 0;
|
||||
let bitmap_x = (offset.x + x) as isize;
|
||||
let bitmap_y = (offset.y + y) as isize;
|
||||
if !bitmap.set_optional(bitmap_x, bitmap_y, canvas_val) {
|
||||
return Err(OutOfBounds(x, y));
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
Ok(())
|
||||
}
|
||||
}
|
||||
|
||||
impl Default for FontRenderer8x8 {
|
||||
fn default() -> Self {
|
||||
let utf8_font = SystemSource::new()
|
||||
.select_best_match(&[FamilyName::Monospace], &Properties::new())
|
||||
.unwrap()
|
||||
.load()
|
||||
.unwrap();
|
||||
FontRenderer8x8::new(utf8_font, Some('?'))
|
||||
}
|
||||
}
|
54
src/gui.rs
54
src/gui.rs
|
@ -20,9 +20,9 @@ pub struct App<'t> {
|
|||
display: &'t RwLock<Bitmap>,
|
||||
luma: &'t RwLock<BrightnessGrid>,
|
||||
window: Option<Window>,
|
||||
pixels: Option<Pixels>,
|
||||
stop_udp_tx: Sender<()>,
|
||||
cli: &'t Cli,
|
||||
logical_size: LogicalSize<u16>,
|
||||
}
|
||||
|
||||
const SPACER_HEIGHT: usize = 4;
|
||||
|
@ -40,18 +40,42 @@ impl<'t> App<'t> {
|
|||
stop_udp_tx: Sender<()>,
|
||||
cli: &'t Cli,
|
||||
) -> Self {
|
||||
let logical_size = {
|
||||
let height = if cli.spacers {
|
||||
let num_spacers = (PIXEL_HEIGHT / TILE_SIZE) - 1;
|
||||
PIXEL_HEIGHT + num_spacers * SPACER_HEIGHT
|
||||
} else {
|
||||
PIXEL_HEIGHT
|
||||
};
|
||||
LogicalSize::new(PIXEL_WIDTH as u16, height as u16)
|
||||
};
|
||||
|
||||
App {
|
||||
display,
|
||||
luma,
|
||||
stop_udp_tx,
|
||||
pixels: None,
|
||||
window: None,
|
||||
cli,
|
||||
logical_size,
|
||||
}
|
||||
}
|
||||
|
||||
fn draw(&mut self) {
|
||||
let pixels = self.pixels.as_mut().unwrap();
|
||||
let window = self.window.as_ref().unwrap();
|
||||
let mut pixels = {
|
||||
let window_size = window.inner_size();
|
||||
let surface_texture = SurfaceTexture::new(
|
||||
window_size.width,
|
||||
window_size.height,
|
||||
&window,
|
||||
);
|
||||
Pixels::new(
|
||||
self.logical_size.width as u32,
|
||||
self.logical_size.height as u32,
|
||||
surface_texture,
|
||||
)
|
||||
.unwrap()
|
||||
};
|
||||
|
||||
let mut frame = pixels.frame_mut().chunks_exact_mut(4);
|
||||
let display = self.display.read().unwrap();
|
||||
|
@ -95,35 +119,13 @@ impl<'t> App<'t> {
|
|||
|
||||
impl ApplicationHandler<AppEvents> for App<'_> {
|
||||
fn resumed(&mut self, event_loop: &ActiveEventLoop) {
|
||||
let height = if self.cli.spacers {
|
||||
let num_spacers = (PIXEL_HEIGHT / TILE_SIZE) - 1;
|
||||
PIXEL_HEIGHT + num_spacers * SPACER_HEIGHT
|
||||
} else {
|
||||
PIXEL_HEIGHT
|
||||
};
|
||||
|
||||
let size = LogicalSize::new(PIXEL_WIDTH as u16, height as u16);
|
||||
let attributes = Window::default_attributes()
|
||||
.with_title("servicepoint-simulator")
|
||||
.with_inner_size(size)
|
||||
.with_inner_size(self.logical_size)
|
||||
.with_transparent(false);
|
||||
|
||||
let window = event_loop.create_window(attributes).unwrap();
|
||||
self.window = Some(window);
|
||||
let window = self.window.as_ref().unwrap();
|
||||
|
||||
let pixels = {
|
||||
let window_size = window.inner_size();
|
||||
let surface_texture = SurfaceTexture::new(
|
||||
window_size.width,
|
||||
window_size.height,
|
||||
&window,
|
||||
);
|
||||
Pixels::new(size.width as u32, size.height as u32, surface_texture)
|
||||
.unwrap()
|
||||
};
|
||||
|
||||
self.pixels = Some(pixels);
|
||||
}
|
||||
|
||||
fn user_event(&mut self, event_loop: &ActiveEventLoop, event: AppEvents) {
|
||||
|
|
16
src/main.rs
16
src/main.rs
|
@ -11,13 +11,14 @@ use servicepoint::*;
|
|||
use winit::event_loop::{ControlFlow, EventLoop};
|
||||
|
||||
use crate::execute_command::execute_command;
|
||||
use crate::font::BitmapFont;
|
||||
use crate::font::Cp437Font;
|
||||
use crate::font_renderer::FontRenderer8x8;
|
||||
use crate::gui::{App, AppEvents};
|
||||
|
||||
mod execute_command;
|
||||
mod font;
|
||||
mod font_renderer;
|
||||
mod gui;
|
||||
mod static_font;
|
||||
|
||||
#[derive(Parser, Debug)]
|
||||
struct Cli {
|
||||
|
@ -56,14 +57,14 @@ fn main() {
|
|||
luma.fill(Brightness::MAX);
|
||||
let luma = RwLock::new(luma);
|
||||
|
||||
run(&display, &luma, socket, BitmapFont::default(), &cli);
|
||||
run(&display, &luma, socket, Cp437Font::default(), &cli);
|
||||
}
|
||||
|
||||
fn run(
|
||||
display_ref: &RwLock<Bitmap>,
|
||||
luma_ref: &RwLock<BrightnessGrid>,
|
||||
socket: UdpSocket,
|
||||
font: BitmapFont,
|
||||
cp437_font: Cp437Font,
|
||||
cli: &Cli,
|
||||
) {
|
||||
let (stop_udp_tx, stop_udp_rx) = mpsc::channel();
|
||||
|
@ -80,6 +81,7 @@ fn run(
|
|||
std::thread::scope(move |scope| {
|
||||
let udp_thread = scope.spawn(move || {
|
||||
let mut buf = [0; 8985];
|
||||
let utf8_font = FontRenderer8x8::default();
|
||||
|
||||
while stop_udp_rx.try_recv().is_err() {
|
||||
let (amount, _) = match socket.recv_from(&mut buf) {
|
||||
|
@ -98,8 +100,8 @@ fn run(
|
|||
);
|
||||
}
|
||||
|
||||
let package = match servicepoint::packet::Packet::try_from(&buf[..amount]) {
|
||||
Err(_) => {
|
||||
let package = match servicepoint::Packet::try_from(&buf[..amount]) {
|
||||
Err(_) => {
|
||||
warn!("could not load packet with length {amount} into header");
|
||||
continue;
|
||||
}
|
||||
|
@ -114,7 +116,7 @@ fn run(
|
|||
Ok(val) => val,
|
||||
};
|
||||
|
||||
if !execute_command(command, &font, display_ref, luma_ref) {
|
||||
if !execute_command(command, &cp437_font, &utf8_font, display_ref, luma_ref) {
|
||||
// hard reset
|
||||
event_proxy
|
||||
.send_event(AppEvents::UdpThreadClosed)
|
||||
|
|
|
@ -1,276 +0,0 @@
|
|||
use crate::font::BitmapFont;
|
||||
use servicepoint::{Bitmap, DataRef, 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(|_| Bitmap::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)
|
||||
}
|
Loading…
Reference in a new issue