2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
extern crate rusttype;
|
|
|
|
|
2016-09-21 05:52:45 +02:00
|
|
|
use std::cmp;
|
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
use primitive::{fast_set32, fast_set64, fast_copy, fast_copy64};
|
2016-09-21 05:52:45 +02:00
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
use self::rusttype::{Font, FontCollection, Scale, point};
|
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(not(feature="rusttype"))]
|
|
|
|
static FONT: &'static [u8] = include_bytes!("../../../res/fonts/unifont.font");
|
2016-09-21 05:52:45 +02:00
|
|
|
|
|
|
|
/// A display
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(not(feature="rusttype"))]
|
|
|
|
pub struct Display {
|
|
|
|
pub width: usize,
|
|
|
|
pub height: usize,
|
|
|
|
pub onscreen: &'static mut [u32],
|
|
|
|
pub offscreen: &'static mut [u32]
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A display
|
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-21 05:52:45 +02:00
|
|
|
pub struct Display {
|
|
|
|
pub width: usize,
|
|
|
|
pub height: usize,
|
|
|
|
pub onscreen: &'static mut [u32],
|
|
|
|
pub offscreen: &'static mut [u32],
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
pub font: Font<'static>,
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
pub font_bold: Font<'static>,
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
pub font_bold_italic: Font<'static>,
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
pub font_italic: Font<'static>
|
2016-09-21 05:52:45 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
impl Display {
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(not(feature="rusttype"))]
|
|
|
|
pub fn new(width: usize, height: usize, onscreen: &'static mut [u32], offscreen: &'static mut [u32]) -> Display {
|
|
|
|
Display {
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
onscreen: onscreen,
|
|
|
|
offscreen: offscreen
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-21 05:52:45 +02:00
|
|
|
pub fn new(width: usize, height: usize, onscreen: &'static mut [u32], offscreen: &'static mut [u32]) -> Display {
|
|
|
|
Display {
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
onscreen: onscreen,
|
|
|
|
offscreen: offscreen,
|
2016-09-29 21:17:19 +02:00
|
|
|
font: FontCollection::from_bytes(include_bytes!("../../../res/fonts/DejaVuSansMono.ttf")).into_font().unwrap(),
|
|
|
|
font_bold: FontCollection::from_bytes(include_bytes!("../../../res/fonts/DejaVuSansMono-Bold.ttf")).into_font().unwrap(),
|
|
|
|
font_bold_italic: FontCollection::from_bytes(include_bytes!("../../../res/fonts/DejaVuSansMono-BoldOblique.ttf")).into_font().unwrap(),
|
|
|
|
font_italic: FontCollection::from_bytes(include_bytes!("../../../res/fonts/DejaVuSansMono-Oblique.ttf")).into_font().unwrap()
|
2016-09-21 05:52:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Draw a rectangle
|
|
|
|
pub fn rect(&mut self, x: usize, y: usize, w: usize, h: usize, color: u32) {
|
|
|
|
let start_y = cmp::min(self.height - 1, y);
|
|
|
|
let end_y = cmp::min(self.height, y + h);
|
|
|
|
|
|
|
|
let start_x = cmp::min(self.width - 1, x);
|
|
|
|
let len = cmp::min(self.width, x + w) - start_x;
|
|
|
|
|
|
|
|
let mut offscreen_ptr = self.offscreen.as_mut_ptr() as usize;
|
|
|
|
|
|
|
|
let stride = self.width * 4;
|
|
|
|
|
|
|
|
let offset = y * stride + start_x * 4;
|
|
|
|
offscreen_ptr += offset;
|
|
|
|
|
|
|
|
let mut rows = end_y - start_y;
|
|
|
|
while rows > 0 {
|
|
|
|
unsafe {
|
2016-09-21 20:18:48 +02:00
|
|
|
fast_set32(offscreen_ptr as *mut u32, color, len);
|
2016-09-21 05:52:45 +02:00
|
|
|
}
|
|
|
|
offscreen_ptr += stride;
|
|
|
|
rows -= 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Draw a character
|
2016-09-29 21:17:19 +02:00
|
|
|
#[cfg(not(feature="rusttype"))]
|
|
|
|
pub fn char(&mut self, x: usize, y: usize, character: char, color: u32, _bold: bool, _italic: bool) {
|
|
|
|
if x + 8 <= self.width && y + 16 <= self.height {
|
2016-09-29 21:44:34 +02:00
|
|
|
let mut dst = self.offscreen.as_mut_ptr() as usize + (y * self.width + x) * 4;
|
2016-09-29 21:17:19 +02:00
|
|
|
|
|
|
|
let font_i = 16 * (character as usize);
|
|
|
|
if font_i + 16 <= FONT.len() {
|
|
|
|
for row in 0..16 {
|
|
|
|
let row_data = FONT[font_i + row];
|
|
|
|
for col in 0..8 {
|
|
|
|
if (row_data >> (7 - col)) & 1 == 1 {
|
2016-09-29 21:44:34 +02:00
|
|
|
unsafe { *((dst + col * 4) as *mut u32) = color; }
|
2016-09-29 21:17:19 +02:00
|
|
|
}
|
|
|
|
}
|
2016-09-29 21:44:34 +02:00
|
|
|
dst += self.width * 4;
|
2016-09-29 21:17:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Draw a character
|
|
|
|
#[cfg(feature="rusttype")]
|
2016-09-23 00:15:38 +02:00
|
|
|
pub fn char(&mut self, x: usize, y: usize, character: char, color: u32, bold: bool, italic: bool) {
|
|
|
|
let width = self.width;
|
|
|
|
let height = self.height;
|
2016-09-29 21:17:19 +02:00
|
|
|
let offscreen = self.offscreen.as_mut_ptr() as usize;
|
2016-09-23 00:15:38 +02:00
|
|
|
|
|
|
|
let font = if bold && italic {
|
|
|
|
&self.font_bold_italic
|
|
|
|
} else if bold {
|
|
|
|
&self.font_bold
|
|
|
|
} else if italic {
|
|
|
|
&self.font_italic
|
|
|
|
} else {
|
|
|
|
&self.font
|
|
|
|
};
|
|
|
|
|
|
|
|
if let Some(glyph) = font.glyph(character){
|
|
|
|
let scale = Scale::uniform(16.0);
|
|
|
|
let v_metrics = font.v_metrics(scale);
|
|
|
|
let point = point(0.0, v_metrics.ascent);
|
2016-09-23 00:57:26 +02:00
|
|
|
let glyph = glyph.scaled(scale).positioned(point);
|
|
|
|
if let Some(bb) = glyph.pixel_bounding_box() {
|
|
|
|
glyph.draw(|off_x, off_y, v| {
|
|
|
|
let off_x = x + (off_x as i32 + bb.min.x) as usize;
|
|
|
|
let off_y = y + (off_y as i32 + bb.min.y) as usize;
|
|
|
|
// There's still a possibility that the glyph clips the boundaries of the bitmap
|
|
|
|
if off_x < width && off_y < height {
|
2016-09-28 20:19:30 +02:00
|
|
|
if v > 0.0 {
|
|
|
|
let f_a = (v * 255.0) as u32;
|
|
|
|
let f_r = (((color >> 16) & 0xFF) * f_a)/255;
|
|
|
|
let f_g = (((color >> 8) & 0xFF) * f_a)/255;
|
|
|
|
let f_b = ((color & 0xFF) * f_a)/255;
|
2016-09-23 00:57:26 +02:00
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
let offscreen_ptr = (offscreen + (off_y * width + off_x) * 4) as *mut u32;
|
2016-09-28 20:19:30 +02:00
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
let bg = unsafe { *offscreen_ptr };
|
2016-09-28 20:19:30 +02:00
|
|
|
|
|
|
|
let b_a = 255 - f_a;
|
|
|
|
let b_r = (((bg >> 16) & 0xFF) * b_a)/255;
|
|
|
|
let b_g = (((bg >> 8) & 0xFF) * b_a)/255;
|
|
|
|
let b_b = ((bg & 0xFF) * b_a)/255;
|
|
|
|
|
|
|
|
let c = ((f_r + b_r) << 16) | ((f_g + b_g) << 8) | (f_b + b_b);
|
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
unsafe { *offscreen_ptr = c; }
|
2016-09-23 00:57:26 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-09-21 05:52:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Scroll display
|
|
|
|
pub fn scroll(&mut self, rows: usize, color: u32) {
|
|
|
|
let data = (color as u64) << 32 | color as u64;
|
|
|
|
|
|
|
|
let width = self.width/2;
|
|
|
|
let height = self.height;
|
|
|
|
if rows > 0 && rows < height {
|
|
|
|
let off1 = rows * width;
|
|
|
|
let off2 = height * width - off1;
|
|
|
|
unsafe {
|
|
|
|
let data_ptr = self.offscreen.as_mut_ptr() as *mut u64;
|
|
|
|
fast_copy64(data_ptr, data_ptr.offset(off1 as isize), off2);
|
|
|
|
fast_set64(data_ptr.offset(off2 as isize), data, off1);
|
2016-09-29 21:17:19 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Copy from offscreen to onscreen
|
|
|
|
pub fn sync(&mut self, x: usize, y: usize, w: usize, h: usize) {
|
|
|
|
let start_y = cmp::min(self.height - 1, y);
|
|
|
|
let end_y = cmp::min(self.height, y + h);
|
|
|
|
|
|
|
|
let start_x = cmp::min(self.width - 1, x);
|
|
|
|
let len = (cmp::min(self.width, x + w) - start_x) * 4;
|
|
|
|
|
|
|
|
let mut offscreen_ptr = self.offscreen.as_mut_ptr() as usize;
|
|
|
|
let mut onscreen_ptr = self.onscreen.as_mut_ptr() as usize;
|
2016-09-21 05:52:45 +02:00
|
|
|
|
2016-09-29 21:17:19 +02:00
|
|
|
let stride = self.width * 4;
|
|
|
|
|
|
|
|
let offset = y * stride + start_x * 4;
|
|
|
|
offscreen_ptr += offset;
|
|
|
|
onscreen_ptr += offset;
|
|
|
|
|
|
|
|
let mut rows = end_y - start_y;
|
|
|
|
while rows > 0 {
|
|
|
|
unsafe {
|
|
|
|
fast_copy(onscreen_ptr as *mut u8, offscreen_ptr as *const u8, len);
|
2016-09-21 05:52:45 +02:00
|
|
|
}
|
2016-09-29 21:17:19 +02:00
|
|
|
offscreen_ptr += stride;
|
|
|
|
onscreen_ptr += stride;
|
|
|
|
rows -= 1;
|
2016-09-21 05:52:45 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|