chore: make yuzu REUSE compliant
[REUSE] is a specification that aims at making file copyright
information consistent, so that it can be both human and machine
readable. It basically requires that all files have a header containing
copyright and licensing information. When this isn't possible, like
when dealing with binary assets, generated files or embedded third-party
dependencies, it is permitted to insert copyright information in the
`.reuse/dep5` file.
Oh, and it also requires that all the licenses used in the project are
present in the `LICENSES` folder, that's why the diff is so huge.
This can be done automatically with `reuse download --all`.
The `reuse` tool also contains a handy subcommand that analyzes the
project and tells whether or not the project is (still) compliant,
`reuse lint`.
Following REUSE has a few advantages over the current approach:
- Copyright information is easy to access for users / downstream
- Files like `dist/license.md` do not need to exist anymore, as
`.reuse/dep5` is used instead
- `reuse lint` makes it easy to ensure that copyright information of
files like binary assets / images is always accurate and up to date
To add copyright information of files that didn't have it I looked up
who committed what and when, for each file. As yuzu contributors do not
have to sign a CLA or similar I couldn't assume that copyright ownership
was of the "yuzu Emulator Project", so I used the name and/or email of
the commit author instead.
[REUSE]: https://reuse.software
Follow-up to 01cf05bc75b1e47beb08937439f3ed9339e7b254
2022-05-14 17:06:02 -07:00
|
|
|
// SPDX-FileCopyrightText: 2017 Citra Emulator Project
|
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2021-09-20 12:56:55 -07:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include <functional>
|
|
|
|
#include <memory>
|
|
|
|
#include <string>
|
|
|
|
#include <unordered_map>
|
|
|
|
#include <utility>
|
2022-11-21 08:31:18 -08:00
|
|
|
#include <vector>
|
2021-09-20 12:56:55 -07:00
|
|
|
#include "common/logging/log.h"
|
|
|
|
#include "common/param_package.h"
|
2021-11-02 21:50:30 -07:00
|
|
|
#include "common/uuid.h"
|
2021-09-20 12:56:55 -07:00
|
|
|
|
2021-10-30 20:23:10 -07:00
|
|
|
namespace Common::Input {
|
2021-09-20 12:56:55 -07:00
|
|
|
|
2023-03-11 19:10:38 -08:00
|
|
|
// Type of data that is expected to receive or send
|
2021-09-20 12:56:55 -07:00
|
|
|
enum class InputType {
|
|
|
|
None,
|
|
|
|
Battery,
|
|
|
|
Button,
|
|
|
|
Stick,
|
|
|
|
Analog,
|
|
|
|
Trigger,
|
|
|
|
Motion,
|
|
|
|
Touch,
|
|
|
|
Color,
|
|
|
|
Vibration,
|
|
|
|
Nfc,
|
2022-06-18 21:32:07 -07:00
|
|
|
IrSensor,
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Internal battery charge level
|
2021-11-01 13:17:53 -07:00
|
|
|
enum class BatteryLevel : u32 {
|
2021-10-18 22:12:24 -07:00
|
|
|
None,
|
2021-09-20 12:56:55 -07:00
|
|
|
Empty,
|
|
|
|
Critical,
|
|
|
|
Low,
|
|
|
|
Medium,
|
|
|
|
Full,
|
|
|
|
Charging,
|
|
|
|
};
|
|
|
|
|
2021-10-10 22:43:11 -07:00
|
|
|
enum class PollingMode {
|
2021-11-03 21:35:45 -07:00
|
|
|
// Constant polling of buttons, analogs and motion data
|
2021-10-10 22:43:11 -07:00
|
|
|
Active,
|
2021-11-03 21:35:45 -07:00
|
|
|
// Only update on button change, digital analogs
|
2023-03-07 18:15:46 -08:00
|
|
|
Passive,
|
2021-11-03 21:35:45 -07:00
|
|
|
// Enable near field communication polling
|
|
|
|
NFC,
|
|
|
|
// Enable infrared camera polling
|
2021-10-10 22:43:11 -07:00
|
|
|
IR,
|
2022-12-20 09:34:33 -08:00
|
|
|
// Enable ring controller polling
|
|
|
|
Ring,
|
2021-10-10 22:43:11 -07:00
|
|
|
};
|
|
|
|
|
2022-06-18 21:32:07 -07:00
|
|
|
enum class CameraFormat {
|
|
|
|
Size320x240,
|
|
|
|
Size160x120,
|
|
|
|
Size80x60,
|
|
|
|
Size40x30,
|
|
|
|
Size20x15,
|
|
|
|
None,
|
|
|
|
};
|
|
|
|
|
2022-12-26 09:11:01 -08:00
|
|
|
// Different results that can happen from a device request
|
|
|
|
enum class DriverResult {
|
|
|
|
Success,
|
|
|
|
WrongReply,
|
|
|
|
Timeout,
|
|
|
|
UnsupportedControllerType,
|
|
|
|
HandleInUse,
|
|
|
|
ErrorReadingData,
|
|
|
|
ErrorWritingData,
|
|
|
|
NoDeviceDetected,
|
2022-12-20 09:34:33 -08:00
|
|
|
InvalidHandle,
|
2023-06-27 23:20:38 -07:00
|
|
|
InvalidParameters,
|
2021-10-10 22:43:11 -07:00
|
|
|
NotSupported,
|
2022-12-26 09:11:01 -08:00
|
|
|
Disabled,
|
2023-06-27 23:20:38 -07:00
|
|
|
Delayed,
|
2021-10-10 22:43:11 -07:00
|
|
|
Unknown,
|
|
|
|
};
|
|
|
|
|
2022-09-24 17:46:49 -07:00
|
|
|
// Nfc reply from the controller
|
|
|
|
enum class NfcState {
|
|
|
|
Success,
|
|
|
|
NewAmiibo,
|
|
|
|
WaitingForAmiibo,
|
|
|
|
AmiiboRemoved,
|
2023-06-16 20:57:21 -07:00
|
|
|
InvalidTagType,
|
2022-09-24 17:46:49 -07:00
|
|
|
NotSupported,
|
|
|
|
WrongDeviceState,
|
|
|
|
WriteFailed,
|
|
|
|
Unknown,
|
|
|
|
};
|
|
|
|
|
2021-10-20 12:41:56 -07:00
|
|
|
// Hint for amplification curve to be used
|
|
|
|
enum class VibrationAmplificationType {
|
|
|
|
Linear,
|
|
|
|
Exponential,
|
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Analog properties for calibration
|
2021-09-20 12:56:55 -07:00
|
|
|
struct AnalogProperties {
|
2021-11-03 21:35:45 -07:00
|
|
|
// Anything below this value will be detected as zero
|
2021-09-20 12:56:55 -07:00
|
|
|
float deadzone{};
|
2023-03-11 19:10:38 -08:00
|
|
|
// Anything above this values will be detected as one
|
2021-09-20 12:56:55 -07:00
|
|
|
float range{1.0f};
|
2021-11-03 21:35:45 -07:00
|
|
|
// Minimum value to be detected as active
|
2021-09-20 12:56:55 -07:00
|
|
|
float threshold{0.5f};
|
2021-11-03 21:35:45 -07:00
|
|
|
// Drift correction applied to the raw data
|
2021-09-20 12:56:55 -07:00
|
|
|
float offset{};
|
2021-11-03 21:35:45 -07:00
|
|
|
// Invert direction of the sensor data
|
2021-09-20 12:56:55 -07:00
|
|
|
bool inverted{};
|
2023-05-05 16:11:53 -07:00
|
|
|
// Invert the state if it's converted to a button
|
|
|
|
bool inverted_button{};
|
2022-09-06 09:20:53 -07:00
|
|
|
// Press once to activate, press again to release
|
|
|
|
bool toggle{};
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Single analog sensor data
|
2021-09-20 12:56:55 -07:00
|
|
|
struct AnalogStatus {
|
|
|
|
float value{};
|
|
|
|
float raw_value{};
|
|
|
|
AnalogProperties properties{};
|
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Button data
|
2021-09-20 12:56:55 -07:00
|
|
|
struct ButtonStatus {
|
2021-11-02 21:50:30 -07:00
|
|
|
Common::UUID uuid{};
|
2021-09-20 12:56:55 -07:00
|
|
|
bool value{};
|
2022-09-06 09:20:53 -07:00
|
|
|
// Invert value of the button
|
2021-09-20 12:56:55 -07:00
|
|
|
bool inverted{};
|
2022-09-06 09:20:53 -07:00
|
|
|
// Press once to activate, press again to release
|
2021-09-20 12:56:55 -07:00
|
|
|
bool toggle{};
|
2023-01-28 16:19:15 -08:00
|
|
|
// Spams the button when active
|
|
|
|
bool turbo{};
|
2022-09-06 09:20:53 -07:00
|
|
|
// Internal lock for the toggle status
|
2021-09-20 12:56:55 -07:00
|
|
|
bool locked{};
|
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Internal battery data
|
2021-09-20 12:56:55 -07:00
|
|
|
using BatteryStatus = BatteryLevel;
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Analog and digital joystick data
|
2021-09-20 12:56:55 -07:00
|
|
|
struct StickStatus {
|
2021-11-02 21:50:30 -07:00
|
|
|
Common::UUID uuid{};
|
2021-09-20 12:56:55 -07:00
|
|
|
AnalogStatus x{};
|
|
|
|
AnalogStatus y{};
|
|
|
|
bool left{};
|
|
|
|
bool right{};
|
|
|
|
bool up{};
|
|
|
|
bool down{};
|
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Analog and digital trigger data
|
2021-09-20 12:56:55 -07:00
|
|
|
struct TriggerStatus {
|
2021-11-02 21:50:30 -07:00
|
|
|
Common::UUID uuid{};
|
2021-09-20 12:56:55 -07:00
|
|
|
AnalogStatus analog{};
|
2021-10-31 08:41:44 -07:00
|
|
|
ButtonStatus pressed{};
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// 3D vector representing motion input
|
2021-09-20 12:56:55 -07:00
|
|
|
struct MotionSensor {
|
|
|
|
AnalogStatus x{};
|
|
|
|
AnalogStatus y{};
|
|
|
|
AnalogStatus z{};
|
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Motion data used to calculate controller orientation
|
2021-09-20 12:56:55 -07:00
|
|
|
struct MotionStatus {
|
2021-11-01 18:49:14 -07:00
|
|
|
// Gyroscope vector measurement in radians/s.
|
2021-09-20 12:56:55 -07:00
|
|
|
MotionSensor gyro{};
|
2021-11-01 18:49:14 -07:00
|
|
|
// Acceleration vector measurement in G force
|
2021-09-20 12:56:55 -07:00
|
|
|
MotionSensor accel{};
|
2021-11-01 18:49:14 -07:00
|
|
|
// Time since last measurement in microseconds
|
2021-09-20 12:56:55 -07:00
|
|
|
u64 delta_timestamp{};
|
2021-11-01 18:49:14 -07:00
|
|
|
// Request to update after reading the value
|
|
|
|
bool force_update{};
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Data of a single point on a touch screen
|
2021-09-20 12:56:55 -07:00
|
|
|
struct TouchStatus {
|
|
|
|
ButtonStatus pressed{};
|
|
|
|
AnalogStatus x{};
|
|
|
|
AnalogStatus y{};
|
2021-10-31 08:41:44 -07:00
|
|
|
int id{};
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Physical controller color in RGB format
|
2021-09-20 12:56:55 -07:00
|
|
|
struct BodyColorStatus {
|
|
|
|
u32 body{};
|
|
|
|
u32 buttons{};
|
2022-12-20 09:34:33 -08:00
|
|
|
u32 left_grip{};
|
|
|
|
u32 right_grip{};
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// HD rumble data
|
2021-09-20 12:56:55 -07:00
|
|
|
struct VibrationStatus {
|
|
|
|
f32 low_amplitude{};
|
|
|
|
f32 low_frequency{};
|
|
|
|
f32 high_amplitude{};
|
|
|
|
f32 high_frequency{};
|
2021-10-20 12:41:56 -07:00
|
|
|
VibrationAmplificationType type;
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Physical controller LED pattern
|
2021-09-20 12:56:55 -07:00
|
|
|
struct LedStatus {
|
|
|
|
bool led_1{};
|
|
|
|
bool led_2{};
|
|
|
|
bool led_3{};
|
|
|
|
bool led_4{};
|
|
|
|
};
|
|
|
|
|
2023-03-11 19:10:38 -08:00
|
|
|
// Raw data from camera
|
2022-06-18 21:32:07 -07:00
|
|
|
struct CameraStatus {
|
|
|
|
CameraFormat format{CameraFormat::None};
|
|
|
|
std::vector<u8> data{};
|
|
|
|
};
|
|
|
|
|
2022-09-24 17:46:49 -07:00
|
|
|
struct NfcStatus {
|
2023-06-16 20:57:21 -07:00
|
|
|
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;
|
2022-09-24 17:46:49 -07:00
|
|
|
};
|
|
|
|
|
2021-11-21 12:12:01 -08:00
|
|
|
// 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,
|
2022-12-20 09:34:33 -08:00
|
|
|
|
|
|
|
// Joycon button names
|
2021-11-21 12:12:01 -08:00
|
|
|
ButtonLeft,
|
|
|
|
ButtonRight,
|
|
|
|
ButtonDown,
|
|
|
|
ButtonUp,
|
|
|
|
ButtonA,
|
|
|
|
ButtonB,
|
|
|
|
ButtonX,
|
|
|
|
ButtonY,
|
2022-12-20 09:34:33 -08:00
|
|
|
ButtonPlus,
|
|
|
|
ButtonMinus,
|
|
|
|
ButtonHome,
|
|
|
|
ButtonCapture,
|
|
|
|
ButtonStickL,
|
|
|
|
ButtonStickR,
|
|
|
|
TriggerL,
|
|
|
|
TriggerZL,
|
|
|
|
TriggerSL,
|
|
|
|
TriggerR,
|
|
|
|
TriggerZR,
|
|
|
|
TriggerSR,
|
|
|
|
|
|
|
|
// GC button names
|
|
|
|
TriggerZ,
|
2021-11-21 12:12:01 -08:00
|
|
|
ButtonStart,
|
2021-11-26 13:45:37 -08:00
|
|
|
|
|
|
|
// DS4 button names
|
|
|
|
L1,
|
|
|
|
L2,
|
|
|
|
L3,
|
|
|
|
R1,
|
|
|
|
R2,
|
|
|
|
R3,
|
|
|
|
Circle,
|
|
|
|
Cross,
|
|
|
|
Square,
|
|
|
|
Triangle,
|
|
|
|
Share,
|
|
|
|
Options,
|
2022-01-30 07:15:29 -08:00
|
|
|
Home,
|
|
|
|
Touch,
|
2022-01-16 16:03:11 -08:00
|
|
|
|
|
|
|
// Mouse buttons
|
|
|
|
ButtonMouseWheel,
|
|
|
|
ButtonBackward,
|
|
|
|
ButtonForward,
|
|
|
|
ButtonTask,
|
|
|
|
ButtonExtra,
|
2021-11-21 12:12:01 -08:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Callback data consisting of an input type and the equivalent data status
|
2021-09-20 12:56:55 -07:00
|
|
|
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{};
|
2022-10-09 10:49:07 -07:00
|
|
|
CameraFormat camera_status{CameraFormat::None};
|
2023-06-16 20:57:21 -07:00
|
|
|
NfcStatus nfc_status{};
|
2022-10-09 10:49:07 -07:00
|
|
|
std::vector<u8> raw_data{};
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Triggered once every input change
|
2021-09-20 12:56:55 -07:00
|
|
|
struct InputCallback {
|
2021-12-13 18:09:28 -08:00
|
|
|
std::function<void(const CallbackStatus&)> on_change;
|
2021-09-20 12:56:55 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
/// An abstract class template for an input device (a button, an analog input, etc.).
|
|
|
|
class InputDevice {
|
|
|
|
public:
|
|
|
|
virtual ~InputDevice() = default;
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Force input device to update data regardless of the current state
|
2021-12-13 17:43:09 -08:00
|
|
|
virtual void ForceUpdate() {}
|
2021-10-24 18:28:54 -07:00
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Sets the function to be triggered when input changes
|
2021-09-20 12:56:55 -07:00
|
|
|
void SetCallback(InputCallback callback_) {
|
|
|
|
callback = std::move(callback_);
|
|
|
|
}
|
|
|
|
|
2021-11-03 21:35:45 -07:00
|
|
|
// Triggers the function set in the callback
|
2021-12-13 18:09:28 -08:00
|
|
|
void TriggerOnChange(const CallbackStatus& status) {
|
2021-09-20 12:56:55 -07:00
|
|
|
if (callback.on_change) {
|
|
|
|
callback.on_change(status);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
InputCallback callback;
|
|
|
|
};
|
|
|
|
|
2021-10-10 22:43:11 -07:00
|
|
|
/// An abstract class template for an output device (rumble, LED pattern, polling mode).
|
|
|
|
class OutputDevice {
|
|
|
|
public:
|
|
|
|
virtual ~OutputDevice() = default;
|
|
|
|
|
2022-12-26 09:11:01 -08:00
|
|
|
virtual DriverResult SetLED([[maybe_unused]] const LedStatus& led_status) {
|
|
|
|
return DriverResult::NotSupported;
|
|
|
|
}
|
2021-10-10 22:43:11 -07:00
|
|
|
|
2022-12-26 09:11:01 -08:00
|
|
|
virtual DriverResult SetVibration([[maybe_unused]] const VibrationStatus& vibration_status) {
|
|
|
|
return DriverResult::NotSupported;
|
2021-10-10 22:43:11 -07:00
|
|
|
}
|
|
|
|
|
2022-10-20 22:23:12 -07:00
|
|
|
virtual bool IsVibrationEnabled() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2022-12-26 09:11:01 -08:00
|
|
|
virtual DriverResult SetPollingMode([[maybe_unused]] PollingMode polling_mode) {
|
|
|
|
return DriverResult::NotSupported;
|
2021-10-10 22:43:11 -07:00
|
|
|
}
|
2022-06-18 21:32:07 -07:00
|
|
|
|
2022-12-26 09:11:01 -08:00
|
|
|
virtual DriverResult SetCameraFormat([[maybe_unused]] CameraFormat camera_format) {
|
|
|
|
return DriverResult::NotSupported;
|
2022-06-18 21:32:07 -07:00
|
|
|
}
|
2022-09-24 17:46:49 -07:00
|
|
|
|
2022-09-24 20:52:33 -07:00
|
|
|
virtual NfcState SupportsNfc() const {
|
2022-09-24 17:46:49 -07:00
|
|
|
return NfcState::NotSupported;
|
|
|
|
}
|
|
|
|
|
2023-06-16 20:57:21 -07:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2022-09-24 17:46:49 -07:00
|
|
|
virtual NfcState WriteNfcData([[maybe_unused]] const std::vector<u8>& data) {
|
|
|
|
return NfcState::NotSupported;
|
|
|
|
}
|
2023-06-16 20:57:21 -07:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2021-10-10 22:43:11 -07:00
|
|
|
};
|
|
|
|
|
2021-09-20 12:56:55 -07:00
|
|
|
/// 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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-28 06:19:01 -08:00
|
|
|
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));
|
|
|
|
}
|
|
|
|
|
2021-09-20 12:56:55 -07:00
|
|
|
/**
|
|
|
|
* 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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-28 06:19:01 -08:00
|
|
|
inline void UnregisterInputFactory(const std::string& name) {
|
|
|
|
UnregisterFactory<InputDevice>(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void UnregisterOutputFactory(const std::string& name) {
|
|
|
|
UnregisterFactory<OutputDevice>(name);
|
|
|
|
}
|
|
|
|
|
2021-09-20 12:56:55 -07:00
|
|
|
/**
|
2023-03-11 19:10:38 -08:00
|
|
|
* Create an input device from given parameters.
|
2021-09-20 12:56:55 -07:00
|
|
|
* @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);
|
|
|
|
}
|
|
|
|
|
2022-11-28 06:19:01 -08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2021-09-20 12:56:55 -07:00
|
|
|
/**
|
2022-11-28 06:11:56 -08:00
|
|
|
* Create an input device from given parameters.
|
2021-09-20 12:56:55 -07:00
|
|
|
* @tparam InputDeviceType the type of input devices to create
|
2022-11-28 06:11:56 -08:00
|
|
|
* @param package A ParamPackage that contains all parameters for creating the device
|
2021-09-20 12:56:55 -07:00
|
|
|
*/
|
|
|
|
template <typename InputDeviceType>
|
2022-11-28 06:11:56 -08:00
|
|
|
std::unique_ptr<InputDeviceType> CreateDevice(const ParamPackage& package) {
|
2021-09-20 12:56:55 -07:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2022-11-28 06:19:01 -08:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2021-10-30 20:23:10 -07:00
|
|
|
} // namespace Common::Input
|