6 changed files with 120 additions and 107 deletions
@ -0,0 +1,95 @@ |
|||||||
|
/// Displayrotation
|
||||||
|
#[derive(Clone, Copy)] |
||||||
|
pub enum DisplayRotation { |
||||||
|
/// No rotation
|
||||||
|
Rotate0, |
||||||
|
/// Rotate by 90 degrees clockwise
|
||||||
|
Rotate90, |
||||||
|
/// Rotate by 180 degrees clockwise
|
||||||
|
Rotate180, |
||||||
|
/// Rotate 270 degrees clockwise
|
||||||
|
Rotate270, |
||||||
|
} |
||||||
|
|
||||||
|
impl Default for DisplayRotation { |
||||||
|
fn default() -> Self { |
||||||
|
DisplayRotation::Rotate0 |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
pub trait Display { |
||||||
|
fn buffer(&self) -> &[u8]; |
||||||
|
fn set_rotation(&mut self, rotation: DisplayRotation); |
||||||
|
fn rotation(&self) -> DisplayRotation; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
pub(crate) fn outside_display(x: u32, y: u32, width: u32, height: u32, rotation: DisplayRotation) -> bool { |
||||||
|
match rotation { |
||||||
|
DisplayRotation::Rotate0 | DisplayRotation::Rotate180 => { |
||||||
|
if x >= width || y >= height { |
||||||
|
return true; |
||||||
|
} |
||||||
|
}, |
||||||
|
DisplayRotation::Rotate90 | DisplayRotation::Rotate270 => { |
||||||
|
if y >= width || x >= height { |
||||||
|
return true; |
||||||
|
}
|
||||||
|
} |
||||||
|
} |
||||||
|
false |
||||||
|
} |
||||||
|
|
||||||
|
pub(crate) fn rotation(x: u32, y: u32, width: u32, height: u32, rotation: DisplayRotation) -> (u32, u8) { |
||||||
|
match rotation { |
||||||
|
DisplayRotation::Rotate0 => ( |
||||||
|
x / 8 + (width / 8) * y, |
||||||
|
0x80 >> (x % 8), |
||||||
|
), |
||||||
|
DisplayRotation::Rotate90 => ( |
||||||
|
(width - 1 - y) / 8 + (width / 8) * x, |
||||||
|
0x01 << (y % 8), |
||||||
|
), |
||||||
|
DisplayRotation::Rotate180 => ( |
||||||
|
((width / 8) * height - 1) - (x / 8 + (width / 8) * y), |
||||||
|
0x01 << (x % 8), |
||||||
|
), |
||||||
|
DisplayRotation::Rotate270 => ( |
||||||
|
y / 8 + (height - 1 - x) * (width / 8), |
||||||
|
0x80 >> (y % 8), |
||||||
|
), |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
#[cfg(test)] |
||||||
|
mod tests { |
||||||
|
use super::{DisplayRotation, outside_display, rotation}; |
||||||
|
|
||||||
|
#[test] |
||||||
|
fn rotation_overflow() { |
||||||
|
use epd4in2::constants::{WIDTH, HEIGHT}; |
||||||
|
let width = WIDTH as u32; |
||||||
|
let height = HEIGHT as u32; |
||||||
|
test_rotation_overflow(width, height, DisplayRotation::Rotate0); |
||||||
|
test_rotation_overflow(width, height, DisplayRotation::Rotate90); |
||||||
|
test_rotation_overflow(width, height, DisplayRotation::Rotate180); |
||||||
|
test_rotation_overflow(width, height, DisplayRotation::Rotate270); |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
fn test_rotation_overflow(width: u32, height: u32, rotation2: DisplayRotation) { |
||||||
|
let max_value = width / 8 * height; |
||||||
|
for x in 0..(width + height) { //limit x because it runs too long
|
||||||
|
for y in 0..(u32::max_value()) { |
||||||
|
if outside_display(x, y, width, height, rotation2) { |
||||||
|
break; |
||||||
|
} else { |
||||||
|
let (idx, _) = rotation(x, y, width, height, rotation2); |
||||||
|
assert!(idx < max_value); |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
|
} |
||||||
Loading…
Reference in new issue