2024-03-05 18:37:57 +05:30
|
|
|
use std::sync::mpsc::TryRecvError;
|
|
|
|
use crate::util::{EmulatorError, EmulatorResult};
|
|
|
|
|
|
|
|
|
2024-03-10 14:26:42 +05:30
|
|
|
/// An emulated keyboard that receives keyboard data as input
|
2024-03-05 18:37:57 +05:30
|
|
|
pub struct Keyboard {
|
2024-03-10 14:26:42 +05:30
|
|
|
/// Current keyboard state
|
2024-03-05 18:37:57 +05:30
|
|
|
bitflags: u16,
|
2024-03-10 14:26:42 +05:30
|
|
|
/// Receives keyboard events from main thread
|
2024-03-05 18:37:57 +05:30
|
|
|
keyboard_event_receiver: std::sync::mpsc::Receiver<KeyboardEvent>,
|
|
|
|
}
|
2025-02-14 08:51:59 +00:00
|
|
|
#[derive(Eq,PartialEq, PartialOrd, Clone, Copy, Debug)]
|
|
|
|
pub enum Key{
|
|
|
|
K0=0,K1,K2,K3,K4,K5,K6,K7,K8,K9,KA,KB,KC,KD,KE,KF
|
|
|
|
}
|
2024-03-05 18:37:57 +05:30
|
|
|
|
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
|
|
|
pub enum KeyboardEvent {
|
2025-02-14 08:51:59 +00:00
|
|
|
KeyUp(Key),
|
|
|
|
KeyDown(Key),
|
2024-03-05 18:37:57 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
impl Keyboard {
|
|
|
|
pub fn new(keyboard_event_receiver: std::sync::mpsc::Receiver<KeyboardEvent>) -> Keyboard {
|
|
|
|
Keyboard {
|
|
|
|
bitflags: 0,
|
|
|
|
keyboard_event_receiver,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-10 14:26:42 +05:30
|
|
|
/// Update keyboard based on pending keyboard events.
|
|
|
|
/// If no events are presents, it will return without any action.
|
2025-02-14 08:51:59 +00:00
|
|
|
pub fn update_keyboard_registers(&mut self) -> EmulatorResult<()> {
|
2024-03-05 18:37:57 +05:30
|
|
|
loop {
|
|
|
|
let keyboard_event_recv_res = self.keyboard_event_receiver.try_recv();
|
|
|
|
match keyboard_event_recv_res {
|
|
|
|
Ok(event) => {
|
2024-03-06 08:18:07 +05:30
|
|
|
log::debug!("Processing {:?}",event);
|
2024-03-05 18:37:57 +05:30
|
|
|
self.update_keyboard_state(event);
|
|
|
|
}
|
|
|
|
Err(TryRecvError::Empty) => {
|
|
|
|
break Ok(());
|
|
|
|
}
|
|
|
|
Err(TryRecvError::Disconnected) => {
|
|
|
|
break Err(EmulatorError::IOError("Keyboard updater disconnected".into()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Query if key is down
|
|
|
|
pub fn query_key_down(&self, key_num: u8) -> bool {
|
2025-02-16 12:09:17 +00:00
|
|
|
(self.bitflags & (1 << key_num)) == (1 << key_num)
|
2024-03-05 18:37:57 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
fn update_keyboard_state(&mut self, keyboard_event: KeyboardEvent) {
|
|
|
|
match keyboard_event {
|
2024-03-05 21:56:24 +05:30
|
|
|
KeyboardEvent::KeyUp(key) => {
|
2025-02-14 08:51:59 +00:00
|
|
|
self.bitflags &= !((1u16 << (key as u16)) as u16);
|
2024-03-05 21:56:24 +05:30
|
|
|
}
|
|
|
|
KeyboardEvent::KeyDown(key) => {
|
2025-02-14 08:51:59 +00:00
|
|
|
self.bitflags |= 1 << (key as u16);
|
2024-03-05 21:56:24 +05:30
|
|
|
}
|
2024-03-05 18:37:57 +05:30
|
|
|
}
|
|
|
|
}
|
2025-02-14 08:51:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
2025-02-16 12:09:17 +00:00
|
|
|
use super::{Key, Keyboard};
|
2025-02-14 08:51:59 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_key_assignment(){
|
|
|
|
assert_eq!(0,Key::K0 as u16);
|
|
|
|
assert_eq!(15,Key::KF as u16);
|
|
|
|
}
|
2025-02-16 12:09:17 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_key_at_nothing_pressed(){
|
|
|
|
let (_sender,receiver) = std::sync::mpsc::sync_channel(1);
|
|
|
|
let keyboard = Keyboard::new(receiver);
|
|
|
|
assert_no_key_pressed(&keyboard);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_key_down_then_up(){
|
|
|
|
let (sender,receiver) = std::sync::mpsc::sync_channel(1);
|
|
|
|
let mut keyboard = Keyboard::new(receiver);
|
|
|
|
assert_no_key_pressed(&keyboard);
|
|
|
|
|
|
|
|
|
|
|
|
sender.try_send(super::KeyboardEvent::KeyDown(Key::K0)).expect("Could not send");
|
|
|
|
keyboard.update_keyboard_registers().expect("Could not update keyboard");
|
|
|
|
|
|
|
|
|
|
|
|
assert_eq!(1,keyboard.bitflags);
|
|
|
|
assert_eq!(true,keyboard.query_key_down(0));
|
|
|
|
for i in 1..=0xF {
|
|
|
|
assert_eq!(false,keyboard.query_key_down(i));
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
sender.try_send(super::KeyboardEvent::KeyUp(Key::K0)).expect("Could not send");
|
|
|
|
keyboard.update_keyboard_registers().expect("Could not update keyboard");
|
|
|
|
assert_no_key_pressed(&keyboard);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn assert_no_key_pressed(keyboard: &Keyboard){
|
|
|
|
assert_eq!(0,keyboard.bitflags);
|
|
|
|
for i in 0..=0xF {
|
|
|
|
assert_eq!(false,keyboard.query_key_down(i),"Failed to match at index {}, bitflags at {}",i,keyboard.bitflags);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2025-02-14 08:51:59 +00:00
|
|
|
}
|