mirror of https://github.com/stenzek/duckstation
				
				
				
			
			You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
	
	
		
			419 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
			
		
		
	
	
			419 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C++
		
	
| // SPDX-FileCopyrightText: 2019-2024 Connor McLaughlin <stenzek@gmail.com>
 | |
| // SPDX-License-Identifier: CC-BY-NC-ND-4.0
 | |
| 
 | |
| #pragma once
 | |
| 
 | |
| #include <functional>
 | |
| #include <mutex>
 | |
| #include <optional>
 | |
| #include <string_view>
 | |
| #include <tuple>
 | |
| #include <utility>
 | |
| #include <variant>
 | |
| 
 | |
| #include "common/settings_interface.h"
 | |
| #include "common/types.h"
 | |
| 
 | |
| #include "core/input_types.h"
 | |
| #include "window_info.h"
 | |
| 
 | |
| class Error;
 | |
| class SmallStringBase;
 | |
| 
 | |
| /// Class, or source of an input event.
 | |
| enum class InputSourceType : u32
 | |
| {
 | |
|   Keyboard,
 | |
|   Pointer,
 | |
|   Sensor,
 | |
| #ifdef _WIN32
 | |
|   DInput,
 | |
|   XInput,
 | |
|   RawInput,
 | |
| #endif
 | |
| #ifdef ENABLE_SDL
 | |
|   SDL,
 | |
| #endif
 | |
| #ifdef __ANDROID__
 | |
|   Android,
 | |
| #endif
 | |
|   Count,
 | |
| };
 | |
| 
 | |
| /// Subtype of a key for an input source.
 | |
| enum class InputSubclass : u32
 | |
| {
 | |
|   None = 0,
 | |
| 
 | |
|   PointerButton = 0,
 | |
|   PointerAxis = 1,
 | |
| 
 | |
|   ControllerButton = 0,
 | |
|   ControllerAxis = 1,
 | |
|   ControllerHat = 2,
 | |
|   ControllerMotor = 3,
 | |
|   ControllerHaptic = 4,
 | |
| 
 | |
|   SensorAccelerometer = 0,
 | |
| };
 | |
| 
 | |
| enum class InputModifier : u32
 | |
| {
 | |
|   None = 0,
 | |
|   Negate,   ///< Input * -1, gets the negative side of the axis
 | |
|   FullAxis, ///< (Input * 0.5) + 0.5, uses both the negative and positive side of the axis together
 | |
| };
 | |
| 
 | |
| /// A composite type representing a full input key which is part of an event.
 | |
| union InputBindingKey
 | |
| {
 | |
|   struct
 | |
|   {
 | |
|     InputSourceType source_type : 4;
 | |
|     u32 source_index : 8;             ///< controller number
 | |
|     InputSubclass source_subtype : 3; ///< if 1, binding is for an axis and not a button (used for controllers)
 | |
|     InputModifier modifier : 2;
 | |
|     u32 invert : 1; ///< if 1, value is inverted prior to being sent to the sink
 | |
|     u32 : 14;
 | |
|     u32 data;
 | |
|   };
 | |
| 
 | |
|   u64 bits;
 | |
| 
 | |
|   bool operator==(const InputBindingKey& k) const { return bits == k.bits; }
 | |
|   bool operator!=(const InputBindingKey& k) const { return bits != k.bits; }
 | |
| 
 | |
|   /// Removes the direction bit from the key, which is used to look up the bindings for it.
 | |
|   /// This is because negative bindings should still fire when they reach zero again.
 | |
|   InputBindingKey MaskDirection() const
 | |
|   {
 | |
|     InputBindingKey r;
 | |
|     r.bits = bits;
 | |
|     r.modifier = InputModifier::None;
 | |
|     r.invert = 0;
 | |
|     return r;
 | |
|   }
 | |
| };
 | |
| static_assert(sizeof(InputBindingKey) == sizeof(u64), "Input binding key is 64 bits");
 | |
| 
 | |
| /// Hashability for InputBindingKey
 | |
| struct InputBindingKeyHash
 | |
| {
 | |
|   std::size_t operator()(const InputBindingKey& k) const { return std::hash<u64>{}(k.bits); }
 | |
| };
 | |
| 
 | |
| /// Callback type for a binary event. Usually used for hotkeys.
 | |
| using InputButtonEventHandler = std::function<void(s32 value)>;
 | |
| 
 | |
| /// Callback types for a normalized event. Usually used for pads.
 | |
| using InputAxisEventHandler = std::function<void(float value)>;
 | |
| 
 | |
| /// ------------------------------------------------------------------------
 | |
| /// Event Handler Type
 | |
| /// ------------------------------------------------------------------------
 | |
| /// This class acts as an adapter to convert from normalized values to
 | |
| /// binary values when the callback is a binary/button handler. That way
 | |
| /// you don't need to convert float->bool in your callbacks.
 | |
| using InputEventHandler = std::variant<InputAxisEventHandler, InputButtonEventHandler>;
 | |
| 
 | |
| /// Input monitoring for external access.
 | |
| struct InputInterceptHook
 | |
| {
 | |
|   enum class CallbackResult
 | |
|   {
 | |
|     StopProcessingEvent,
 | |
|     ContinueProcessingEvent,
 | |
|     RemoveHookAndStopProcessingEvent,
 | |
|     RemoveHookAndContinueProcessingEvent,
 | |
|   };
 | |
| 
 | |
|   using Callback = std::function<CallbackResult(InputBindingKey key, float value)>;
 | |
| };
 | |
| 
 | |
| /// Hotkeys are actions (e.g. toggle frame limit) which can be bound to keys or chords.
 | |
| /// The handler is called with an integer representing the key state, where 0 means that
 | |
| /// one or more keys were released, 1 means all the keys were pressed, and -1 means that
 | |
| /// the hotkey was cancelled due to a chord with more keys being activated.
 | |
| struct HotkeyInfo
 | |
| {
 | |
|   const char* name;
 | |
|   const char* category;
 | |
|   const char* display_name;
 | |
|   void (*handler)(s32 pressed);
 | |
| };
 | |
| #define DECLARE_HOTKEY_LIST(name) extern const HotkeyInfo name[]
 | |
| #define BEGIN_HOTKEY_LIST(name) const HotkeyInfo name[] = {
 | |
| #define DEFINE_HOTKEY(name, category, display_name, handler) {(name), (category), (display_name), (handler)},
 | |
| #define END_HOTKEY_LIST()                                                                                              \
 | |
|   {                                                                                                                    \
 | |
|     nullptr, nullptr, nullptr, nullptr                                                                                 \
 | |
|   }                                                                                                                    \
 | |
|   }                                                                                                                    \
 | |
|   ;
 | |
| 
 | |
| DECLARE_HOTKEY_LIST(g_common_hotkeys);
 | |
| DECLARE_HOTKEY_LIST(g_host_hotkeys);
 | |
| 
 | |
| /// Generic input bindings. These roughly match a DualShock 4 or XBox One controller.
 | |
| /// They are used for automatic binding to PS2 controller types, and for big picture mode navigation.
 | |
| enum class GenericInputBinding : u8;
 | |
| using GenericInputBindingMapping = std::vector<std::pair<GenericInputBinding, std::string>>;
 | |
| 
 | |
| /// Host mouse relative axes are X, Y, wheel horizontal, wheel vertical.
 | |
| enum class InputPointerAxis : u8
 | |
| {
 | |
|   X,
 | |
|   Y,
 | |
|   WheelX,
 | |
|   WheelY,
 | |
|   Count
 | |
| };
 | |
| 
 | |
| /// External input source class.
 | |
| class InputSource;
 | |
| 
 | |
| /// Force feedback interface.
 | |
| class ForceFeedbackDevice
 | |
| {
 | |
| public:
 | |
|   enum class Effect
 | |
|   {
 | |
|     Constant,
 | |
|   };
 | |
| 
 | |
|   virtual ~ForceFeedbackDevice();
 | |
| 
 | |
|   virtual void SetConstantForce(s32 level) = 0;
 | |
| 
 | |
|   virtual void DisableForce(Effect force) = 0;
 | |
| };
 | |
| 
 | |
| namespace InputManager {
 | |
| 
 | |
| /// Maximum number of buttons that can be part of a chord.
 | |
| static constexpr u32 MAX_KEYS_PER_BINDING = 4;
 | |
| 
 | |
| /// Maximum number of output vibration motors per pad.
 | |
| static constexpr u32 MAX_MOTORS_PER_PAD = 2;
 | |
| 
 | |
| /// Minimum interval between vibration updates when the effect is continuous.
 | |
| static constexpr double VIBRATION_UPDATE_INTERVAL_SECONDS = 0.5; // 500ms
 | |
| 
 | |
| /// Maximum number of host mouse devices.
 | |
| static constexpr u32 MAX_POINTER_DEVICES = 8;
 | |
| static constexpr u32 MAX_POINTER_BUTTONS = 3;
 | |
| 
 | |
| /// Maximum number of software cursors. We allocate an extra two for controllers with
 | |
| /// positioning data from the controller instead of a mouse.
 | |
| static constexpr u32 MAX_SOFTWARE_CURSORS = MAX_POINTER_DEVICES + 2;
 | |
| 
 | |
| /// Number of macro buttons per controller.
 | |
| static constexpr u32 NUM_MACRO_BUTTONS_PER_CONTROLLER = 4;
 | |
| 
 | |
| /// Returns a pointer to the external input source class, if present.
 | |
| InputSource* GetInputSourceInterface(InputSourceType type);
 | |
| 
 | |
| /// Converts an input class to a string.
 | |
| const char* InputSourceToString(InputSourceType clazz);
 | |
| 
 | |
| /// Returns the default state for an input source.
 | |
| bool GetInputSourceDefaultEnabled(InputSourceType type);
 | |
| 
 | |
| /// Parses an input class string.
 | |
| std::optional<InputSourceType> ParseInputSourceString(std::string_view str);
 | |
| 
 | |
| /// Parses a pointer device string, i.e. tells you which pointer is specified.
 | |
| std::optional<u32> GetIndexFromPointerBinding(std::string_view str);
 | |
| 
 | |
| /// Returns the device name for a pointer index (e.g. Pointer-0).
 | |
| std::string GetPointerDeviceName(u32 pointer_index);
 | |
| 
 | |
| /// Converts a key code from a human-readable string to an identifier.
 | |
| std::optional<u32> ConvertHostKeyboardStringToCode(std::string_view str);
 | |
| 
 | |
| /// Converts a key code from an identifier to a human-readable string.
 | |
| std::optional<std::string> ConvertHostKeyboardCodeToString(u32 code);
 | |
| 
 | |
| /// Converts a key code from an identifier to an icon which can be drawn.
 | |
| const char* ConvertHostKeyboardCodeToIcon(u32 code);
 | |
| 
 | |
| /// Creates a key for a host-specific key code.
 | |
| InputBindingKey MakeHostKeyboardKey(u32 key_code);
 | |
| 
 | |
| /// Creates a key for a host-specific button.
 | |
| InputBindingKey MakePointerButtonKey(u32 index, u32 button_index);
 | |
| 
 | |
| /// Creates a key for a host-specific mouse relative event
 | |
| /// (axis 0 = horizontal, 1 = vertical, 2 = wheel horizontal, 3 = wheel vertical).
 | |
| InputBindingKey MakePointerAxisKey(u32 index, InputPointerAxis axis);
 | |
| 
 | |
| /// Creates a key for a host-specific sensor.
 | |
| InputBindingKey MakeSensorAxisKey(InputSubclass sensor, u32 axis);
 | |
| 
 | |
| /// Parses an input binding key string.
 | |
| std::optional<InputBindingKey> ParseInputBindingKey(std::string_view binding);
 | |
| 
 | |
| /// Converts a input key to a string.
 | |
| std::string ConvertInputBindingKeyToString(InputBindingInfo::Type binding_type, InputBindingKey key);
 | |
| 
 | |
| /// Converts a chord of binding keys to a string.
 | |
| std::string ConvertInputBindingKeysToString(InputBindingInfo::Type binding_type, const InputBindingKey* keys,
 | |
|                                             size_t num_keys);
 | |
| 
 | |
| /// Represents a binding with icon fonts, if available.
 | |
| /// Optionally maps icon fonts to a different style, e.g. xbox icons -> PS buttons.
 | |
| using BindingIconMappingFunction = std::string_view (*)(std::string_view);
 | |
| bool PrettifyInputBinding(SmallStringBase& binding, BindingIconMappingFunction mapper = nullptr);
 | |
| 
 | |
| /// Returns a list of all hotkeys.
 | |
| std::vector<const HotkeyInfo*> GetHotkeyList();
 | |
| 
 | |
| /// Enumerates available devices. Returns a pair of the prefix (e.g. SDL-0) and the device name.
 | |
| using DeviceList = std::vector<std::tuple<InputBindingKey, std::string, std::string>>;
 | |
| DeviceList EnumerateDevices();
 | |
| 
 | |
| /// Enumerates available vibration motors at the time of call.
 | |
| using VibrationMotorList = std::vector<InputBindingKey>;
 | |
| VibrationMotorList EnumerateVibrationMotors(std::optional<InputBindingKey> for_device = std::nullopt);
 | |
| 
 | |
| /// Retrieves bindings that match the generic bindings for the specified device.
 | |
| GenericInputBindingMapping GetGenericBindingMapping(std::string_view device);
 | |
| 
 | |
| /// Returns true if the specified input source is enabled.
 | |
| bool IsInputSourceEnabled(const SettingsInterface& si, InputSourceType type);
 | |
| 
 | |
| /// Re-parses the config and registers all hotkey and pad bindings.
 | |
| void ReloadBindings(const SettingsInterface& si, const SettingsInterface& hotkey_binding_si);
 | |
| 
 | |
| /// Re-parses the sources part of the config and initializes any backends.
 | |
| void ReloadSources(const SettingsInterface& si, std::unique_lock<std::mutex>& settings_lock);
 | |
| 
 | |
| /// Called when a device change is triggered by the system (DBT_DEVNODES_CHANGED on Windows).
 | |
| /// Returns true if any device changes are detected.
 | |
| bool ReloadDevices();
 | |
| 
 | |
| /// Shuts down any enabled input sources.
 | |
| void CloseSources();
 | |
| 
 | |
| /// Polls input sources for events (e.g. external controllers).
 | |
| void PollSources();
 | |
| 
 | |
| /// Returns true if any bindings exist for the specified key.
 | |
| /// Can be safely called on another thread.
 | |
| bool HasAnyBindingsForKey(InputBindingKey key);
 | |
| 
 | |
| /// Returns true if any bindings exist for the specified source + index.
 | |
| /// Can be safely called on another thread.
 | |
| bool HasAnyBindingsForSource(InputBindingKey key);
 | |
| 
 | |
| /// Parses a string binding into its components. Use with external AddBinding().
 | |
| bool ParseBindingAndGetSource(std::string_view binding, InputBindingKey* key, InputSource** source);
 | |
| 
 | |
| /// Externally adds a fixed binding. Be sure to call *after* ReloadBindings() otherwise it will be lost.
 | |
| void AddBinding(std::string_view binding, const InputEventHandler& handler);
 | |
| 
 | |
| /// Adds an external vibration binding.
 | |
| void AddVibrationBinding(u32 pad_index, const InputBindingKey* motor_0_binding, InputSource* motor_0_source,
 | |
|                          const InputBindingKey* motor_1_binding, InputSource* motor_1_source);
 | |
| 
 | |
| /// Updates internal state for any binds for this key, and fires callbacks as needed.
 | |
| /// Returns true if anything was bound to this key, otherwise false.
 | |
| bool InvokeEvents(InputBindingKey key, float value, GenericInputBinding generic_key = GenericInputBinding::Unknown);
 | |
| 
 | |
| /// Clears internal state for any binds with a matching source/index.
 | |
| void ClearBindStateFromSource(InputBindingKey key);
 | |
| 
 | |
| /// Sets a hook which can be used to intercept events before they're processed by the normal bindings.
 | |
| /// This is typically used when binding new controls to detect what gets pressed.
 | |
| void SetHook(InputInterceptHook::Callback callback);
 | |
| 
 | |
| /// Removes any currently-active interception hook.
 | |
| void RemoveHook();
 | |
| 
 | |
| /// Returns true if there is an interception hook present.
 | |
| bool HasHook();
 | |
| 
 | |
| /// Internal method used by pads to dispatch vibration updates to input sources.
 | |
| /// Intensity is normalized from 0 to 1.
 | |
| void SetPadVibrationIntensity(u32 pad_index, float large_or_single_motor_intensity, float small_motor_intensity);
 | |
| 
 | |
| /// Zeros all vibration intensities. Call when pausing.
 | |
| /// The pad vibration state will internally remain, so that when emulation is unpaused, the effect resumes.
 | |
| void PauseVibration();
 | |
| 
 | |
| /// Returns the number of currently-connected pointer devices.
 | |
| u32 GetPointerCount();
 | |
| 
 | |
| /// Reads absolute pointer position.
 | |
| std::pair<float, float> GetPointerAbsolutePosition(u32 index);
 | |
| 
 | |
| /// Updates absolute pointer position. Can call from UI thread, use when the host only reports absolute coordinates.
 | |
| void UpdatePointerAbsolutePosition(u32 index, float x, float y);
 | |
| 
 | |
| /// Resets the accumulated pointer movement. Use when pointer tracking was interrupted.
 | |
| void ResetPointerRelativeDelta(u32 index);
 | |
| 
 | |
| /// Updates relative pointer position. Can call from the UI thread, use when host supports relative coordinate
 | |
| /// reporting.
 | |
| void UpdatePointerRelativeDelta(u32 index, InputPointerAxis axis, float d, bool raw_input = false);
 | |
| 
 | |
| /// Updates host mouse mode (relative/cursor hiding).
 | |
| void UpdateRelativeMouseMode();
 | |
| void UpdateHostMouseMode();
 | |
| bool IsRelativeMouseModeActive();
 | |
| 
 | |
| /// Sets the state of the specified macro button.
 | |
| void SetMacroButtonState(u32 pad, u32 index, bool state);
 | |
| 
 | |
| /// Returns true if the raw input source is being used.
 | |
| bool IsUsingRawInput();
 | |
| 
 | |
| /// Updates InputManager's view of the window size, used for clamping raw input coordinates.
 | |
| void SetDisplayWindowSize(float width, float height);
 | |
| std::pair<float, float> GetDisplayWindowSize();
 | |
| 
 | |
| /// Restores default configuration.
 | |
| void SetDefaultSourceConfig(SettingsInterface& si);
 | |
| 
 | |
| /// Clears all bindings for a given port.
 | |
| void ClearPortBindings(SettingsInterface& si, u32 port);
 | |
| 
 | |
| /// Copies pad configuration from one interface (ini) to another.
 | |
| void CopyConfiguration(SettingsInterface* dest_si, const SettingsInterface& src_si, bool copy_pad_config = true,
 | |
|                        bool copy_source_config = true, bool copy_pad_bindings = true, bool copy_hotkey_bindings = true);
 | |
| 
 | |
| /// Performs automatic controller mapping with the provided list of generic mappings.
 | |
| bool MapController(SettingsInterface& si, u32 controller,
 | |
|                    const std::vector<std::pair<GenericInputBinding, std::string>>& mapping,
 | |
|                    bool clear_existing_mappings);
 | |
| 
 | |
| /// Returns the name of the first physical device mapped to the emulated controller, "None", or "Multiple Devices".
 | |
| std::string GetPhysicalDeviceForController(SettingsInterface& si, u32 controller);
 | |
| 
 | |
| /// Returns a list of input profiles available.
 | |
| std::vector<std::string> GetInputProfileNames();
 | |
| 
 | |
| /// Called when a new input device is connected.
 | |
| void OnInputDeviceConnected(InputBindingKey key, std::string_view identifier, std::string_view device_name);
 | |
| 
 | |
| /// Called when an input device is disconnected.
 | |
| void OnInputDeviceDisconnected(InputBindingKey key, std::string_view identifier);
 | |
| 
 | |
| /// Creates a force feedback device interface for the specified source and device.
 | |
| std::unique_ptr<ForceFeedbackDevice> CreateForceFeedbackDevice(const std::string_view device, Error* error = nullptr);
 | |
| } // namespace InputManager
 | |
| 
 | |
| namespace Host {
 | |
| /// Adds any fixed bindings from the host.
 | |
| void AddFixedInputBindings(const SettingsInterface& si);
 | |
| 
 | |
| /// Called when a new input device is connected.
 | |
| void OnInputDeviceConnected(InputBindingKey key, std::string_view identifier, std::string_view device_name);
 | |
| 
 | |
| /// Called when an input device is disconnected.
 | |
| void OnInputDeviceDisconnected(InputBindingKey key, std::string_view identifier);
 | |
| 
 | |
| /// Enables "relative" mouse mode, locking the cursor position and returning relative coordinates.
 | |
| void SetMouseMode(bool relative, bool hide_cursor);
 | |
| } // namespace Host
 |