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
This commit is contained in:
Daniel Collin 2015-12-18 20:21:47 +01:00
parent 2a05ff3659
commit aa5d84eb2b
4 changed files with 102 additions and 2 deletions

View file

@ -8,8 +8,9 @@ build = "build.rs"
gcc = "0.3.19" gcc = "0.3.19"
[dependencies] [dependencies]
libc = "0.1.10" libc = "0.2"
user32-sys = "0.1.2" user32-sys = "0.1.2"
winapi = "0.2.4" winapi = "0.2.4"
kernel32-sys = "0.1.4" kernel32-sys = "0.1.4"
gdi32-sys = "0.1.1" gdi32-sys = "0.1.1"
time = "0.1.34"

View file

@ -31,6 +31,18 @@ fn main() {
noise &= 0xFF; noise &= 0xFF;
*i = (noise << 16) | (noise << 8) | noise; *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); window.update(&buffer);
} }
} }

View file

@ -46,6 +46,15 @@ pub enum Vsync {
BestGuess, 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 { pub enum Key {
Key0 = 0, Key0 = 0,
Key1 = 1, Key1 = 1,

View file

@ -4,10 +4,12 @@ extern crate user32;
extern crate kernel32; extern crate kernel32;
extern crate winapi; extern crate winapi;
extern crate gdi32; extern crate gdi32;
extern crate time;
use Scale; use Scale;
use Vsync; use Vsync;
use Key; use Key;
use KeyRepeat;
use std::ptr; use std::ptr;
use std::os::windows::ffi::OsStrExt; use std::os::windows::ffi::OsStrExt;
@ -216,11 +218,16 @@ pub struct Window {
dc: Option<HDC>, dc: Option<HDC>,
window: Option<HWND>, window: Option<HWND>,
keys: [bool; 512], keys: [bool; 512],
keys_down_duration: [f32; 512],
buffer: Vec<u32>, buffer: Vec<u32>,
is_open : bool, is_open : bool,
scale_factor: i32, scale_factor: i32,
width: i32, width: i32,
height: i32, height: i32,
prev_time: f64,
delta_time: f32,
key_repeat_delay: f32,
key_repeat_rate: f32,
} }
impl Window { impl Window {
@ -308,6 +315,11 @@ impl Window {
dc: Some(user32::GetDC(handle.unwrap())), dc: Some(user32::GetDC(handle.unwrap())),
window: Some(handle.unwrap()), window: Some(handle.unwrap()),
keys: [false; 512], 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(), buffer: Vec::new(),
is_open: true, is_open: true,
scale_factor: scale_factor, scale_factor: scale_factor,
@ -336,11 +348,61 @@ impl Window {
Some(keys) Some(keys)
} }
pub fn get_keys_pressed(&self, repeat: KeyRepeat) -> Option<Vec<Key>> {
let mut index: u16 = 0;
let mut keys: Vec<Key> = 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] #[inline]
pub fn is_key_down(&self, key: Key) -> bool { pub fn is_key_down(&self, key: Key) -> bool {
return self.keys[key as usize]; 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] #[inline]
pub fn is_open(&self) -> bool { pub fn is_open(&self) -> bool {
return self.is_open return self.is_open
@ -351,6 +413,23 @@ impl Window {
let mut msg = mem::uninitialized(); let mut msg = mem::uninitialized();
let window = self.window.unwrap(); 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 // TODO: Optimize
self.buffer = buffer.iter().cloned().collect(); self.buffer = buffer.iter().cloned().collect();
@ -366,7 +445,6 @@ impl Window {
} }
unsafe fn get_scale_factor(width: usize, height: usize, scale: Scale) -> i32 { unsafe fn get_scale_factor(width: usize, height: usize, scale: Scale) -> i32 {
// TODO: Implement best fit
let factor: i32 = match scale { let factor: i32 = match scale {
Scale::X1 => 1, Scale::X1 => 1,
Scale::X2 => 2, Scale::X2 => 2,