From aa5d84eb2be71e674f65fa635304147563d51886 Mon Sep 17 00:00:00 2001 From: Daniel Collin Date: Fri, 18 Dec 2015 20:21:47 +0100 Subject: [PATCH] Added support for key repeat * Added get_keys_pressed(KeyRepeat::Yes/No) in order for the user to decide if repeat should be used or not * added is_key_pressed(key, KeyRepeat::Yes/No) * Added set_key_repeat_delay/rate in order to control the rate/delays of keys --- Cargo.toml | 3 +- examples/noise.rs | 12 +++++++ src/lib.rs | 9 ++++++ src/windows.rs | 80 ++++++++++++++++++++++++++++++++++++++++++++++- 4 files changed, 102 insertions(+), 2 deletions(-) diff --git a/Cargo.toml b/Cargo.toml index 6333080..066f0ca 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -8,8 +8,9 @@ build = "build.rs" gcc = "0.3.19" [dependencies] -libc = "0.1.10" +libc = "0.2" user32-sys = "0.1.2" winapi = "0.2.4" kernel32-sys = "0.1.4" gdi32-sys = "0.1.1" +time = "0.1.34" diff --git a/examples/noise.rs b/examples/noise.rs index 7142140..fd9c4a7 100644 --- a/examples/noise.rs +++ b/examples/noise.rs @@ -31,6 +31,18 @@ fn main() { noise &= 0xFF; *i = (noise << 16) | (noise << 8) | noise; } + + window.get_keys_pressed(KeyRepeat::No).map(|keys| { + for t in keys.iter() { + match *t { + Key::W => println!("holding w!"), + Key::T => println!("holding t!"), + _ => (), + } + } + }); + + window.update(&buffer); } } diff --git a/src/lib.rs b/src/lib.rs index b7ba5a1..d93872f 100644 --- a/src/lib.rs +++ b/src/lib.rs @@ -46,6 +46,15 @@ pub enum Vsync { BestGuess, } +/// Used for is_key_pressed and get_keys_pressed() to indicated if repeat of presses is wanted +#[derive(PartialEq, Clone, Copy)] +pub enum KeyRepeat { + /// Use repeat + Yes, + /// Don't use repeat + No, +} + pub enum Key { Key0 = 0, Key1 = 1, diff --git a/src/windows.rs b/src/windows.rs index edd0ffa..dd8893d 100644 --- a/src/windows.rs +++ b/src/windows.rs @@ -4,10 +4,12 @@ extern crate user32; extern crate kernel32; extern crate winapi; extern crate gdi32; +extern crate time; use Scale; use Vsync; use Key; +use KeyRepeat; use std::ptr; use std::os::windows::ffi::OsStrExt; @@ -216,11 +218,16 @@ pub struct Window { dc: Option, window: Option, keys: [bool; 512], + keys_down_duration: [f32; 512], buffer: Vec, is_open : bool, scale_factor: i32, width: i32, height: i32, + prev_time: f64, + delta_time: f32, + key_repeat_delay: f32, + key_repeat_rate: f32, } impl Window { @@ -308,6 +315,11 @@ impl Window { dc: Some(user32::GetDC(handle.unwrap())), window: Some(handle.unwrap()), keys: [false; 512], + keys_down_duration: [-1.0; 512], + prev_time: time::precise_time_s(), + delta_time: 0.0, + key_repeat_delay: 0.250, + key_repeat_rate: 0.050, buffer: Vec::new(), is_open: true, scale_factor: scale_factor, @@ -336,11 +348,61 @@ impl Window { Some(keys) } + pub fn get_keys_pressed(&self, repeat: KeyRepeat) -> Option> { + let mut index: u16 = 0; + let mut keys: Vec = Vec::new(); + + for i in self.keys.iter() { + if *i { + unsafe { + if Self::key_pressed(self, index as usize, repeat) { + keys.push(mem::transmute(index as u8)); + } + } + } + + index += 1; + } + + Some(keys) + } + #[inline] pub fn is_key_down(&self, key: Key) -> bool { return self.keys[key as usize]; } + #[inline] + pub fn set_key_repeat_delay(&mut self, delay: f32) { + self.key_repeat_delay = delay; + } + + #[inline] + pub fn set_key_repeat_rate(&mut self, rate: f32) { + self.key_repeat_rate = rate; + } + + pub fn key_pressed(&self, index: usize, repeat: KeyRepeat) -> bool { + let t = self.keys_down_duration[index]; + + if t == 0.0 { + return true; + } + + if repeat == KeyRepeat::Yes && t > self.key_repeat_delay { + let delay = self.key_repeat_delay; + let rate = self.key_repeat_rate; + if ((((t - delay) % rate) > rate * 0.5)) != (((t - delay - self.delta_time) % rate) > rate * 0.5) { + return true; + } + } + return false; + } + + pub fn is_key_pressed(&self, key: Key, repeat: KeyRepeat) -> bool { + return Self::key_pressed(self, key as usize, repeat); + } + #[inline] pub fn is_open(&self) -> bool { return self.is_open @@ -351,6 +413,23 @@ impl Window { let mut msg = mem::uninitialized(); let window = self.window.unwrap(); + let current_time = time::precise_time_s(); + let delta_time = (current_time - self.prev_time) as f32; + self.prev_time = current_time; + self.delta_time = delta_time; + + for i in 0..self.keys.len() { + if self.keys[i] { + if self.keys_down_duration[i] < 0.0 { + self.keys_down_duration[i] = 0.0; + } else { + self.keys_down_duration[i] += delta_time; + } + } else { + self.keys_down_duration[i] = -1.0; + } + } + // TODO: Optimize self.buffer = buffer.iter().cloned().collect(); @@ -366,7 +445,6 @@ impl Window { } unsafe fn get_scale_factor(width: usize, height: usize, scale: Scale) -> i32 { - // TODO: Implement best fit let factor: i32 = match scale { Scale::X1 => 1, Scale::X2 => 2,