Fix up winkey GetKeystroke to not use GetAsyncKeyState, and disable GetState if Xenia is not the active window.

This commit is contained in:
Dr. Chat 2015-08-18 18:50:33 -05:00
parent 38064acd51
commit 29ed27bfc5
3 changed files with 161 additions and 96 deletions

View File

@ -13,5 +13,6 @@ project("xenia-hid-winkey")
defines({ defines({
}) })
includedirs({ includedirs({
project_root.."/third_party/elemental-forms/src",
}) })
local_platform_files() local_platform_files()

View File

@ -10,14 +10,41 @@
#include "xenia/hid/winkey/winkey_input_driver.h" #include "xenia/hid/winkey/winkey_input_driver.h"
#include "xenia/base/platform_win.h" #include "xenia/base/platform_win.h"
#include "xenia/emulator.h"
#include "xenia/hid/hid_flags.h" #include "xenia/hid/hid_flags.h"
#include "xenia/hid/input_system.h"
#include "xenia/ui/window.h"
namespace xe { namespace xe {
namespace hid { namespace hid {
namespace winkey { namespace winkey {
WinKeyInputDriver::WinKeyInputDriver(InputSystem* input_system) WinKeyInputDriver::WinKeyInputDriver(InputSystem* input_system)
: InputDriver(input_system), packet_number_(1) {} : InputDriver(input_system), packet_number_(1) {
// Register a key listener.
input_system_->emulator()->display_window()->on_key_down.AddListener(
[this](ui::KeyEvent* evt) {
std::lock_guard<std::mutex> lock(key_event_mutex_);
KeyEvent key;
key.vkey = evt->key_code();
key.transition = true;
key.prev_state = evt->prev_state();
key.repeat_count = evt->repeat_count();
key_events_.push(key);
});
input_system_->emulator()->display_window()->on_key_up.AddListener(
[this](ui::KeyEvent* evt) {
std::lock_guard<std::mutex> lock(key_event_mutex_);
KeyEvent key;
key.vkey = evt->key_code();
key.transition = false;
key.prev_state = evt->prev_state();
key.repeat_count = evt->repeat_count();
key_events_.push(key);
});
}
WinKeyInputDriver::~WinKeyInputDriver() = default; WinKeyInputDriver::~WinKeyInputDriver() = default;
@ -64,6 +91,7 @@ X_RESULT WinKeyInputDriver::GetState(uint32_t user_index,
int16_t thumb_rx = 0; int16_t thumb_rx = 0;
int16_t thumb_ry = 0; int16_t thumb_ry = 0;
if (input_system_->emulator()->display_window()->has_focus()) {
if (IS_KEY_TOGGLED(VK_CAPITAL)) { if (IS_KEY_TOGGLED(VK_CAPITAL)) {
// dpad toggled // dpad toggled
if (IS_KEY_DOWN(0x41)) { if (IS_KEY_DOWN(0x41)) {
@ -145,6 +173,7 @@ X_RESULT WinKeyInputDriver::GetState(uint32_t user_index,
// X // X
buttons |= 0x0010; // XINPUT_GAMEPAD_START buttons |= 0x0010; // XINPUT_GAMEPAD_START
} }
}
out_state->packet_number = packet_number_; out_state->packet_number = packet_number_;
out_state->gamepad.buttons = buttons; out_state->gamepad.buttons = buttons;
@ -180,84 +209,107 @@ X_RESULT WinKeyInputDriver::GetKeystroke(uint32_t user_index, uint32_t flags,
uint16_t keystroke_flags = 0; uint16_t keystroke_flags = 0;
uint8_t hid_code = 0; uint8_t hid_code = 0;
// Pop from the queue.
key_event_mutex_.lock();
if (key_events_.size() == 0) {
key_event_mutex_.unlock();
// No keys!
return X_ERROR_EMPTY;
}
KeyEvent evt = key_events_.front();
key_events_.pop();
key_event_mutex_.unlock();
// TODO: Some other way to toggle this...
if (IS_KEY_TOGGLED(VK_CAPITAL)) { if (IS_KEY_TOGGLED(VK_CAPITAL)) {
// dpad toggled // dpad toggled
if (IS_KEY_DOWN(0x41)) { if (evt.vkey == (0x41)) {
// A // A
virtual_key = 0x5812; // VK_PAD_DPAD_LEFT virtual_key = 0x5812; // VK_PAD_DPAD_LEFT
} else if (IS_KEY_DOWN(0x44)) { } else if (evt.vkey == (0x44)) {
// D // D
virtual_key = 0x5813; // VK_PAD_DPAD_RIGHT virtual_key = 0x5813; // VK_PAD_DPAD_RIGHT
} else if (IS_KEY_DOWN(0x53)) { } else if (evt.vkey == (0x53)) {
// S // S
virtual_key = 0x5811; // VK_PAD_DPAD_DOWN virtual_key = 0x5811; // VK_PAD_DPAD_DOWN
} else if (IS_KEY_DOWN(0x57)) { } else if (evt.vkey == (0x57)) {
// W // W
virtual_key = 0x5810; // VK_PAD_DPAD_UP virtual_key = 0x5810; // VK_PAD_DPAD_UP
} }
} else { } else {
// left stick // left stick
if (IS_KEY_DOWN(0x57)) { if (evt.vkey == (0x57)) {
// W // W
virtual_key = 0x5820; // VK_PAD_LTHUMB_UP virtual_key = 0x5820; // VK_PAD_LTHUMB_UP
} }
if (IS_KEY_DOWN(0x53)) { if (evt.vkey == (0x53)) {
// S // S
virtual_key = 0x5821; // VK_PAD_LTHUMB_DOWN virtual_key = 0x5821; // VK_PAD_LTHUMB_DOWN
} }
if (IS_KEY_DOWN(0x44)) { if (evt.vkey == (0x44)) {
// D // D
virtual_key = 0x5822; // VK_PAD_LTHUMB_RIGHT virtual_key = 0x5822; // VK_PAD_LTHUMB_RIGHT
} }
if (IS_KEY_DOWN(0x41)) { if (evt.vkey == (0x41)) {
// A // A
virtual_key = 0x5823; // VK_PAD_LTHUMB_LEFT virtual_key = 0x5823; // VK_PAD_LTHUMB_LEFT
} }
} }
// Right stick // Right stick
if (IS_KEY_DOWN(0x26)) { if (evt.vkey == (0x26)) {
// Up // Up
virtual_key = 0x5830; virtual_key = 0x5830;
} }
if (IS_KEY_DOWN(0x28)) { if (evt.vkey == (0x28)) {
// Down // Down
virtual_key = 0x5831; virtual_key = 0x5831;
} }
if (IS_KEY_DOWN(0x27)) { if (evt.vkey == (0x27)) {
// Right // Right
virtual_key = 0x5832; virtual_key = 0x5832;
} }
if (IS_KEY_DOWN(0x25)) { if (evt.vkey == (0x25)) {
// Left // Left
virtual_key = 0x5833; virtual_key = 0x5833;
} }
if (IS_KEY_DOWN(0x4C)) { if (evt.vkey == (0x4C)) {
// L // L
virtual_key = 0x5802; // VK_PAD_X virtual_key = 0x5802; // VK_PAD_X
} else if (IS_KEY_DOWN(VK_OEM_7)) { } else if (evt.vkey == (VK_OEM_7)) {
// ' // '
virtual_key = 0x5801; // VK_PAD_B virtual_key = 0x5801; // VK_PAD_B
} else if (IS_KEY_DOWN(VK_OEM_1)) { } else if (evt.vkey == (VK_OEM_1)) {
// ; // ;
virtual_key = 0x5800; // VK_PAD_A virtual_key = 0x5800; // VK_PAD_A
} else if (IS_KEY_DOWN(0x50)) { } else if (evt.vkey == (0x50)) {
// P // P
virtual_key = 0x5803; // VK_PAD_Y virtual_key = 0x5803; // VK_PAD_Y
} }
if (IS_KEY_DOWN(0x58)) { if (evt.vkey == (0x58)) {
// X // X
virtual_key = 0x5814; // VK_PAD_START virtual_key = 0x5814; // VK_PAD_START
} }
if (IS_KEY_DOWN(0x5A)) { if (evt.vkey == (0x5A)) {
// Z // Z
virtual_key = 0x5815; // VK_PAD_BACK virtual_key = 0x5815; // VK_PAD_BACK
} }
if (virtual_key != 0) { if (virtual_key != 0) {
keystroke_flags |= 0x0001; // XINPUT_KEYSTROKE_DOWN if (evt.transition == true) {
keystroke_flags |= 0x0001; // XINPUT_KEYSTROKE_KEYDOWN
} else if (evt.transition == false) {
keystroke_flags |= 0x0002; // XINPUT_KEYSTROKE_KEYUP
}
if (evt.prev_state == evt.transition) {
keystroke_flags |= 0x0004; // XINPUT_KEYSTROKE_REPEAT
}
result = X_ERROR_SUCCESS; result = X_ERROR_SUCCESS;
} }

View File

@ -10,6 +10,9 @@
#ifndef XENIA_HID_WINKEY_WINKEY_INPUT_DRIVER_H_ #ifndef XENIA_HID_WINKEY_WINKEY_INPUT_DRIVER_H_
#define XENIA_HID_WINKEY_WINKEY_INPUT_DRIVER_H_ #define XENIA_HID_WINKEY_WINKEY_INPUT_DRIVER_H_
#include <mutex>
#include <queue>
#include "xenia/hid/input_driver.h" #include "xenia/hid/input_driver.h"
namespace xe { namespace xe {
@ -31,6 +34,15 @@ class WinKeyInputDriver : public InputDriver {
X_INPUT_KEYSTROKE* out_keystroke) override; X_INPUT_KEYSTROKE* out_keystroke) override;
protected: protected:
struct KeyEvent {
int vkey = 0;
int repeat_count = 0;
bool transition = false; // going up(false) or going down(true)
bool prev_state = false; // down(true) or up(false)
};
std::queue<KeyEvent> key_events_;
std::mutex key_event_mutex_;
uint32_t packet_number_; uint32_t packet_number_;
}; };