yuzu/src/core/hid/emulated_console.cpp

251 lines
7.8 KiB
C++
Raw Normal View History

// SPDX-FileCopyrightText: Copyright 2021 yuzu Emulator Project
// SPDX-License-Identifier: GPL-2.0-or-later
2021-09-20 20:43:16 -04:00
#include "common/settings.h"
2021-09-20 20:43:16 -04:00
#include "core/hid/emulated_console.h"
#include "core/hid/input_converter.h"
namespace Core::HID {
2021-10-23 00:04:06 -04:00
EmulatedConsole::EmulatedConsole() = default;
2021-09-20 20:43:16 -04:00
EmulatedConsole::~EmulatedConsole() = default;
void EmulatedConsole::ReloadFromSettings() {
2021-11-04 00:35:45 -04:00
// Using first motion device from player 1. No need to assign any unique config at the moment
2021-09-20 20:43:16 -04:00
const auto& player = Settings::values.players.GetValue()[0];
motion_params = Common::ParamPackage(player.motions[0]);
ReloadInput();
}
2021-10-21 00:18:04 -04:00
void EmulatedConsole::SetTouchParams() {
// TODO(german77): Support any number of fingers
2021-09-20 20:43:16 -04:00
std::size_t index = 0;
2021-10-21 00:18:04 -04:00
// Hardcode mouse, touchscreen and cemuhook parameters
2021-11-14 15:09:29 -05:00
if (!Settings::values.mouse_enabled) {
// We can't use mouse as touch if native mouse is enabled
touch_params[index++] = Common::ParamPackage{"engine:mouse,axis_x:10,axis_y:11,button:0"};
}
touch_params[index++] =
Common::ParamPackage{"engine:touch,axis_x:0,axis_y:1,button:0,touch_id:0"};
touch_params[index++] =
Common::ParamPackage{"engine:touch,axis_x:2,axis_y:3,button:1,touch_id:1"};
touch_params[index++] =
Common::ParamPackage{"engine:touch,axis_x:4,axis_y:5,button:2,touch_id:2"};
touch_params[index++] =
Common::ParamPackage{"engine:touch,axis_x:6,axis_y:7,button:3,touch_id:3"};
touch_params[index++] =
Common::ParamPackage{"engine:cemuhookudp,axis_x:17,axis_y:18,button:65536,touch_id:0"};
touch_params[index++] =
Common::ParamPackage{"engine:cemuhookudp,axis_x:19,axis_y:20,button:131072,touch_id:1"};
2021-10-21 00:18:04 -04:00
2021-09-20 20:43:16 -04:00
const auto button_index =
static_cast<u64>(Settings::values.touch_from_button_map_index.GetValue());
const auto& touch_buttons = Settings::values.touch_from_button_maps[button_index].buttons;
2021-10-21 00:18:04 -04:00
2021-11-04 00:35:45 -04:00
// Map the rest of the fingers from touch from button configuration
2021-09-20 20:43:16 -04:00
for (const auto& config_entry : touch_buttons) {
2021-11-14 15:09:29 -05:00
if (index >= touch_params.size()) {
continue;
}
2021-09-20 20:43:16 -04:00
Common::ParamPackage params{config_entry};
Common::ParamPackage touch_button_params;
const int x = params.Get("x", 0);
const int y = params.Get("y", 0);
params.Erase("x");
params.Erase("y");
touch_button_params.Set("engine", "touch_from_button");
touch_button_params.Set("button", params.Serialize());
touch_button_params.Set("x", x);
touch_button_params.Set("y", y);
touch_button_params.Set("touch_id", static_cast<int>(index));
2021-10-21 00:18:04 -04:00
touch_params[index] = touch_button_params;
index++;
}
}
2021-09-20 20:43:16 -04:00
2021-10-21 00:18:04 -04:00
void EmulatedConsole::ReloadInput() {
2021-11-04 00:35:45 -04:00
// If you load any device here add the equivalent to the UnloadInput() function
2021-10-21 00:18:04 -04:00
SetTouchParams();
2021-11-04 00:35:45 -04:00
motion_devices = Common::Input::CreateDevice<Common::Input::InputDevice>(motion_params);
2021-10-21 00:18:04 -04:00
if (motion_devices) {
motion_devices->SetCallback({
.on_change =
[this](const Common::Input::CallbackStatus& callback) { SetMotion(callback); },
});
2021-10-21 00:18:04 -04:00
}
2021-11-04 00:35:45 -04:00
// Unique index for identifying touch device source
2021-10-21 00:18:04 -04:00
std::size_t index = 0;
for (auto& touch_device : touch_devices) {
touch_device = Common::Input::CreateDevice<Common::Input::InputDevice>(touch_params[index]);
2021-10-21 00:18:04 -04:00
if (!touch_device) {
continue;
}
touch_device->SetCallback({
.on_change =
[this, index](const Common::Input::CallbackStatus& callback) {
SetTouch(callback, index);
},
});
2021-09-20 20:43:16 -04:00
index++;
}
}
void EmulatedConsole::UnloadInput() {
motion_devices.reset();
for (auto& touch : touch_devices) {
touch.reset();
}
}
void EmulatedConsole::EnableConfiguration() {
is_configuring = true;
SaveCurrentConfig();
}
void EmulatedConsole::DisableConfiguration() {
is_configuring = false;
}
bool EmulatedConsole::IsConfiguring() const {
return is_configuring;
}
void EmulatedConsole::SaveCurrentConfig() {
if (!is_configuring) {
return;
}
}
void EmulatedConsole::RestoreConfig() {
if (!is_configuring) {
return;
}
ReloadFromSettings();
}
Common::ParamPackage EmulatedConsole::GetMotionParam() const {
return motion_params;
}
void EmulatedConsole::SetMotionParam(Common::ParamPackage param) {
motion_params = param;
ReloadInput();
}
void EmulatedConsole::SetMotion(const Common::Input::CallbackStatus& callback) {
std::unique_lock lock{mutex};
2021-09-20 20:43:16 -04:00
auto& raw_status = console.motion_values.raw_status;
auto& emulated = console.motion_values.emulated;
raw_status = TransformToMotion(callback);
emulated.SetAcceleration(Common::Vec3f{
raw_status.accel.x.value,
raw_status.accel.y.value,
raw_status.accel.z.value,
});
emulated.SetGyroscope(Common::Vec3f{
raw_status.gyro.x.value,
raw_status.gyro.y.value,
raw_status.gyro.z.value,
});
emulated.UpdateRotation(raw_status.delta_timestamp);
emulated.UpdateOrientation(raw_status.delta_timestamp);
if (is_configuring) {
lock.unlock();
2021-09-20 20:43:16 -04:00
TriggerOnChange(ConsoleTriggerType::Motion);
return;
}
auto& motion = console.motion_state;
motion.accel = emulated.GetAcceleration();
motion.gyro = emulated.GetGyroscope();
motion.rotation = emulated.GetRotations();
2021-09-20 20:43:16 -04:00
motion.orientation = emulated.GetOrientation();
motion.quaternion = emulated.GetQuaternion();
motion.gyro_bias = emulated.GetGyroBias();
motion.is_at_rest = !emulated.IsMoving(motion_sensitivity);
// Find what is this value
motion.verticalization_error = 0.0f;
2021-09-20 20:43:16 -04:00
lock.unlock();
2021-09-20 20:43:16 -04:00
TriggerOnChange(ConsoleTriggerType::Motion);
}
void EmulatedConsole::SetTouch(const Common::Input::CallbackStatus& callback, std::size_t index) {
2021-09-20 20:43:16 -04:00
if (index >= console.touch_values.size()) {
return;
}
std::unique_lock lock{mutex};
2021-09-20 20:43:16 -04:00
console.touch_values[index] = TransformToTouch(callback);
if (is_configuring) {
lock.unlock();
2021-09-20 20:43:16 -04:00
TriggerOnChange(ConsoleTriggerType::Touch);
return;
}
// TODO(german77): Remap touch id in sequential order
2021-09-20 20:43:16 -04:00
console.touch_state[index] = {
.position = {console.touch_values[index].x.value, console.touch_values[index].y.value},
.id = static_cast<u32>(console.touch_values[index].id),
2021-09-20 20:43:16 -04:00
.pressed = console.touch_values[index].pressed.value,
};
lock.unlock();
2021-09-20 20:43:16 -04:00
TriggerOnChange(ConsoleTriggerType::Touch);
}
ConsoleMotionValues EmulatedConsole::GetMotionValues() const {
std::scoped_lock lock{mutex};
2021-09-20 20:43:16 -04:00
return console.motion_values;
}
TouchValues EmulatedConsole::GetTouchValues() const {
std::scoped_lock lock{mutex};
2021-09-20 20:43:16 -04:00
return console.touch_values;
}
ConsoleMotion EmulatedConsole::GetMotion() const {
std::scoped_lock lock{mutex};
2021-09-20 20:43:16 -04:00
return console.motion_state;
}
TouchFingerState EmulatedConsole::GetTouch() const {
std::scoped_lock lock{mutex};
2021-09-20 20:43:16 -04:00
return console.touch_state;
}
void EmulatedConsole::TriggerOnChange(ConsoleTriggerType type) {
std::scoped_lock lock{callback_mutex};
2021-10-23 00:04:06 -04:00
for (const auto& poller_pair : callback_list) {
2021-09-20 20:43:16 -04:00
const ConsoleUpdateCallback& poller = poller_pair.second;
if (poller.on_change) {
poller.on_change(type);
}
}
}
int EmulatedConsole::SetCallback(ConsoleUpdateCallback update_callback) {
std::scoped_lock lock{callback_mutex};
2021-09-20 20:43:16 -04:00
callback_list.insert_or_assign(last_callback_key, update_callback);
return last_callback_key++;
}
void EmulatedConsole::DeleteCallback(int key) {
std::scoped_lock lock{callback_mutex};
2021-11-01 16:17:53 -04:00
const auto& iterator = callback_list.find(key);
if (iterator == callback_list.end()) {
2021-09-20 20:43:16 -04:00
LOG_ERROR(Input, "Tried to delete non-existent callback {}", key);
return;
}
2021-11-01 16:17:53 -04:00
callback_list.erase(iterator);
2021-09-20 20:43:16 -04:00
}
} // namespace Core::HID