|
|
|
@ -5,6 +5,7 @@ use crate::{
|
|
|
|
|
trezorhal::{display, qr, time},
|
|
|
|
|
ui::lerp::Lerp,
|
|
|
|
|
};
|
|
|
|
|
use core::slice;
|
|
|
|
|
|
|
|
|
|
use super::geometry::{Offset, Point, Rect};
|
|
|
|
|
|
|
|
|
@ -253,6 +254,104 @@ pub fn get_color_table(fg_color: Color, bg_color: Color) -> [Color; 16] {
|
|
|
|
|
table
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub struct Glyph {
|
|
|
|
|
pub width: i32,
|
|
|
|
|
pub height: i32,
|
|
|
|
|
pub adv: i32,
|
|
|
|
|
pub bearing_x: i32,
|
|
|
|
|
pub bearing_y: i32,
|
|
|
|
|
data: &'static [u8],
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Glyph {
|
|
|
|
|
/// Construct a `Glyph` from a raw pointer.
|
|
|
|
|
///
|
|
|
|
|
/// # Safety
|
|
|
|
|
///
|
|
|
|
|
/// This function is unsafe because the caller has to guarantee that `data`
|
|
|
|
|
/// is pointing to a memory containing a valid glyph data, that is:
|
|
|
|
|
/// - contains valid glyph metadata
|
|
|
|
|
/// - data has appropriate size
|
|
|
|
|
/// - data must have static lifetime
|
|
|
|
|
pub unsafe fn load(data: *const u8) -> Self {
|
|
|
|
|
unsafe {
|
|
|
|
|
let width = *data.offset(0) as i32;
|
|
|
|
|
let height = *data.offset(1) as i32;
|
|
|
|
|
|
|
|
|
|
let data_bits = constant::FONT_BPP * width * height;
|
|
|
|
|
|
|
|
|
|
let data_bytes = if data_bits % 8 == 0 {
|
|
|
|
|
data_bits / 8
|
|
|
|
|
} else {
|
|
|
|
|
(data_bits / 8) + 1
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
Glyph {
|
|
|
|
|
width,
|
|
|
|
|
height,
|
|
|
|
|
adv: *data.offset(2) as i32,
|
|
|
|
|
bearing_x: *data.offset(3) as i32,
|
|
|
|
|
bearing_y: *data.offset(4) as i32,
|
|
|
|
|
data: slice::from_raw_parts(data.offset(5), data_bytes as usize),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn print(&self, pos: Point, colortable: [Color; 16]) -> i32 {
|
|
|
|
|
let bearing = Offset::new(self.bearing_x, -self.bearing_y);
|
|
|
|
|
let size = Offset::new(self.width, self.height);
|
|
|
|
|
let pos_adj = pos + bearing;
|
|
|
|
|
let r = Rect::from_top_left_and_size(pos_adj, size);
|
|
|
|
|
|
|
|
|
|
let area = r.translate(get_offset());
|
|
|
|
|
let window = area.clamp(constant::screen());
|
|
|
|
|
|
|
|
|
|
set_window(window);
|
|
|
|
|
|
|
|
|
|
for y in window.y0..window.y1 {
|
|
|
|
|
for x in window.x0..window.x1 {
|
|
|
|
|
let p = Point::new(x, y);
|
|
|
|
|
let r = p - pos_adj;
|
|
|
|
|
let c = self.get_pixel_data(r);
|
|
|
|
|
pixeldata(colortable[c as usize]);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
self.adv
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn unpack_bpp1(&self, a: i32) -> u8 {
|
|
|
|
|
let c_data = self.data[(a / 8) as usize];
|
|
|
|
|
((c_data >> (7 - (a % 8))) & 0x01) * 15
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn unpack_bpp2(&self, a: i32) -> u8 {
|
|
|
|
|
let c_data = self.data[(a / 4) as usize];
|
|
|
|
|
((c_data >> (6 - (a % 4) * 2)) & 0x03) * 5
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn unpack_bpp4(&self, a: i32) -> u8 {
|
|
|
|
|
let c_data = self.data[(a / 2) as usize];
|
|
|
|
|
(c_data >> (4 - (a % 2) * 4)) & 0x0F
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn unpack_bpp8(&self, a: i32) -> u8 {
|
|
|
|
|
let c_data = self.data[a as usize];
|
|
|
|
|
c_data >> 4
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn get_pixel_data(&self, p: Offset) -> u8 {
|
|
|
|
|
let a = p.x + p.y * self.width;
|
|
|
|
|
|
|
|
|
|
match constant::FONT_BPP {
|
|
|
|
|
1 => self.unpack_bpp1(a),
|
|
|
|
|
2 => self.unpack_bpp2(a),
|
|
|
|
|
4 => self.unpack_bpp4(a),
|
|
|
|
|
8 => self.unpack_bpp8(a),
|
|
|
|
|
_ => 0,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
|
|
|
|
pub struct Font(i32);
|
|
|
|
|
|
|
|
|
@ -276,6 +375,27 @@ impl Font {
|
|
|
|
|
pub fn line_height(self) -> i32 {
|
|
|
|
|
constant::LINE_SPACE + self.text_height()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn get_glyph(self, char_byte: u8) -> Option<Glyph> {
|
|
|
|
|
let gl_data = display::get_char_glyph(char_byte, self.0);
|
|
|
|
|
|
|
|
|
|
if gl_data.is_null() {
|
|
|
|
|
return None;
|
|
|
|
|
}
|
|
|
|
|
unsafe { Some(Glyph::load(gl_data)) }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn display_text(self, text: &str, baseline: Point, fg_color: Color, bg_color: Color) {
|
|
|
|
|
let colortable = get_color_table(fg_color, bg_color);
|
|
|
|
|
let mut adv_total = 0;
|
|
|
|
|
for c in text.bytes() {
|
|
|
|
|
let g = self.get_glyph(c);
|
|
|
|
|
if let Some(gly) = g {
|
|
|
|
|
let adv = gly.print(baseline + Offset::new(adv_total, 0), colortable);
|
|
|
|
|
adv_total += adv;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Copy, Clone, PartialEq, Eq)]
|
|
|
|
|