| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  | // Copyright 2017 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <memory>
 | 
					
						
							|  |  |  | #include <string>
 | 
					
						
							|  |  |  | #include <tuple>
 | 
					
						
							|  |  |  | #include <unordered_map>
 | 
					
						
							|  |  |  | #include <utility>
 | 
					
						
							|  |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "common/param_package.h"
 | 
					
						
							| 
									
										
										
										
											2017-08-06 22:54:19 +03:00
										 |  |  | #include "common/vector_math.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace Input { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-01-15 11:25:15 +01:00
										 |  |  | enum class AnalogDirection : u8 { | 
					
						
							|  |  |  |     RIGHT, | 
					
						
							|  |  |  |     LEFT, | 
					
						
							|  |  |  |     UP, | 
					
						
							|  |  |  |     DOWN, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  | /// An abstract class template for an input device (a button, an analog input, etc.).
 | 
					
						
							|  |  |  | template <typename StatusType> | 
					
						
							|  |  |  | class InputDevice { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     virtual ~InputDevice() = default; | 
					
						
							|  |  |  |     virtual StatusType GetStatus() const { | 
					
						
							|  |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2020-01-15 11:25:15 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-15 06:22:01 -05:00
										 |  |  |     virtual bool GetAnalogDirectionStatus([[maybe_unused]] AnalogDirection direction) const { | 
					
						
							| 
									
										
										
										
											2020-01-15 11:25:15 +01:00
										 |  |  |         return {}; | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-11-23 18:11:21 -08:00
										 |  |  |     virtual bool SetRumblePlay([[maybe_unused]] f32 amp_low, [[maybe_unused]] f32 freq_low, | 
					
						
							|  |  |  |                                [[maybe_unused]] f32 amp_high, | 
					
						
							|  |  |  |                                [[maybe_unused]] f32 freq_high) const { | 
					
						
							| 
									
										
										
										
											2020-07-10 21:20:50 -05:00
										 |  |  |         return {}; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02: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) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_ERROR(Input, "Factory '{}' already registered", name); | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02: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) { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |         LOG_ERROR(Input, "Factory '{}' not registered", name); | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Create an input device from given paramters. | 
					
						
							|  |  |  |  * @tparam InputDeviceType the type of input devices to create | 
					
						
							|  |  |  |  * @param params a serialized ParamPackage string contains all parameters for creating the device | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template <typename InputDeviceType> | 
					
						
							|  |  |  | std::unique_ptr<InputDeviceType> CreateDevice(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") { | 
					
						
							| 
									
										
										
										
											2018-07-02 10:13:26 -06:00
										 |  |  |             LOG_ERROR(Input, "Unknown engine name: {}", engine); | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |         return std::make_unique<InputDeviceType>(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return pair->second->Create(package); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 22:46:39 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A button device is an input device that returns bool as status. | 
					
						
							|  |  |  |  * true for pressed; false for released. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | using ButtonDevice = InputDevice<bool>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 23:58:03 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * An analog device is an input device that returns a tuple of x and y coordinates as status. The | 
					
						
							|  |  |  |  * coordinates are within the unit circle. x+ is defined as right direction, and y+ is defined as up | 
					
						
							|  |  |  |  * direction | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | using AnalogDevice = InputDevice<std::tuple<float, float>>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-10-20 13:55:25 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A vibration device is an input device that returns an unsigned byte as status. | 
					
						
							|  |  |  |  * It represents whether the vibration device supports vibration or not. | 
					
						
							|  |  |  |  * If the status returns 1, it supports vibration. Otherwise, it does not support vibration. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | using VibrationDevice = InputDevice<u8>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-06 22:54:19 +03:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2020-09-02 19:59:34 -05:00
										 |  |  |  * A motion status is an object that returns a tuple of accelerometer state vector, | 
					
						
							| 
									
										
										
										
											2020-09-04 21:48:03 -05:00
										 |  |  |  * gyroscope state vector, rotation state vector and orientation state matrix. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For both vectors: | 
					
						
							|  |  |  |  *   x+ is the same direction as RIGHT on D-pad. | 
					
						
							|  |  |  |  *   y+ is normal to the touch screen, pointing outward. | 
					
						
							|  |  |  |  *   z+ is the same direction as UP on D-pad. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For accelerometer state vector | 
					
						
							|  |  |  |  *   Units: g (gravitational acceleration) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For gyroscope state vector: | 
					
						
							|  |  |  |  *   Orientation is determined by right-hand rule. | 
					
						
							|  |  |  |  *   Units: deg/sec | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For rotation state vector | 
					
						
							|  |  |  |  *   Units: rotations | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * For orientation state matrix | 
					
						
							|  |  |  |  *   x vector | 
					
						
							|  |  |  |  *   y vector | 
					
						
							|  |  |  |  *   z vector | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-09-02 19:59:34 -05:00
										 |  |  | using MotionStatus = std::tuple<Common::Vec3<float>, Common::Vec3<float>, Common::Vec3<float>, | 
					
						
							|  |  |  |                                 std::array<Common::Vec3f, 3>>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * A motion device is an input device that returns a motion status object | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | using MotionDevice = InputDevice<MotionStatus>; | 
					
						
							| 
									
										
										
										
											2020-09-04 21:48:03 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-08-08 21:34:17 +03:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2021-01-01 12:32:29 -06:00
										 |  |  |  * A touch status is an object that returns an array of 16 tuple elements of two floats and a bool. | 
					
						
							|  |  |  |  * The floats are x and y coordinates in the range 0.0 - 1.0, and the bool indicates whether it is | 
					
						
							|  |  |  |  * pressed. | 
					
						
							| 
									
										
										
										
											2017-08-08 21:34:17 +03:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-01-01 12:32:29 -06:00
										 |  |  | using TouchStatus = std::array<std::tuple<float, float, bool>, 16>; | 
					
						
							| 
									
										
										
										
											2020-11-17 22:55:09 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * A touch device is an input device that returns a touch status object | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | using TouchDevice = InputDevice<TouchStatus>; | 
					
						
							| 
									
										
										
										
											2017-08-08 21:34:17 +03:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-01 21:52:51 -04:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A mouse device is an input device that returns a tuple of two floats and four ints. | 
					
						
							|  |  |  |  * The first two floats are X and Y device coordinates of the mouse (from 0-1). | 
					
						
							|  |  |  |  * The s32s are the mouse wheel. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | using MouseDevice = InputDevice<std::tuple<float, float, s32, s32>>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-20 21:52:32 +02:00
										 |  |  | } // namespace Input
 |