mirror of
https://git.suyu.dev/suyu/suyu.git
synced 2024-11-18 12:52:47 +01:00
525 lines
13 KiB
C++
525 lines
13 KiB
C++
// SPDX-FileCopyrightText: 2017 Citra Emulator Project
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
#pragma once
|
|
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <string>
|
|
#include <unordered_map>
|
|
#include <utility>
|
|
#include <vector>
|
|
#include "common/logging/log.h"
|
|
#include "common/param_package.h"
|
|
#include "common/uuid.h"
|
|
|
|
namespace Common::Input {
|
|
|
|
// Type of data that is expected to receive or send
|
|
enum class InputType {
|
|
None,
|
|
Battery,
|
|
Button,
|
|
Stick,
|
|
Analog,
|
|
Trigger,
|
|
Motion,
|
|
Touch,
|
|
Color,
|
|
Vibration,
|
|
Nfc,
|
|
IrSensor,
|
|
};
|
|
|
|
// Internal battery charge level
|
|
enum class BatteryLevel : u32 {
|
|
None,
|
|
Empty,
|
|
Critical,
|
|
Low,
|
|
Medium,
|
|
Full,
|
|
Charging,
|
|
};
|
|
|
|
enum class PollingMode {
|
|
// Constant polling of buttons, analogs and motion data
|
|
Active,
|
|
// Only update on button change, digital analogs
|
|
Passive,
|
|
// Enable near field communication polling
|
|
NFC,
|
|
// Enable infrared camera polling
|
|
IR,
|
|
// Enable ring controller polling
|
|
Ring,
|
|
};
|
|
|
|
enum class CameraFormat {
|
|
Size320x240,
|
|
Size160x120,
|
|
Size80x60,
|
|
Size40x30,
|
|
Size20x15,
|
|
None,
|
|
};
|
|
|
|
// Different results that can happen from a device request
|
|
enum class DriverResult {
|
|
Success,
|
|
WrongReply,
|
|
Timeout,
|
|
UnsupportedControllerType,
|
|
HandleInUse,
|
|
ErrorReadingData,
|
|
ErrorWritingData,
|
|
NoDeviceDetected,
|
|
InvalidHandle,
|
|
InvalidParameters,
|
|
NotSupported,
|
|
Disabled,
|
|
Delayed,
|
|
Unknown,
|
|
};
|
|
|
|
// Nfc reply from the controller
|
|
enum class NfcState {
|
|
Success,
|
|
NewAmiibo,
|
|
WaitingForAmiibo,
|
|
AmiiboRemoved,
|
|
InvalidTagType,
|
|
NotSupported,
|
|
WrongDeviceState,
|
|
WriteFailed,
|
|
Unknown,
|
|
};
|
|
|
|
// Hint for amplification curve to be used
|
|
enum class VibrationAmplificationType {
|
|
Linear,
|
|
Exponential,
|
|
};
|
|
|
|
// Analog properties for calibration
|
|
struct AnalogProperties {
|
|
// Anything below this value will be detected as zero
|
|
float deadzone{};
|
|
// Anything above this values will be detected as one
|
|
float range{1.0f};
|
|
// Minimum value to be detected as active
|
|
float threshold{0.5f};
|
|
// Drift correction applied to the raw data
|
|
float offset{};
|
|
// Invert direction of the sensor data
|
|
bool inverted{};
|
|
// Invert the state if it's converted to a button
|
|
bool inverted_button{};
|
|
// Press once to activate, press again to release
|
|
bool toggle{};
|
|
};
|
|
|
|
// Single analog sensor data
|
|
struct AnalogStatus {
|
|
float value{};
|
|
float raw_value{};
|
|
AnalogProperties properties{};
|
|
};
|
|
|
|
// Button data
|
|
struct ButtonStatus {
|
|
Common::UUID uuid{};
|
|
bool value{};
|
|
// Invert value of the button
|
|
bool inverted{};
|
|
// Press once to activate, press again to release
|
|
bool toggle{};
|
|
// Spams the button when active
|
|
bool turbo{};
|
|
// Internal lock for the toggle status
|
|
bool locked{};
|
|
};
|
|
|
|
// Internal battery data
|
|
using BatteryStatus = BatteryLevel;
|
|
|
|
// Analog and digital joystick data
|
|
struct StickStatus {
|
|
Common::UUID uuid{};
|
|
AnalogStatus x{};
|
|
AnalogStatus y{};
|
|
bool left{};
|
|
bool right{};
|
|
bool up{};
|
|
bool down{};
|
|
};
|
|
|
|
// Analog and digital trigger data
|
|
struct TriggerStatus {
|
|
Common::UUID uuid{};
|
|
AnalogStatus analog{};
|
|
ButtonStatus pressed{};
|
|
};
|
|
|
|
// 3D vector representing motion input
|
|
struct MotionSensor {
|
|
AnalogStatus x{};
|
|
AnalogStatus y{};
|
|
AnalogStatus z{};
|
|
};
|
|
|
|
// Motion data used to calculate controller orientation
|
|
struct MotionStatus {
|
|
// Gyroscope vector measurement in radians/s.
|
|
MotionSensor gyro{};
|
|
// Acceleration vector measurement in G force
|
|
MotionSensor accel{};
|
|
// Time since last measurement in microseconds
|
|
u64 delta_timestamp{};
|
|
// Request to update after reading the value
|
|
bool force_update{};
|
|
};
|
|
|
|
// Data of a single point on a touch screen
|
|
struct TouchStatus {
|
|
ButtonStatus pressed{};
|
|
AnalogStatus x{};
|
|
AnalogStatus y{};
|
|
int id{};
|
|
};
|
|
|
|
// Physical controller color in RGB format
|
|
struct BodyColorStatus {
|
|
u32 body{};
|
|
u32 buttons{};
|
|
u32 left_grip{};
|
|
u32 right_grip{};
|
|
};
|
|
|
|
// HD rumble data
|
|
struct VibrationStatus {
|
|
f32 low_amplitude{};
|
|
f32 low_frequency{};
|
|
f32 high_amplitude{};
|
|
f32 high_frequency{};
|
|
VibrationAmplificationType type;
|
|
};
|
|
|
|
// Physical controller LED pattern
|
|
struct LedStatus {
|
|
bool led_1{};
|
|
bool led_2{};
|
|
bool led_3{};
|
|
bool led_4{};
|
|
};
|
|
|
|
// Raw data from camera
|
|
struct CameraStatus {
|
|
CameraFormat format{CameraFormat::None};
|
|
std::vector<u8> data{};
|
|
};
|
|
|
|
struct NfcStatus {
|
|
NfcState state{NfcState::Unknown};
|
|
u8 uuid_length;
|
|
u8 protocol;
|
|
u8 tag_type;
|
|
std::array<u8, 10> uuid;
|
|
};
|
|
|
|
struct MifareData {
|
|
u8 command;
|
|
u8 sector;
|
|
std::array<u8, 0x6> key;
|
|
std::array<u8, 0x10> data;
|
|
};
|
|
|
|
struct MifareRequest {
|
|
std::array<MifareData, 0x10> data;
|
|
};
|
|
|
|
// List of buttons to be passed to Qt that can be translated
|
|
enum class ButtonNames {
|
|
Undefined,
|
|
Invalid,
|
|
// This will display the engine name instead of the button name
|
|
Engine,
|
|
// This will display the button by value instead of the button name
|
|
Value,
|
|
|
|
// Joycon button names
|
|
ButtonLeft,
|
|
ButtonRight,
|
|
ButtonDown,
|
|
ButtonUp,
|
|
ButtonA,
|
|
ButtonB,
|
|
ButtonX,
|
|
ButtonY,
|
|
ButtonPlus,
|
|
ButtonMinus,
|
|
ButtonHome,
|
|
ButtonCapture,
|
|
ButtonStickL,
|
|
ButtonStickR,
|
|
TriggerL,
|
|
TriggerZL,
|
|
TriggerSL,
|
|
TriggerR,
|
|
TriggerZR,
|
|
TriggerSR,
|
|
|
|
// GC button names
|
|
TriggerZ,
|
|
ButtonStart,
|
|
|
|
// DS4 button names
|
|
L1,
|
|
L2,
|
|
L3,
|
|
R1,
|
|
R2,
|
|
R3,
|
|
Circle,
|
|
Cross,
|
|
Square,
|
|
Triangle,
|
|
Share,
|
|
Options,
|
|
Home,
|
|
Touch,
|
|
|
|
// Mouse buttons
|
|
ButtonMouseWheel,
|
|
ButtonBackward,
|
|
ButtonForward,
|
|
ButtonTask,
|
|
ButtonExtra,
|
|
};
|
|
|
|
// Callback data consisting of an input type and the equivalent data status
|
|
struct CallbackStatus {
|
|
InputType type{InputType::None};
|
|
ButtonStatus button_status{};
|
|
StickStatus stick_status{};
|
|
AnalogStatus analog_status{};
|
|
TriggerStatus trigger_status{};
|
|
MotionStatus motion_status{};
|
|
TouchStatus touch_status{};
|
|
BodyColorStatus color_status{};
|
|
BatteryStatus battery_status{};
|
|
VibrationStatus vibration_status{};
|
|
CameraFormat camera_status{CameraFormat::None};
|
|
NfcStatus nfc_status{};
|
|
std::vector<u8> raw_data{};
|
|
};
|
|
|
|
// Triggered once every input change
|
|
struct InputCallback {
|
|
std::function<void(const CallbackStatus&)> on_change;
|
|
};
|
|
|
|
/// An abstract class template for an input device (a button, an analog input, etc.).
|
|
class InputDevice {
|
|
public:
|
|
virtual ~InputDevice() = default;
|
|
|
|
// Force input device to update data regardless of the current state
|
|
virtual void ForceUpdate() {}
|
|
|
|
// Sets the function to be triggered when input changes
|
|
void SetCallback(InputCallback callback_) {
|
|
callback = std::move(callback_);
|
|
}
|
|
|
|
// Triggers the function set in the callback
|
|
void TriggerOnChange(const CallbackStatus& status) {
|
|
if (callback.on_change) {
|
|
callback.on_change(status);
|
|
}
|
|
}
|
|
|
|
private:
|
|
InputCallback callback;
|
|
};
|
|
|
|
/// An abstract class template for an output device (rumble, LED pattern, polling mode).
|
|
class OutputDevice {
|
|
public:
|
|
virtual ~OutputDevice() = default;
|
|
|
|
virtual DriverResult SetLED([[maybe_unused]] const LedStatus& led_status) {
|
|
return DriverResult::NotSupported;
|
|
}
|
|
|
|
virtual DriverResult SetVibration([[maybe_unused]] const VibrationStatus& vibration_status) {
|
|
return DriverResult::NotSupported;
|
|
}
|
|
|
|
virtual bool IsVibrationEnabled() {
|
|
return false;
|
|
}
|
|
|
|
virtual DriverResult SetPollingMode([[maybe_unused]] PollingMode polling_mode) {
|
|
return DriverResult::NotSupported;
|
|
}
|
|
|
|
virtual DriverResult SetCameraFormat([[maybe_unused]] CameraFormat camera_format) {
|
|
return DriverResult::NotSupported;
|
|
}
|
|
|
|
virtual NfcState SupportsNfc() const {
|
|
return NfcState::NotSupported;
|
|
}
|
|
|
|
virtual NfcState StartNfcPolling() {
|
|
return NfcState::NotSupported;
|
|
}
|
|
|
|
virtual NfcState StopNfcPolling() {
|
|
return NfcState::NotSupported;
|
|
}
|
|
|
|
virtual NfcState ReadAmiiboData([[maybe_unused]] std::vector<u8>& out_data) {
|
|
return NfcState::NotSupported;
|
|
}
|
|
|
|
virtual NfcState WriteNfcData([[maybe_unused]] const std::vector<u8>& data) {
|
|
return NfcState::NotSupported;
|
|
}
|
|
|
|
virtual NfcState ReadMifareData([[maybe_unused]] const MifareRequest& request,
|
|
[[maybe_unused]] MifareRequest& out_data) {
|
|
return NfcState::NotSupported;
|
|
}
|
|
|
|
virtual NfcState WriteMifareData([[maybe_unused]] const MifareRequest& request) {
|
|
return NfcState::NotSupported;
|
|
}
|
|
};
|
|
|
|
/// An abstract class template for a factory that can create input devices.
|
|
template <typename InputDeviceType>
|
|
class Factory {
|
|
public:
|
|
virtual ~Factory() = default;
|
|
virtual std::unique_ptr<InputDeviceType> Create(const Common::ParamPackage&) = 0;
|
|
};
|
|
|
|
namespace Impl {
|
|
|
|
template <typename InputDeviceType>
|
|
using FactoryListType = std::unordered_map<std::string, std::shared_ptr<Factory<InputDeviceType>>>;
|
|
|
|
template <typename InputDeviceType>
|
|
struct FactoryList {
|
|
static FactoryListType<InputDeviceType> list;
|
|
};
|
|
|
|
template <typename InputDeviceType>
|
|
FactoryListType<InputDeviceType> FactoryList<InputDeviceType>::list;
|
|
|
|
} // namespace Impl
|
|
|
|
/**
|
|
* Registers an input device factory.
|
|
* @tparam InputDeviceType the type of input devices the factory can create
|
|
* @param name the name of the factory. Will be used to match the "engine" parameter when creating
|
|
* a device
|
|
* @param factory the factory object to register
|
|
*/
|
|
template <typename InputDeviceType>
|
|
void RegisterFactory(const std::string& name, std::shared_ptr<Factory<InputDeviceType>> factory) {
|
|
auto pair = std::make_pair(name, std::move(factory));
|
|
if (!Impl::FactoryList<InputDeviceType>::list.insert(std::move(pair)).second) {
|
|
LOG_ERROR(Input, "Factory '{}' already registered", name);
|
|
}
|
|
}
|
|
|
|
inline void RegisterInputFactory(const std::string& name,
|
|
std::shared_ptr<Factory<InputDevice>> factory) {
|
|
RegisterFactory<InputDevice>(name, std::move(factory));
|
|
}
|
|
|
|
inline void RegisterOutputFactory(const std::string& name,
|
|
std::shared_ptr<Factory<OutputDevice>> factory) {
|
|
RegisterFactory<OutputDevice>(name, std::move(factory));
|
|
}
|
|
|
|
/**
|
|
* Unregisters an input device factory.
|
|
* @tparam InputDeviceType the type of input devices the factory can create
|
|
* @param name the name of the factory to unregister
|
|
*/
|
|
template <typename InputDeviceType>
|
|
void UnregisterFactory(const std::string& name) {
|
|
if (Impl::FactoryList<InputDeviceType>::list.erase(name) == 0) {
|
|
LOG_ERROR(Input, "Factory '{}' not registered", name);
|
|
}
|
|
}
|
|
|
|
inline void UnregisterInputFactory(const std::string& name) {
|
|
UnregisterFactory<InputDevice>(name);
|
|
}
|
|
|
|
inline void UnregisterOutputFactory(const std::string& name) {
|
|
UnregisterFactory<OutputDevice>(name);
|
|
}
|
|
|
|
/**
|
|
* Create an input device from given parameters.
|
|
* @tparam InputDeviceType the type of input devices to create
|
|
* @param params a serialized ParamPackage string that contains all parameters for creating the
|
|
* device
|
|
*/
|
|
template <typename InputDeviceType>
|
|
std::unique_ptr<InputDeviceType> CreateDeviceFromString(const std::string& params) {
|
|
const Common::ParamPackage package(params);
|
|
const std::string engine = package.Get("engine", "null");
|
|
const auto& factory_list = Impl::FactoryList<InputDeviceType>::list;
|
|
const auto pair = factory_list.find(engine);
|
|
if (pair == factory_list.end()) {
|
|
if (engine != "null") {
|
|
LOG_ERROR(Input, "Unknown engine name: {}", engine);
|
|
}
|
|
return std::make_unique<InputDeviceType>();
|
|
}
|
|
return pair->second->Create(package);
|
|
}
|
|
|
|
inline std::unique_ptr<InputDevice> CreateInputDeviceFromString(const std::string& params) {
|
|
return CreateDeviceFromString<InputDevice>(params);
|
|
}
|
|
|
|
inline std::unique_ptr<OutputDevice> CreateOutputDeviceFromString(const std::string& params) {
|
|
return CreateDeviceFromString<OutputDevice>(params);
|
|
}
|
|
|
|
/**
|
|
* Create an input device from given parameters.
|
|
* @tparam InputDeviceType the type of input devices to create
|
|
* @param package A ParamPackage that contains all parameters for creating the device
|
|
*/
|
|
template <typename InputDeviceType>
|
|
std::unique_ptr<InputDeviceType> CreateDevice(const ParamPackage& package) {
|
|
const std::string engine = package.Get("engine", "null");
|
|
const auto& factory_list = Impl::FactoryList<InputDeviceType>::list;
|
|
const auto pair = factory_list.find(engine);
|
|
if (pair == factory_list.end()) {
|
|
if (engine != "null") {
|
|
LOG_ERROR(Input, "Unknown engine name: {}", engine);
|
|
}
|
|
return std::make_unique<InputDeviceType>();
|
|
}
|
|
return pair->second->Create(package);
|
|
}
|
|
|
|
inline std::unique_ptr<InputDevice> CreateInputDevice(const ParamPackage& package) {
|
|
return CreateDevice<InputDevice>(package);
|
|
}
|
|
|
|
inline std::unique_ptr<OutputDevice> CreateOutputDevice(const ParamPackage& package) {
|
|
return CreateDevice<OutputDevice>(package);
|
|
}
|
|
|
|
} // namespace Common::Input
|