2017-01-21 11:53:03 +02:00
|
|
|
// Copyright 2017 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
|
|
|
#include <memory>
|
2018-09-11 03:29:59 +02:00
|
|
|
#include <thread>
|
2021-09-20 17:43:13 -05:00
|
|
|
#include "common/input.h"
|
2017-01-21 11:53:03 +02:00
|
|
|
#include "common/param_package.h"
|
2021-09-20 17:43:13 -05:00
|
|
|
#include "input_common/drivers/gc_adapter.h"
|
|
|
|
#include "input_common/drivers/keyboard.h"
|
|
|
|
#include "input_common/drivers/mouse.h"
|
|
|
|
#include "input_common/drivers/tas_input.h"
|
|
|
|
#include "input_common/drivers/touch_screen.h"
|
|
|
|
#include "input_common/drivers/udp_client.h"
|
|
|
|
#include "input_common/helpers/stick_from_buttons.h"
|
|
|
|
#include "input_common/helpers/touch_from_buttons.h"
|
|
|
|
#include "input_common/input_engine.h"
|
|
|
|
#include "input_common/input_mapping.h"
|
|
|
|
#include "input_common/input_poller.h"
|
2017-01-21 11:53:03 +02:00
|
|
|
#include "input_common/main.h"
|
2017-01-21 17:33:48 +02:00
|
|
|
#ifdef HAVE_SDL2
|
2021-09-20 17:43:13 -05:00
|
|
|
#include "input_common/drivers/sdl_driver.h"
|
2017-01-21 17:33:48 +02:00
|
|
|
#endif
|
2017-01-21 11:53:03 +02:00
|
|
|
|
|
|
|
namespace InputCommon {
|
|
|
|
|
2020-08-27 15:16:47 -04:00
|
|
|
struct InputSubsystem::Impl {
|
|
|
|
void Initialize() {
|
2021-09-20 17:43:13 -05:00
|
|
|
mapping_factory = std::make_shared<MappingFactory>();
|
|
|
|
MappingCallback mapping_callback{[this](MappingData data) { RegisterInput(data); }};
|
|
|
|
|
|
|
|
keyboard = std::make_shared<Keyboard>("keyboard");
|
|
|
|
keyboard->SetMappingCallback(mapping_callback);
|
|
|
|
keyboard_factory = std::make_shared<InputFactory>(keyboard);
|
|
|
|
Input::RegisterFactory<Input::InputDevice>(keyboard->GetEngineName(), keyboard_factory);
|
|
|
|
|
|
|
|
mouse = std::make_shared<Mouse>("mouse");
|
|
|
|
mouse->SetMappingCallback(mapping_callback);
|
|
|
|
mouse_factory = std::make_shared<InputFactory>(mouse);
|
|
|
|
Input::RegisterFactory<Input::InputDevice>(mouse->GetEngineName(), mouse_factory);
|
|
|
|
|
|
|
|
touch_screen = std::make_shared<TouchScreen>("touch");
|
|
|
|
touch_screen_factory = std::make_shared<InputFactory>(touch_screen);
|
|
|
|
Input::RegisterFactory<Input::InputDevice>(touch_screen->GetEngineName(),
|
|
|
|
touch_screen_factory);
|
|
|
|
|
|
|
|
gcadapter = std::make_shared<GCAdapter>("gcpad");
|
|
|
|
gcadapter->SetMappingCallback(mapping_callback);
|
2021-10-11 00:43:11 -05:00
|
|
|
gcadapter_input_factory = std::make_shared<InputFactory>(gcadapter);
|
|
|
|
gcadapter_output_factory = std::make_shared<OutputFactory>(gcadapter);
|
2021-10-15 19:07:47 -05:00
|
|
|
Input::RegisterFactory<Input::InputDevice>(gcadapter->GetEngineName(),
|
|
|
|
gcadapter_input_factory);
|
|
|
|
Input::RegisterFactory<Input::OutputDevice>(gcadapter->GetEngineName(),
|
|
|
|
gcadapter_output_factory);
|
2021-09-20 17:43:13 -05:00
|
|
|
|
|
|
|
udp_client = std::make_shared<CemuhookUDP::UDPClient>("cemuhookudp");
|
|
|
|
udp_client->SetMappingCallback(mapping_callback);
|
|
|
|
udp_client_factory = std::make_shared<InputFactory>(udp_client);
|
|
|
|
Input::RegisterFactory<Input::InputDevice>(udp_client->GetEngineName(), udp_client_factory);
|
|
|
|
|
|
|
|
tas_input = std::make_shared<TasInput::Tas>("tas");
|
|
|
|
tas_input->SetMappingCallback(mapping_callback);
|
|
|
|
tas_input_factory = std::make_shared<InputFactory>(tas_input);
|
|
|
|
Input::RegisterFactory<Input::InputDevice>(tas_input->GetEngineName(), tas_input_factory);
|
|
|
|
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
sdl = std::make_shared<SDLDriver>("sdl");
|
|
|
|
sdl->SetMappingCallback(mapping_callback);
|
2021-10-11 00:43:11 -05:00
|
|
|
sdl_input_factory = std::make_shared<InputFactory>(sdl);
|
|
|
|
sdl_output_factory = std::make_shared<OutputFactory>(sdl);
|
|
|
|
Input::RegisterFactory<Input::InputDevice>(sdl->GetEngineName(), sdl_input_factory);
|
|
|
|
Input::RegisterFactory<Input::OutputDevice>(sdl->GetEngineName(), sdl_output_factory);
|
2021-09-20 17:43:13 -05:00
|
|
|
#endif
|
|
|
|
|
|
|
|
Input::RegisterFactory<Input::InputDevice>("touch_from_button",
|
|
|
|
std::make_shared<TouchFromButton>());
|
|
|
|
Input::RegisterFactory<Input::InputDevice>("analog_from_button",
|
|
|
|
std::make_shared<StickFromButton>());
|
2020-08-27 15:16:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
void Shutdown() {
|
2021-09-20 17:43:13 -05:00
|
|
|
Input::UnregisterFactory<Input::InputDevice>(keyboard->GetEngineName());
|
|
|
|
keyboard.reset();
|
|
|
|
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>(mouse->GetEngineName());
|
|
|
|
mouse.reset();
|
|
|
|
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>(touch_screen->GetEngineName());
|
|
|
|
touch_screen.reset();
|
|
|
|
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>(gcadapter->GetEngineName());
|
|
|
|
gcadapter.reset();
|
|
|
|
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>(udp_client->GetEngineName());
|
|
|
|
udp_client.reset();
|
|
|
|
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>(tas_input->GetEngineName());
|
|
|
|
tas_input.reset();
|
|
|
|
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>(sdl->GetEngineName());
|
|
|
|
sdl.reset();
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>("touch_from_button");
|
|
|
|
Input::UnregisterFactory<Input::InputDevice>("analog_from_button");
|
2020-08-27 15:16:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] std::vector<Common::ParamPackage> GetInputDevices() const {
|
|
|
|
std::vector<Common::ParamPackage> devices = {
|
2021-09-20 17:43:13 -05:00
|
|
|
Common::ParamPackage{{"display", "Any"}, {"engine", "any"}},
|
2020-08-27 15:16:47 -04:00
|
|
|
};
|
2021-09-20 17:43:13 -05:00
|
|
|
|
|
|
|
auto keyboard_devices = keyboard->GetInputDevices();
|
|
|
|
devices.insert(devices.end(), keyboard_devices.begin(), keyboard_devices.end());
|
|
|
|
auto mouse_devices = mouse->GetInputDevices();
|
|
|
|
devices.insert(devices.end(), mouse_devices.begin(), mouse_devices.end());
|
|
|
|
auto gcadapter_devices = gcadapter->GetInputDevices();
|
|
|
|
devices.insert(devices.end(), gcadapter_devices.begin(), gcadapter_devices.end());
|
|
|
|
auto tas_input_devices = tas_input->GetInputDevices();
|
|
|
|
devices.insert(devices.end(), tas_input_devices.begin(), tas_input_devices.end());
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
auto sdl_devices = sdl->GetInputDevices();
|
|
|
|
devices.insert(devices.end(), sdl_devices.begin(), sdl_devices.end());
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return devices;
|
2020-08-27 15:16:47 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] AnalogMapping GetAnalogMappingForDevice(
|
|
|
|
const Common::ParamPackage& params) const {
|
2021-09-20 17:43:13 -05:00
|
|
|
if (!params.Has("engine") || params.Get("engine", "") == "any") {
|
2020-08-27 15:16:47 -04:00
|
|
|
return {};
|
|
|
|
}
|
2021-09-20 17:43:13 -05:00
|
|
|
const std::string engine = params.Get("engine", "");
|
|
|
|
if (engine == gcadapter->GetEngineName()) {
|
|
|
|
return gcadapter->GetAnalogMappingForDevice(params);
|
|
|
|
}
|
|
|
|
if (engine == tas_input->GetEngineName()) {
|
|
|
|
return tas_input->GetAnalogMappingForDevice(params);
|
|
|
|
}
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
if (engine == sdl->GetEngineName()) {
|
|
|
|
return sdl->GetAnalogMappingForDevice(params);
|
|
|
|
}
|
|
|
|
#endif
|
2020-08-27 15:16:47 -04:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
|
|
|
[[nodiscard]] ButtonMapping GetButtonMappingForDevice(
|
|
|
|
const Common::ParamPackage& params) const {
|
2021-09-20 17:43:13 -05:00
|
|
|
if (!params.Has("engine") || params.Get("engine", "") == "any") {
|
2020-08-27 15:16:47 -04:00
|
|
|
return {};
|
|
|
|
}
|
2021-09-20 17:43:13 -05:00
|
|
|
const std::string engine = params.Get("engine", "");
|
|
|
|
if (engine == gcadapter->GetEngineName()) {
|
|
|
|
return gcadapter->GetButtonMappingForDevice(params);
|
|
|
|
}
|
|
|
|
if (engine == tas_input->GetEngineName()) {
|
|
|
|
return tas_input->GetButtonMappingForDevice(params);
|
|
|
|
}
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
if (engine == sdl->GetEngineName()) {
|
|
|
|
return sdl->GetButtonMappingForDevice(params);
|
|
|
|
}
|
|
|
|
#endif
|
2020-08-27 15:16:47 -04:00
|
|
|
return {};
|
|
|
|
}
|
2018-09-11 03:29:59 +02:00
|
|
|
|
2020-09-04 21:35:42 -05:00
|
|
|
[[nodiscard]] MotionMapping GetMotionMappingForDevice(
|
|
|
|
const Common::ParamPackage& params) const {
|
2021-09-20 17:43:13 -05:00
|
|
|
if (!params.Has("engine") || params.Get("engine", "") == "any") {
|
2020-09-04 21:35:42 -05:00
|
|
|
return {};
|
|
|
|
}
|
2021-09-20 17:43:13 -05:00
|
|
|
const std::string engine = params.Get("engine", "");
|
|
|
|
if (engine == gcadapter->GetEngineName()) {
|
|
|
|
return gcadapter->GetMotionMappingForDevice(params);
|
|
|
|
}
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
if (engine == sdl->GetEngineName()) {
|
|
|
|
return sdl->GetMotionMappingForDevice(params);
|
|
|
|
}
|
|
|
|
#endif
|
2020-09-04 21:35:42 -05:00
|
|
|
return {};
|
|
|
|
}
|
|
|
|
|
2021-09-20 17:43:13 -05:00
|
|
|
std::string GetButtonName(const Common::ParamPackage& params) const {
|
|
|
|
if (!params.Has("engine") || params.Get("engine", "") == "any") {
|
|
|
|
return "Unknown";
|
|
|
|
}
|
|
|
|
const std::string engine = params.Get("engine", "");
|
|
|
|
if (engine == mouse->GetEngineName()) {
|
|
|
|
return mouse->GetUIName(params);
|
|
|
|
}
|
|
|
|
if (engine == gcadapter->GetEngineName()) {
|
|
|
|
return gcadapter->GetUIName(params);
|
|
|
|
}
|
|
|
|
if (engine == udp_client->GetEngineName()) {
|
|
|
|
return udp_client->GetUIName(params);
|
|
|
|
}
|
|
|
|
if (engine == tas_input->GetEngineName()) {
|
|
|
|
return tas_input->GetUIName(params);
|
|
|
|
}
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
if (engine == sdl->GetEngineName()) {
|
|
|
|
return sdl->GetUIName(params);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return "Bad engine";
|
|
|
|
}
|
|
|
|
|
|
|
|
bool IsController(const Common::ParamPackage& params) {
|
|
|
|
const std::string engine = params.Get("engine", "");
|
|
|
|
if (engine == mouse->GetEngineName()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (engine == gcadapter->GetEngineName()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
if (engine == tas_input->GetEngineName()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
if (engine == sdl->GetEngineName()) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void BeginConfiguration() {
|
|
|
|
keyboard->BeginConfiguration();
|
|
|
|
mouse->BeginConfiguration();
|
|
|
|
gcadapter->BeginConfiguration();
|
|
|
|
udp_client->BeginConfiguration();
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
sdl->BeginConfiguration();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void EndConfiguration() {
|
|
|
|
keyboard->EndConfiguration();
|
|
|
|
mouse->EndConfiguration();
|
|
|
|
gcadapter->EndConfiguration();
|
|
|
|
udp_client->EndConfiguration();
|
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
sdl->EndConfiguration();
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void RegisterInput(MappingData data) {
|
|
|
|
mapping_factory->RegisterInput(data);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::shared_ptr<MappingFactory> mapping_factory;
|
2021-10-11 00:43:11 -05:00
|
|
|
|
2021-09-20 17:43:13 -05:00
|
|
|
std::shared_ptr<Keyboard> keyboard;
|
|
|
|
std::shared_ptr<Mouse> mouse;
|
|
|
|
std::shared_ptr<GCAdapter> gcadapter;
|
|
|
|
std::shared_ptr<TouchScreen> touch_screen;
|
2021-10-11 00:43:11 -05:00
|
|
|
std::shared_ptr<TasInput::Tas> tas_input;
|
2021-09-20 17:43:13 -05:00
|
|
|
std::shared_ptr<CemuhookUDP::UDPClient> udp_client;
|
2021-10-11 00:43:11 -05:00
|
|
|
|
|
|
|
std::shared_ptr<InputFactory> keyboard_factory;
|
|
|
|
std::shared_ptr<InputFactory> mouse_factory;
|
|
|
|
std::shared_ptr<InputFactory> gcadapter_input_factory;
|
|
|
|
std::shared_ptr<InputFactory> touch_screen_factory;
|
2021-09-20 17:43:13 -05:00
|
|
|
std::shared_ptr<InputFactory> udp_client_factory;
|
|
|
|
std::shared_ptr<InputFactory> tas_input_factory;
|
2021-10-11 00:43:11 -05:00
|
|
|
|
|
|
|
std::shared_ptr<OutputFactory> gcadapter_output_factory;
|
|
|
|
|
2021-09-20 17:43:13 -05:00
|
|
|
#ifdef HAVE_SDL2
|
|
|
|
std::shared_ptr<SDLDriver> sdl;
|
2021-10-11 00:43:11 -05:00
|
|
|
std::shared_ptr<InputFactory> sdl_input_factory;
|
|
|
|
std::shared_ptr<OutputFactory> sdl_output_factory;
|
2021-09-20 17:43:13 -05:00
|
|
|
#endif
|
2020-08-27 15:16:47 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
InputSubsystem::InputSubsystem() : impl{std::make_unique<Impl>()} {}
|
|
|
|
|
|
|
|
InputSubsystem::~InputSubsystem() = default;
|
|
|
|
|
|
|
|
void InputSubsystem::Initialize() {
|
|
|
|
impl->Initialize();
|
|
|
|
}
|
|
|
|
|
|
|
|
void InputSubsystem::Shutdown() {
|
|
|
|
impl->Shutdown();
|
|
|
|
}
|
|
|
|
|
2021-09-20 17:43:13 -05:00
|
|
|
Keyboard* InputSubsystem::GetKeyboard() {
|
|
|
|
return impl->keyboard.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const Keyboard* InputSubsystem::GetKeyboard() const {
|
|
|
|
return impl->keyboard.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
Mouse* InputSubsystem::GetMouse() {
|
|
|
|
return impl->mouse.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const Mouse* InputSubsystem::GetMouse() const {
|
|
|
|
return impl->mouse.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
TouchScreen* InputSubsystem::GetTouchScreen() {
|
|
|
|
return impl->touch_screen.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const TouchScreen* InputSubsystem::GetTouchScreen() const {
|
|
|
|
return impl->touch_screen.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
TasInput::Tas* InputSubsystem::GetTas() {
|
|
|
|
return impl->tas_input.get();
|
|
|
|
}
|
|
|
|
|
|
|
|
const TasInput::Tas* InputSubsystem::GetTas() const {
|
|
|
|
return impl->tas_input.get();
|
|
|
|
}
|
|
|
|
|
2020-08-27 15:16:47 -04:00
|
|
|
std::vector<Common::ParamPackage> InputSubsystem::GetInputDevices() const {
|
|
|
|
return impl->GetInputDevices();
|
|
|
|
}
|
|
|
|
|
|
|
|
AnalogMapping InputSubsystem::GetAnalogMappingForDevice(const Common::ParamPackage& device) const {
|
|
|
|
return impl->GetAnalogMappingForDevice(device);
|
|
|
|
}
|
2020-06-23 12:47:58 -04:00
|
|
|
|
2020-08-27 15:16:47 -04:00
|
|
|
ButtonMapping InputSubsystem::GetButtonMappingForDevice(const Common::ParamPackage& device) const {
|
|
|
|
return impl->GetButtonMappingForDevice(device);
|
2017-01-21 11:53:03 +02:00
|
|
|
}
|
|
|
|
|
2020-10-14 02:51:14 -04:00
|
|
|
MotionMapping InputSubsystem::GetMotionMappingForDevice(const Common::ParamPackage& device) const {
|
|
|
|
return impl->GetMotionMappingForDevice(device);
|
|
|
|
}
|
|
|
|
|
2021-09-20 17:43:13 -05:00
|
|
|
std::string InputSubsystem::GetButtonName(const Common::ParamPackage& params) const {
|
|
|
|
const std::string toggle = params.Get("toggle", false) ? "~" : "";
|
|
|
|
const std::string inverted = params.Get("inverted", false) ? "!" : "";
|
|
|
|
const std::string button_name = impl->GetButtonName(params);
|
|
|
|
std::string axis_direction = "";
|
|
|
|
if (params.Has("axis")) {
|
|
|
|
axis_direction = params.Get("invert", "+");
|
|
|
|
}
|
|
|
|
return fmt::format("{}{}{}{}", toggle, inverted, button_name, axis_direction);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool InputSubsystem::IsController(const Common::ParamPackage& params) const {
|
|
|
|
return impl->IsController(params);
|
|
|
|
}
|
|
|
|
|
2020-08-29 20:56:51 +02:00
|
|
|
void InputSubsystem::ReloadInputDevices() {
|
2021-09-20 17:43:13 -05:00
|
|
|
impl->udp_client.get()->ReloadSockets();
|
|
|
|
}
|
|
|
|
|
|
|
|
void InputSubsystem::BeginMapping(Polling::InputType type) {
|
|
|
|
impl->BeginConfiguration();
|
|
|
|
impl->mapping_factory->BeginMapping(type);
|
|
|
|
}
|
|
|
|
|
|
|
|
const Common::ParamPackage InputSubsystem::GetNextInput() const {
|
|
|
|
return impl->mapping_factory->GetNextInput();
|
2020-07-14 19:01:36 +02:00
|
|
|
}
|
|
|
|
|
2021-09-20 17:43:13 -05:00
|
|
|
void InputSubsystem::StopMapping() const {
|
|
|
|
impl->EndConfiguration();
|
|
|
|
impl->mapping_factory->StopMapping();
|
2020-06-21 12:36:28 -04:00
|
|
|
}
|
|
|
|
|
2017-01-21 11:53:03 +02:00
|
|
|
std::string GenerateKeyboardParam(int key_code) {
|
2021-07-22 19:59:26 -05:00
|
|
|
Common::ParamPackage param;
|
|
|
|
param.Set("engine", "keyboard");
|
|
|
|
param.Set("code", key_code);
|
|
|
|
param.Set("toggle", false);
|
2017-01-21 11:53:03 +02:00
|
|
|
return param.Serialize();
|
|
|
|
}
|
|
|
|
|
2017-01-21 13:04:00 +02:00
|
|
|
std::string GenerateAnalogParamFromKeys(int key_up, int key_down, int key_left, int key_right,
|
|
|
|
int key_modifier, float modifier_scale) {
|
|
|
|
Common::ParamPackage circle_pad_param{
|
|
|
|
{"engine", "analog_from_button"},
|
|
|
|
{"up", GenerateKeyboardParam(key_up)},
|
|
|
|
{"down", GenerateKeyboardParam(key_down)},
|
|
|
|
{"left", GenerateKeyboardParam(key_left)},
|
|
|
|
{"right", GenerateKeyboardParam(key_right)},
|
|
|
|
{"modifier", GenerateKeyboardParam(key_modifier)},
|
|
|
|
{"modifier_scale", std::to_string(modifier_scale)},
|
|
|
|
};
|
|
|
|
return circle_pad_param.Serialize();
|
|
|
|
}
|
2017-01-21 11:53:03 +02:00
|
|
|
} // namespace InputCommon
|