early-access version 2199
This commit is contained in:
		| @@ -1,7 +1,7 @@ | ||||
| yuzu emulator early access | ||||
| ============= | ||||
|  | ||||
| This is the source code for early-access 2198. | ||||
| This is the source code for early-access 2199. | ||||
|  | ||||
| ## Legal Notice | ||||
|  | ||||
|   | ||||
| @@ -73,7 +73,6 @@ add_library(common STATIC | ||||
|     hex_util.h | ||||
|     host_memory.cpp | ||||
|     host_memory.h | ||||
|     input.h | ||||
|     intrusive_red_black_tree.h | ||||
|     literals.h | ||||
|     logging/backend.cpp | ||||
|   | ||||
| @@ -6,6 +6,7 @@ | ||||
|  | ||||
| #include <algorithm> | ||||
| #include <array> | ||||
| #include <atomic> | ||||
| #include <map> | ||||
| #include <optional> | ||||
| #include <string> | ||||
| @@ -559,15 +560,19 @@ struct Values { | ||||
|     Setting<bool> enable_accurate_vibrations{false, "enable_accurate_vibrations"}; | ||||
|  | ||||
|     Setting<bool> motion_enabled{true, "motion_enabled"}; | ||||
|     BasicSetting<std::string> motion_device{"engine:motion_emu,update_period:100,sensitivity:0.01", | ||||
|                                             "motion_device"}; | ||||
|     BasicSetting<std::string> udp_input_servers{"127.0.0.1:26760", "udp_input_servers"}; | ||||
|  | ||||
|     BasicSetting<bool> pause_tas_on_load{true, "pause_tas_on_load"}; | ||||
|     BasicSetting<bool> tas_enable{false, "tas_enable"}; | ||||
|     BasicSetting<bool> tas_loop{false, "tas_loop"}; | ||||
|     BasicSetting<bool> tas_swap_controllers{true, "tas_swap_controllers"}; | ||||
|  | ||||
|     BasicSetting<bool> mouse_panning{false, "mouse_panning"}; | ||||
|     BasicRangedSetting<u8> mouse_panning_sensitivity{10, 1, 100, "mouse_panning_sensitivity"}; | ||||
|     BasicSetting<bool> mouse_enabled{false, "mouse_enabled"}; | ||||
|     std::string mouse_device; | ||||
|     MouseButtonsRaw mouse_buttons; | ||||
|  | ||||
|     BasicSetting<bool> emulate_analog_keyboard{false, "emulate_analog_keyboard"}; | ||||
| @@ -581,11 +586,14 @@ struct Values { | ||||
|  | ||||
|     TouchscreenInput touchscreen; | ||||
|  | ||||
|     BasicSetting<bool> use_touch_from_button{false, "use_touch_from_button"}; | ||||
|     BasicSetting<std::string> touch_device{"min_x:100,min_y:50,max_x:1800,max_y:850", | ||||
|                                            "touch_device"}; | ||||
|     BasicSetting<int> touch_from_button_map_index{0, "touch_from_button_map"}; | ||||
|     std::vector<TouchFromButtonMap> touch_from_button_maps; | ||||
|  | ||||
|     std::atomic_bool is_device_reload_pending{true}; | ||||
|  | ||||
|     // Data Storage | ||||
|     BasicSetting<bool> use_virtual_sd{true, "use_virtual_sd"}; | ||||
|     BasicSetting<bool> gamecard_inserted{false, "gamecard_inserted"}; | ||||
|   | ||||
| @@ -62,22 +62,11 @@ enum Values : int { | ||||
|  | ||||
| constexpr int STICK_HID_BEGIN = LStick; | ||||
| constexpr int STICK_HID_END = NumAnalogs; | ||||
| constexpr int NUM_STICKS_HID = NumAnalogs; | ||||
|  | ||||
| extern const std::array<const char*, NumAnalogs> mapping; | ||||
| } // namespace NativeAnalog | ||||
|  | ||||
| namespace NativeTrigger { | ||||
| enum Values : int { | ||||
|     LTrigger, | ||||
|     RTrigger, | ||||
|  | ||||
|     NumTriggers, | ||||
| }; | ||||
|  | ||||
| constexpr int TRIGGER_HID_BEGIN = LTrigger; | ||||
| constexpr int TRIGGER_HID_END = NumTriggers; | ||||
| } // namespace NativeTrigger | ||||
|  | ||||
| namespace NativeVibration { | ||||
| enum Values : int { | ||||
|     LeftVibrationDevice, | ||||
| @@ -129,6 +118,7 @@ extern const std::array<const char*, NumMouseButtons> mapping; | ||||
| namespace NativeKeyboard { | ||||
| enum Keys { | ||||
|     None, | ||||
|     Error, | ||||
|  | ||||
|     A = 4, | ||||
|     B, | ||||
| @@ -166,22 +156,22 @@ enum Keys { | ||||
|     N8, | ||||
|     N9, | ||||
|     N0, | ||||
|     Return, | ||||
|     Enter, | ||||
|     Escape, | ||||
|     Backspace, | ||||
|     Tab, | ||||
|     Space, | ||||
|     Minus, | ||||
|     Plus, | ||||
|     OpenBracket, | ||||
|     CloseBracket, | ||||
|     Pipe, | ||||
|     Equal, | ||||
|     LeftBrace, | ||||
|     RightBrace, | ||||
|     Backslash, | ||||
|     Tilde, | ||||
|     Semicolon, | ||||
|     Quote, | ||||
|     Backquote, | ||||
|     Apostrophe, | ||||
|     Grave, | ||||
|     Comma, | ||||
|     Period, | ||||
|     Dot, | ||||
|     Slash, | ||||
|     CapsLockKey, | ||||
|  | ||||
| @@ -198,7 +188,7 @@ enum Keys { | ||||
|     F11, | ||||
|     F12, | ||||
|  | ||||
|     PrintScreen, | ||||
|     SystemRequest, | ||||
|     ScrollLockKey, | ||||
|     Pause, | ||||
|     Insert, | ||||
| @@ -267,18 +257,8 @@ enum Keys { | ||||
|     ScrollLockActive, | ||||
|     KPComma, | ||||
|  | ||||
|     Ro = 0x87, | ||||
|     KatakanaHiragana, | ||||
|     Yen, | ||||
|     Henkan, | ||||
|     Muhenkan, | ||||
|     NumPadCommaPc98, | ||||
|  | ||||
|     HangulEnglish = 0x90, | ||||
|     Hanja, | ||||
|     KatakanaKey, | ||||
|     HiraganaKey, | ||||
|     ZenkakuHankaku, | ||||
|     KPLeftParenthesis, | ||||
|     KPRightParenthesis, | ||||
|  | ||||
|     LeftControlKey = 0xE0, | ||||
|     LeftShiftKey, | ||||
| @@ -327,8 +307,6 @@ enum Modifiers { | ||||
|     CapsLock, | ||||
|     ScrollLock, | ||||
|     NumLock, | ||||
|     Katakana, | ||||
|     Hiragana, | ||||
|  | ||||
|     NumKeyboardMods, | ||||
| }; | ||||
|   | ||||
| @@ -132,23 +132,11 @@ add_library(core STATIC | ||||
|     frontend/emu_window.h | ||||
|     frontend/framebuffer_layout.cpp | ||||
|     frontend/framebuffer_layout.h | ||||
|     frontend/input_interpreter.cpp | ||||
|     frontend/input_interpreter.h | ||||
|     frontend/input.h | ||||
|     hardware_interrupt_manager.cpp | ||||
|     hardware_interrupt_manager.h | ||||
|     hid/emulated_console.cpp | ||||
|     hid/emulated_console.h | ||||
|     hid/emulated_controller.cpp | ||||
|     hid/emulated_controller.h | ||||
|     hid/emulated_devices.cpp | ||||
|     hid/emulated_devices.h | ||||
|     hid/hid_core.cpp | ||||
|     hid/hid_core.h | ||||
|     hid/hid_types.h | ||||
|     hid/input_converter.cpp | ||||
|     hid/input_converter.h | ||||
|     hid/input_interpreter.cpp | ||||
|     hid/input_interpreter.h | ||||
|     hid/motion_input.cpp | ||||
|     hid/motion_input.h | ||||
|     hle/api_version.h | ||||
|     hle/ipc.h | ||||
|     hle/ipc_helpers.h | ||||
| @@ -414,7 +402,6 @@ add_library(core STATIC | ||||
|     hle/service/hid/hid.h | ||||
|     hle/service/hid/irs.cpp | ||||
|     hle/service/hid/irs.h | ||||
|     hle/service/hid/ring_lifo.h | ||||
|     hle/service/hid/xcd.cpp | ||||
|     hle/service/hid/xcd.h | ||||
|     hle/service/hid/errors.h | ||||
|   | ||||
| @@ -27,7 +27,6 @@ | ||||
| #include "core/file_sys/vfs_concat.h" | ||||
| #include "core/file_sys/vfs_real.h" | ||||
| #include "core/hardware_interrupt_manager.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/kernel/k_process.h" | ||||
| #include "core/hle/kernel/k_scheduler.h" | ||||
| #include "core/hle/kernel/kernel.h" | ||||
| @@ -127,7 +126,7 @@ FileSys::VirtualFile GetGameFileFromPath(const FileSys::VirtualFilesystem& vfs, | ||||
|  | ||||
| struct System::Impl { | ||||
|     explicit Impl(System& system) | ||||
|         : kernel{system}, fs_controller{system}, memory{system}, hid_core{}, | ||||
|         : kernel{system}, fs_controller{system}, memory{system}, | ||||
|           cpu_manager{system}, reporter{system}, applet_manager{system}, time_manager{system} {} | ||||
|  | ||||
|     SystemResultStatus Run() { | ||||
| @@ -392,7 +391,6 @@ struct System::Impl { | ||||
|     std::unique_ptr<Hardware::InterruptManager> interrupt_manager; | ||||
|     std::unique_ptr<Core::DeviceMemory> device_memory; | ||||
|     Core::Memory::Memory memory; | ||||
|     Core::HID::HIDCore hid_core; | ||||
|     CpuManager cpu_manager; | ||||
|     std::atomic_bool is_powered_on{}; | ||||
|     bool exit_lock = false; | ||||
| @@ -617,14 +615,6 @@ const Kernel::KernelCore& System::Kernel() const { | ||||
|     return impl->kernel; | ||||
| } | ||||
|  | ||||
| HID::HIDCore& System::HIDCore() { | ||||
|     return impl->hid_core; | ||||
| } | ||||
|  | ||||
| const HID::HIDCore& System::HIDCore() const { | ||||
|     return impl->hid_core; | ||||
| } | ||||
|  | ||||
| Timing::CoreTiming& System::CoreTiming() { | ||||
|     return impl->core_timing; | ||||
| } | ||||
| @@ -835,6 +825,8 @@ void System::ApplySettings() { | ||||
|     if (IsPoweredOn()) { | ||||
|         Renderer().RefreshBaseSettings(); | ||||
|     } | ||||
|  | ||||
|     Service::HID::ReloadInputDevices(); | ||||
| } | ||||
|  | ||||
| } // namespace Core | ||||
|   | ||||
| @@ -89,10 +89,6 @@ namespace Core::Hardware { | ||||
| class InterruptManager; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| class HIDCore; | ||||
| } | ||||
|  | ||||
| namespace Core { | ||||
|  | ||||
| class ARM_Interface; | ||||
| @@ -289,12 +285,6 @@ public: | ||||
|     /// Provides a constant reference to the kernel instance. | ||||
|     [[nodiscard]] const Kernel::KernelCore& Kernel() const; | ||||
|  | ||||
|     /// Gets a mutable reference to the HID interface. | ||||
|     [[nodiscard]] HID::HIDCore& HIDCore(); | ||||
|  | ||||
|     /// Gets an immutable reference to the HID interface. | ||||
|     [[nodiscard]] const HID::HIDCore& HIDCore() const; | ||||
|  | ||||
|     /// Provides a reference to the internal PerfStats instance. | ||||
|     [[nodiscard]] Core::PerfStats& GetPerfStats(); | ||||
|  | ||||
|   | ||||
| @@ -5,15 +5,16 @@ | ||||
| #include "common/assert.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "core/frontend/applets/controller.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hle/service/hid/controllers/npad.h" | ||||
| #include "core/hle/service/hid/hid.h" | ||||
| #include "core/hle/service/sm/sm.h" | ||||
|  | ||||
| namespace Core::Frontend { | ||||
|  | ||||
| ControllerApplet::~ControllerApplet() = default; | ||||
|  | ||||
| DefaultControllerApplet::DefaultControllerApplet(HID::HIDCore& hid_core_) : hid_core{hid_core_} {} | ||||
| DefaultControllerApplet::DefaultControllerApplet(Service::SM::ServiceManager& service_manager_) | ||||
|     : service_manager{service_manager_} {} | ||||
|  | ||||
| DefaultControllerApplet::~DefaultControllerApplet() = default; | ||||
|  | ||||
| @@ -21,20 +22,24 @@ void DefaultControllerApplet::ReconfigureControllers(std::function<void()> callb | ||||
|                                                      const ControllerParameters& parameters) const { | ||||
|     LOG_INFO(Service_HID, "called, deducing the best configuration based on the given parameters!"); | ||||
|  | ||||
|     auto& npad = | ||||
|         service_manager.GetService<Service::HID::Hid>("hid") | ||||
|             ->GetAppletResource() | ||||
|             ->GetController<Service::HID::Controller_NPad>(Service::HID::HidController::NPad); | ||||
|  | ||||
|     auto& players = Settings::values.players.GetValue(); | ||||
|  | ||||
|     const std::size_t min_supported_players = | ||||
|         parameters.enable_single_mode ? 1 : parameters.min_players; | ||||
|  | ||||
|     // Disconnect Handheld first. | ||||
|     auto* handheld = hid_core.GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|     handheld->Disconnect(); | ||||
|     npad.DisconnectNpadAtIndex(8); | ||||
|  | ||||
|     // Deduce the best configuration based on the input parameters. | ||||
|     for (std::size_t index = 0; index < hid_core.available_controllers - 2; ++index) { | ||||
|         auto* controller = hid_core.GetEmulatedControllerByIndex(index); | ||||
|  | ||||
|     for (std::size_t index = 0; index < players.size() - 2; ++index) { | ||||
|         // First, disconnect all controllers regardless of the value of keep_controllers_connected. | ||||
|         // This makes it easy to connect the desired controllers. | ||||
|         controller->Disconnect(); | ||||
|         npad.DisconnectNpadAtIndex(index); | ||||
|  | ||||
|         // Only connect the minimum number of required players. | ||||
|         if (index >= min_supported_players) { | ||||
| @@ -44,27 +49,27 @@ void DefaultControllerApplet::ReconfigureControllers(std::function<void()> callb | ||||
|         // Connect controllers based on the following priority list from highest to lowest priority: | ||||
|         // Pro Controller -> Dual Joycons -> Left Joycon/Right Joycon -> Handheld | ||||
|         if (parameters.allow_pro_controller) { | ||||
|             controller->SetNpadStyleIndex(Core::HID::NpadStyleIndex::ProController); | ||||
|             controller->Connect(); | ||||
|             npad.AddNewControllerAt( | ||||
|                 npad.MapSettingsTypeToNPad(Settings::ControllerType::ProController), index); | ||||
|         } else if (parameters.allow_dual_joycons) { | ||||
|             controller->SetNpadStyleIndex(Core::HID::NpadStyleIndex::JoyconDual); | ||||
|             controller->Connect(); | ||||
|             npad.AddNewControllerAt( | ||||
|                 npad.MapSettingsTypeToNPad(Settings::ControllerType::DualJoyconDetached), index); | ||||
|         } else if (parameters.allow_left_joycon && parameters.allow_right_joycon) { | ||||
|             // Assign left joycons to even player indices and right joycons to odd player indices. | ||||
|             // We do this since Captain Toad Treasure Tracker expects a left joycon for Player 1 and | ||||
|             // a right Joycon for Player 2 in 2 Player Assist mode. | ||||
|             if (index % 2 == 0) { | ||||
|                 controller->SetNpadStyleIndex(Core::HID::NpadStyleIndex::JoyconLeft); | ||||
|                 controller->Connect(); | ||||
|                 npad.AddNewControllerAt( | ||||
|                     npad.MapSettingsTypeToNPad(Settings::ControllerType::LeftJoycon), index); | ||||
|             } else { | ||||
|                 controller->SetNpadStyleIndex(Core::HID::NpadStyleIndex::JoyconRight); | ||||
|                 controller->Connect(); | ||||
|                 npad.AddNewControllerAt( | ||||
|                     npad.MapSettingsTypeToNPad(Settings::ControllerType::RightJoycon), index); | ||||
|             } | ||||
|         } else if (index == 0 && parameters.enable_single_mode && parameters.allow_handheld && | ||||
|                    !Settings::values.use_docked_mode.GetValue()) { | ||||
|             // We should *never* reach here under any normal circumstances. | ||||
|             controller->SetNpadStyleIndex(Core::HID::NpadStyleIndex::Handheld); | ||||
|             controller->Connect(); | ||||
|             npad.AddNewControllerAt(npad.MapSettingsTypeToNPad(Settings::ControllerType::Handheld), | ||||
|                                     index); | ||||
|         } else { | ||||
|             UNREACHABLE_MSG("Unable to add a new controller based on the given parameters!"); | ||||
|         } | ||||
|   | ||||
| @@ -8,8 +8,8 @@ | ||||
|  | ||||
| #include "common/common_types.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class HIDCore; | ||||
| namespace Service::SM { | ||||
| class ServiceManager; | ||||
| } | ||||
|  | ||||
| namespace Core::Frontend { | ||||
| @@ -44,14 +44,14 @@ public: | ||||
|  | ||||
| class DefaultControllerApplet final : public ControllerApplet { | ||||
| public: | ||||
|     explicit DefaultControllerApplet(HID::HIDCore& hid_core_); | ||||
|     explicit DefaultControllerApplet(Service::SM::ServiceManager& service_manager_); | ||||
|     ~DefaultControllerApplet() override; | ||||
|  | ||||
|     void ReconfigureControllers(std::function<void()> callback, | ||||
|                                 const ControllerParameters& parameters) const override; | ||||
|  | ||||
| private: | ||||
|     HID::HIDCore& hid_core; | ||||
|     Service::SM::ServiceManager& service_manager; | ||||
| }; | ||||
|  | ||||
| } // namespace Core::Frontend | ||||
|   | ||||
| @@ -3,31 +3,66 @@ | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include <mutex> | ||||
| #include "common/settings.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "core/frontend/input.h" | ||||
|  | ||||
| namespace Core::Frontend { | ||||
|  | ||||
| GraphicsContext::~GraphicsContext() = default; | ||||
|  | ||||
| class EmuWindow::TouchState : public Input::Factory<Input::TouchDevice>, | ||||
|                               public std::enable_shared_from_this<TouchState> { | ||||
| public: | ||||
|     std::unique_ptr<Input::TouchDevice> Create(const Common::ParamPackage&) override { | ||||
|         return std::make_unique<Device>(shared_from_this()); | ||||
|     } | ||||
|  | ||||
|     std::mutex mutex; | ||||
|  | ||||
|     Input::TouchStatus status; | ||||
|  | ||||
| private: | ||||
|     class Device : public Input::TouchDevice { | ||||
|     public: | ||||
|         explicit Device(std::weak_ptr<TouchState>&& touch_state_) : touch_state(touch_state_) {} | ||||
|         Input::TouchStatus GetStatus() const override { | ||||
|             if (auto state = touch_state.lock()) { | ||||
|                 std::lock_guard guard{state->mutex}; | ||||
|                 return state->status; | ||||
|             } | ||||
|             return {}; | ||||
|         } | ||||
|  | ||||
|     private: | ||||
|         std::weak_ptr<TouchState> touch_state; | ||||
|     }; | ||||
| }; | ||||
|  | ||||
| EmuWindow::EmuWindow() { | ||||
|     // TODO: Find a better place to set this. | ||||
|     config.min_client_area_size = | ||||
|         std::make_pair(Layout::MinimumSize::Width, Layout::MinimumSize::Height); | ||||
|     active_config = config; | ||||
|     touch_state = std::make_shared<TouchState>(); | ||||
|     Input::RegisterFactory<Input::TouchDevice>("emu_window", touch_state); | ||||
| } | ||||
|  | ||||
| EmuWindow::~EmuWindow() {} | ||||
| EmuWindow::~EmuWindow() { | ||||
|     Input::UnregisterFactory<Input::TouchDevice>("emu_window"); | ||||
| } | ||||
|  | ||||
| std::pair<f32, f32> EmuWindow::MapToTouchScreen(u32 framebuffer_x, u32 framebuffer_y) const { | ||||
|     std::tie(framebuffer_x, framebuffer_y) = ClipToTouchScreen(framebuffer_x, framebuffer_y); | ||||
|     const float x = | ||||
|         static_cast<float>(framebuffer_x - framebuffer_layout.screen.left) / | ||||
|         static_cast<float>(framebuffer_layout.screen.right - framebuffer_layout.screen.left); | ||||
|     const float y = | ||||
|         static_cast<float>(framebuffer_y - framebuffer_layout.screen.top) / | ||||
|         static_cast<float>(framebuffer_layout.screen.bottom - framebuffer_layout.screen.top); | ||||
|  | ||||
|     return std::make_pair(x, y); | ||||
| /** | ||||
|  * Check if the given x/y coordinates are within the touchpad specified by the framebuffer layout | ||||
|  * @param layout FramebufferLayout object describing the framebuffer size and screen positions | ||||
|  * @param framebuffer_x Framebuffer x-coordinate to check | ||||
|  * @param framebuffer_y Framebuffer y-coordinate to check | ||||
|  * @return True if the coordinates are within the touchpad, otherwise false | ||||
|  */ | ||||
| static bool IsWithinTouchscreen(const Layout::FramebufferLayout& layout, u32 framebuffer_x, | ||||
|                                 u32 framebuffer_y) { | ||||
|     return (framebuffer_y >= layout.screen.top && framebuffer_y < layout.screen.bottom && | ||||
|             framebuffer_x >= layout.screen.left && framebuffer_x < layout.screen.right); | ||||
| } | ||||
|  | ||||
| std::pair<u32, u32> EmuWindow::ClipToTouchScreen(u32 new_x, u32 new_y) const { | ||||
| @@ -40,6 +75,49 @@ std::pair<u32, u32> EmuWindow::ClipToTouchScreen(u32 new_x, u32 new_y) const { | ||||
|     return std::make_pair(new_x, new_y); | ||||
| } | ||||
|  | ||||
| void EmuWindow::TouchPressed(u32 framebuffer_x, u32 framebuffer_y, size_t id) { | ||||
|     if (!IsWithinTouchscreen(framebuffer_layout, framebuffer_x, framebuffer_y)) { | ||||
|         return; | ||||
|     } | ||||
|     if (id >= touch_state->status.size()) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     std::lock_guard guard{touch_state->mutex}; | ||||
|     const float x = | ||||
|         static_cast<float>(framebuffer_x - framebuffer_layout.screen.left) / | ||||
|         static_cast<float>(framebuffer_layout.screen.right - framebuffer_layout.screen.left); | ||||
|     const float y = | ||||
|         static_cast<float>(framebuffer_y - framebuffer_layout.screen.top) / | ||||
|         static_cast<float>(framebuffer_layout.screen.bottom - framebuffer_layout.screen.top); | ||||
|  | ||||
|     touch_state->status[id] = std::make_tuple(x, y, true); | ||||
| } | ||||
|  | ||||
| void EmuWindow::TouchReleased(size_t id) { | ||||
|     if (id >= touch_state->status.size()) { | ||||
|         return; | ||||
|     } | ||||
|     std::lock_guard guard{touch_state->mutex}; | ||||
|     touch_state->status[id] = std::make_tuple(0.0f, 0.0f, false); | ||||
| } | ||||
|  | ||||
| void EmuWindow::TouchMoved(u32 framebuffer_x, u32 framebuffer_y, size_t id) { | ||||
|     if (id >= touch_state->status.size()) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!std::get<2>(touch_state->status[id])) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     if (!IsWithinTouchscreen(framebuffer_layout, framebuffer_x, framebuffer_y)) { | ||||
|         std::tie(framebuffer_x, framebuffer_y) = ClipToTouchScreen(framebuffer_x, framebuffer_y); | ||||
|     } | ||||
|  | ||||
|     TouchPressed(framebuffer_x, framebuffer_y, id); | ||||
| } | ||||
|  | ||||
| void EmuWindow::UpdateCurrentFramebufferLayout(u32 width, u32 height) { | ||||
|     NotifyFramebufferLayoutChanged(Layout::DefaultFrameLayout(width, height)); | ||||
| } | ||||
|   | ||||
| @@ -112,6 +112,28 @@ public: | ||||
|     /// Returns if window is shown (not minimized) | ||||
|     virtual bool IsShown() const = 0; | ||||
|  | ||||
|     /** | ||||
|      * Signal that a touch pressed event has occurred (e.g. mouse click pressed) | ||||
|      * @param framebuffer_x Framebuffer x-coordinate that was pressed | ||||
|      * @param framebuffer_y Framebuffer y-coordinate that was pressed | ||||
|      * @param id Touch event ID | ||||
|      */ | ||||
|     void TouchPressed(u32 framebuffer_x, u32 framebuffer_y, size_t id); | ||||
|  | ||||
|     /** | ||||
|      * Signal that a touch released event has occurred (e.g. mouse click released) | ||||
|      * @param id Touch event ID | ||||
|      */ | ||||
|     void TouchReleased(size_t id); | ||||
|  | ||||
|     /** | ||||
|      * Signal that a touch movement event has occurred (e.g. mouse was moved over the emu window) | ||||
|      * @param framebuffer_x Framebuffer x-coordinate | ||||
|      * @param framebuffer_y Framebuffer y-coordinate | ||||
|      * @param id Touch event ID | ||||
|      */ | ||||
|     void TouchMoved(u32 framebuffer_x, u32 framebuffer_y, size_t id); | ||||
|  | ||||
|     /** | ||||
|      * Returns currently active configuration. | ||||
|      * @note Accesses to the returned object need not be consistent because it may be modified in | ||||
| @@ -190,11 +212,6 @@ protected: | ||||
|         client_area_height = size.second; | ||||
|     } | ||||
|  | ||||
|     /** | ||||
|      * Converts a screen postion into the equivalent touchscreen position. | ||||
|      */ | ||||
|     std::pair<f32, f32> MapToTouchScreen(u32 framebuffer_x, u32 framebuffer_y) const; | ||||
|  | ||||
|     WindowSystemInfo window_info; | ||||
|  | ||||
| private: | ||||
| @@ -220,6 +237,9 @@ private: | ||||
|     WindowConfig config;        ///< Internal configuration (changes pending for being applied in | ||||
|                                 /// ProcessConfigurationChanges) | ||||
|     WindowConfig active_config; ///< Internal active configuration | ||||
|  | ||||
|     class TouchState; | ||||
|     std::shared_ptr<TouchState> touch_state; | ||||
| }; | ||||
|  | ||||
| } // namespace Core::Frontend | ||||
|   | ||||
| @@ -10,9 +10,6 @@ | ||||
| #include "common/string_util.h" | ||||
| #include "core/core.h" | ||||
| #include "core/frontend/applets/controller.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hle/result.h" | ||||
| #include "core/hle/service/am/am.h" | ||||
| #include "core/hle/service/am/applets/applet_controller.h" | ||||
| @@ -28,7 +25,7 @@ namespace Service::AM::Applets { | ||||
| static Core::Frontend::ControllerParameters ConvertToFrontendParameters( | ||||
|     ControllerSupportArgPrivate private_arg, ControllerSupportArgHeader header, bool enable_text, | ||||
|     std::vector<IdentificationColor> identification_colors, std::vector<ExplainText> text) { | ||||
|     Core::HID::NpadStyleTag npad_style_set; | ||||
|     HID::Controller_NPad::NpadStyleSet npad_style_set; | ||||
|     npad_style_set.raw = private_arg.style_set; | ||||
|  | ||||
|     return { | ||||
| @@ -246,11 +243,19 @@ void Controller::Execute() { | ||||
| void Controller::ConfigurationComplete() { | ||||
|     ControllerSupportResultInfo result_info{}; | ||||
|  | ||||
|     const auto& players = Settings::values.players.GetValue(); | ||||
|  | ||||
|     // If enable_single_mode is enabled, player_count is 1 regardless of any other parameters. | ||||
|     // Otherwise, only count connected players from P1-P8. | ||||
|     result_info.player_count = is_single_mode ? 1 : system.HIDCore().GetPlayerCount(); | ||||
|     result_info.player_count = | ||||
|         is_single_mode | ||||
|             ? 1 | ||||
|             : static_cast<s8>(std::count_if(players.begin(), players.end() - 2, | ||||
|                                             [](const auto& player) { return player.connected; })); | ||||
|  | ||||
|     result_info.selected_id = static_cast<u32>(system.HIDCore().GetFirstNpadId()); | ||||
|     result_info.selected_id = HID::Controller_NPad::IndexToNPad(std::distance( | ||||
|         players.begin(), std::find_if(players.begin(), players.end(), | ||||
|                                       [](const auto& player) { return player.connected; }))); | ||||
|  | ||||
|     result_info.result = 0; | ||||
|  | ||||
|   | ||||
| @@ -16,10 +16,6 @@ namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| enum class NpadStyleSet : u32; | ||||
| } | ||||
|  | ||||
| namespace Service::AM::Applets { | ||||
|  | ||||
| using IdentificationColor = std::array<u8, 4>; | ||||
| @@ -56,7 +52,7 @@ struct ControllerSupportArgPrivate { | ||||
|     bool flag_1{}; | ||||
|     ControllerSupportMode mode{}; | ||||
|     ControllerSupportCaller caller{}; | ||||
|     Core::HID::NpadStyleSet style_set{}; | ||||
|     u32 style_set{}; | ||||
|     u32 joy_hold_type{}; | ||||
| }; | ||||
| static_assert(sizeof(ControllerSupportArgPrivate) == 0x14, | ||||
|   | ||||
| @@ -231,7 +231,7 @@ void AppletManager::SetDefaultAppletFrontendSet() { | ||||
| void AppletManager::SetDefaultAppletsIfMissing() { | ||||
|     if (frontend.controller == nullptr) { | ||||
|         frontend.controller = | ||||
|             std::make_unique<Core::Frontend::DefaultControllerApplet>(system.HIDCore()); | ||||
|             std::make_unique<Core::Frontend::DefaultControllerApplet>(system.ServiceManager()); | ||||
|     } | ||||
|  | ||||
|     if (frontend.error == nullptr) { | ||||
|   | ||||
| @@ -4,18 +4,13 @@ | ||||
|  | ||||
| #include "common/settings.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hid/emulated_console.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/hid/controllers/console_sixaxis.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| constexpr std::size_t SHARED_MEMORY_OFFSET = 0x3C200; | ||||
|  | ||||
| Controller_ConsoleSixAxis::Controller_ConsoleSixAxis(Core::HID::HIDCore& hid_core_) | ||||
|     : ControllerBase{hid_core_} { | ||||
|     console = hid_core.GetEmulatedConsole(); | ||||
| } | ||||
|  | ||||
| Controller_ConsoleSixAxis::Controller_ConsoleSixAxis(Core::System& system_) | ||||
|     : ControllerBase{system_} {} | ||||
| Controller_ConsoleSixAxis::~Controller_ConsoleSixAxis() = default; | ||||
|  | ||||
| void Controller_ConsoleSixAxis::OnInit() {} | ||||
| @@ -43,21 +38,25 @@ void Controller_ConsoleSixAxis::OnUpdate(const Core::Timing::CoreTiming& core_ti | ||||
|     cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|  | ||||
|     // Try to read sixaxis sensor states | ||||
|     const auto motion_status = console->GetMotion(); | ||||
|     MotionDevice motion_device{}; | ||||
|     const auto& device = motions[0]; | ||||
|     if (device) { | ||||
|         std::tie(motion_device.accel, motion_device.gyro, motion_device.rotation, | ||||
|                  motion_device.orientation, motion_device.quaternion) = device->GetStatus(); | ||||
|         console_six_axis.is_seven_six_axis_sensor_at_rest = motion_device.gyro.Length2() < 0.0001f; | ||||
|     } | ||||
|  | ||||
|     console_six_axis.is_seven_six_axis_sensor_at_rest = motion_status.is_at_rest; | ||||
|  | ||||
|     cur_entry.accel = motion_status.accel; | ||||
|     cur_entry.accel = motion_device.accel; | ||||
|     // Zero gyro values as they just mess up with the camera | ||||
|     // Note: Probably a correct sensivity setting must be set | ||||
|     cur_entry.gyro = {}; | ||||
|     cur_entry.quaternion = { | ||||
|         { | ||||
|             motion_status.quaternion.xyz.y, | ||||
|             motion_status.quaternion.xyz.x, | ||||
|             -motion_status.quaternion.w, | ||||
|             motion_device.quaternion.xyz.y, | ||||
|             motion_device.quaternion.xyz.x, | ||||
|             -motion_device.quaternion.w, | ||||
|         }, | ||||
|         -motion_status.quaternion.xyz.z, | ||||
|         -motion_device.quaternion.xyz.z, | ||||
|     }; | ||||
|  | ||||
|     console_six_axis.sampling_number++; | ||||
| @@ -71,6 +70,13 @@ void Controller_ConsoleSixAxis::OnUpdate(const Core::Timing::CoreTiming& core_ti | ||||
|     std::memcpy(transfer_memory, &seven_six_axis, sizeof(seven_six_axis)); | ||||
| } | ||||
|  | ||||
| void Controller_ConsoleSixAxis::OnLoadInputDevices() { | ||||
|     const auto player = Settings::values.players.GetValue()[0]; | ||||
|     std::transform(player.motions.begin() + Settings::NativeMotion::MOTION_HID_BEGIN, | ||||
|                    player.motions.begin() + Settings::NativeMotion::MOTION_HID_END, motions.begin(), | ||||
|                    Input::CreateDevice<Input::MotionDevice>); | ||||
| } | ||||
|  | ||||
| void Controller_ConsoleSixAxis::SetTransferMemoryPointer(u8* t_mem) { | ||||
|     is_transfer_memory_set = true; | ||||
|     transfer_memory = t_mem; | ||||
|   | ||||
| @@ -5,20 +5,16 @@ | ||||
| #pragma once | ||||
|  | ||||
| #include <array> | ||||
|  | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/quaternion.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class EmulatedConsole; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_ConsoleSixAxis final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_ConsoleSixAxis(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_ConsoleSixAxis(Core::System& system_); | ||||
|     ~Controller_ConsoleSixAxis() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -30,6 +26,9 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
|     // Called on InitializeSevenSixAxisSensor | ||||
|     void SetTransferMemoryPointer(u8* t_mem); | ||||
|  | ||||
| @@ -39,8 +38,8 @@ public: | ||||
| private: | ||||
|     struct SevenSixAxisState { | ||||
|         INSERT_PADDING_WORDS(4); // unused | ||||
|         s64 sampling_number{}; | ||||
|         s64 sampling_number2{}; | ||||
|         s64_le sampling_number{}; | ||||
|         s64_le sampling_number2{}; | ||||
|         u64 unknown{}; | ||||
|         Common::Vec3f accel{}; | ||||
|         Common::Vec3f gyro{}; | ||||
| @@ -48,24 +47,14 @@ private: | ||||
|     }; | ||||
|     static_assert(sizeof(SevenSixAxisState) == 0x50, "SevenSixAxisState is an invalid size"); | ||||
|  | ||||
|     struct CommonHeader { | ||||
|         s64 timestamp; | ||||
|         s64 total_entry_count; | ||||
|         s64 last_entry_index; | ||||
|         s64 entry_count; | ||||
|     }; | ||||
|     static_assert(sizeof(CommonHeader) == 0x20, "CommonHeader is an invalid size"); | ||||
|  | ||||
|     // TODO(german77): SevenSixAxisMemory doesn't follow the standard lifo. Investigate | ||||
|     struct SevenSixAxisMemory { | ||||
|         CommonHeader header{}; | ||||
|         std::array<SevenSixAxisState, 0x21> sevensixaxis_states{}; | ||||
|     }; | ||||
|     static_assert(sizeof(SevenSixAxisMemory) == 0xA70, "SevenSixAxisMemory is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::ConsoleSixAxisSensorSharedMemoryFormat | ||||
|     struct ConsoleSharedMemory { | ||||
|         u64 sampling_number{}; | ||||
|         u64_le sampling_number{}; | ||||
|         bool is_seven_six_axis_sensor_at_rest{}; | ||||
|         f32 verticalization_error{}; | ||||
|         Common::Vec3f gyro_bias{}; | ||||
| @@ -80,7 +69,9 @@ private: | ||||
|         Common::Quaternion<f32> quaternion; | ||||
|     }; | ||||
|  | ||||
|     Core::HID::EmulatedConsole* console; | ||||
|     using MotionArray = | ||||
|         std::array<std::unique_ptr<Input::MotionDevice>, Settings::NativeMotion::NUM_MOTIONS_HID>; | ||||
|     MotionArray motions; | ||||
|     u8* transfer_memory = nullptr; | ||||
|     bool is_transfer_memory_set = false; | ||||
|     ConsoleSharedMemory console_six_axis{}; | ||||
|   | ||||
| @@ -6,12 +6,12 @@ | ||||
|  | ||||
| namespace Service::HID { | ||||
|  | ||||
| ControllerBase::ControllerBase(Core::HID::HIDCore& hid_core_) : hid_core(hid_core_) {} | ||||
| ControllerBase::ControllerBase(Core::System& system_) : system(system_) {} | ||||
| ControllerBase::~ControllerBase() = default; | ||||
|  | ||||
| void ControllerBase::ActivateController() { | ||||
|     if (is_activated) { | ||||
|         return; | ||||
|         OnRelease(); | ||||
|     } | ||||
|     is_activated = true; | ||||
|     OnInit(); | ||||
|   | ||||
| @@ -11,14 +11,14 @@ namespace Core::Timing { | ||||
| class CoreTiming; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| class HIDCore; | ||||
| namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace Service::HID { | ||||
| class ControllerBase { | ||||
| public: | ||||
|     explicit ControllerBase(Core::HID::HIDCore& hid_core_); | ||||
|     explicit ControllerBase(Core::System& system_); | ||||
|     virtual ~ControllerBase(); | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -35,6 +35,9 @@ public: | ||||
|     virtual void OnMotionUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                 std::size_t size) {} | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     virtual void OnLoadInputDevices() = 0; | ||||
|  | ||||
|     void ActivateController(); | ||||
|  | ||||
|     void DeactivateController(); | ||||
| @@ -44,6 +47,14 @@ public: | ||||
| protected: | ||||
|     bool is_activated{false}; | ||||
|  | ||||
|     Core::HID::HIDCore& hid_core; | ||||
|     struct CommonHeader { | ||||
|         s64_le timestamp; | ||||
|         s64_le total_entry_count; | ||||
|         s64_le last_entry_index; | ||||
|         s64_le entry_count; | ||||
|     }; | ||||
|     static_assert(sizeof(CommonHeader) == 0x20, "CommonHeader is an invalid size"); | ||||
|  | ||||
|     Core::System& system; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -6,19 +6,15 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hle/service/hid/controllers/debug_pad.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| constexpr std::size_t SHARED_MEMORY_OFFSET = 0x00000; | ||||
|  | ||||
| Controller_DebugPad::Controller_DebugPad(Core::HID::HIDCore& hid_core_) | ||||
|     : ControllerBase{hid_core_} { | ||||
|     controller = hid_core.GetEmulatedController(Core::HID::NpadIdType::Other); | ||||
| } | ||||
| constexpr s32 HID_JOYSTICK_MAX = 0x7fff; | ||||
| [[maybe_unused]] constexpr s32 HID_JOYSTICK_MIN = -0x7fff; | ||||
| enum class JoystickId : std::size_t { Joystick_Left, Joystick_Right }; | ||||
|  | ||||
| Controller_DebugPad::Controller_DebugPad(Core::System& system_) : ControllerBase{system_} {} | ||||
| Controller_DebugPad::~Controller_DebugPad() = default; | ||||
|  | ||||
| void Controller_DebugPad::OnInit() {} | ||||
| @@ -27,29 +23,63 @@ void Controller_DebugPad::OnRelease() {} | ||||
|  | ||||
| void Controller_DebugPad::OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                    std::size_t size) { | ||||
|     shared_memory.header.timestamp = core_timing.GetCPUTicks(); | ||||
|     shared_memory.header.total_entry_count = 17; | ||||
|  | ||||
|     if (!IsControllerActivated()) { | ||||
|         debug_pad_lifo.buffer_count = 0; | ||||
|         debug_pad_lifo.buffer_tail = 0; | ||||
|         std::memcpy(data + SHARED_MEMORY_OFFSET, &debug_pad_lifo, sizeof(debug_pad_lifo)); | ||||
|         shared_memory.header.entry_count = 0; | ||||
|         shared_memory.header.last_entry_index = 0; | ||||
|         return; | ||||
|     } | ||||
|     shared_memory.header.entry_count = 16; | ||||
|  | ||||
|     const auto& last_entry = debug_pad_lifo.ReadCurrentEntry().state; | ||||
|     next_state.sampling_number = last_entry.sampling_number + 1; | ||||
|     const auto& last_entry = shared_memory.pad_states[shared_memory.header.last_entry_index]; | ||||
|     shared_memory.header.last_entry_index = (shared_memory.header.last_entry_index + 1) % 17; | ||||
|     auto& cur_entry = shared_memory.pad_states[shared_memory.header.last_entry_index]; | ||||
|  | ||||
|     cur_entry.sampling_number = last_entry.sampling_number + 1; | ||||
|     cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|  | ||||
|     if (Settings::values.debug_pad_enabled) { | ||||
|         next_state.attribute.connected.Assign(1); | ||||
|         cur_entry.attribute.connected.Assign(1); | ||||
|         auto& pad = cur_entry.pad_state; | ||||
|  | ||||
|         const auto& button_state = controller->GetDebugPadButtons(); | ||||
|         const auto& stick_state = controller->GetSticks(); | ||||
|         using namespace Settings::NativeButton; | ||||
|         pad.a.Assign(buttons[A - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.b.Assign(buttons[B - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.x.Assign(buttons[X - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.y.Assign(buttons[Y - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.l.Assign(buttons[L - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.r.Assign(buttons[R - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.zl.Assign(buttons[ZL - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.zr.Assign(buttons[ZR - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.plus.Assign(buttons[Plus - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.minus.Assign(buttons[Minus - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.d_left.Assign(buttons[DLeft - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.d_up.Assign(buttons[DUp - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.d_right.Assign(buttons[DRight - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|         pad.d_down.Assign(buttons[DDown - BUTTON_HID_BEGIN]->GetStatus()); | ||||
|  | ||||
|         next_state.pad_state = button_state; | ||||
|         next_state.l_stick = stick_state.left; | ||||
|         next_state.r_stick = stick_state.right; | ||||
|         const auto [stick_l_x_f, stick_l_y_f] = | ||||
|             analogs[static_cast<std::size_t>(JoystickId::Joystick_Left)]->GetStatus(); | ||||
|         const auto [stick_r_x_f, stick_r_y_f] = | ||||
|             analogs[static_cast<std::size_t>(JoystickId::Joystick_Right)]->GetStatus(); | ||||
|         cur_entry.l_stick.x = static_cast<s32>(stick_l_x_f * HID_JOYSTICK_MAX); | ||||
|         cur_entry.l_stick.y = static_cast<s32>(stick_l_y_f * HID_JOYSTICK_MAX); | ||||
|         cur_entry.r_stick.x = static_cast<s32>(stick_r_x_f * HID_JOYSTICK_MAX); | ||||
|         cur_entry.r_stick.y = static_cast<s32>(stick_r_y_f * HID_JOYSTICK_MAX); | ||||
|     } | ||||
|  | ||||
|     debug_pad_lifo.WriteNextEntry(next_state); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &debug_pad_lifo, sizeof(debug_pad_lifo)); | ||||
|     std::memcpy(data, &shared_memory, sizeof(SharedMemory)); | ||||
| } | ||||
|  | ||||
| void Controller_DebugPad::OnLoadInputDevices() { | ||||
|     std::transform(Settings::values.debug_pad_buttons.begin(), | ||||
|                    Settings::values.debug_pad_buttons.begin() + | ||||
|                        Settings::NativeButton::NUM_BUTTONS_HID, | ||||
|                    buttons.begin(), Input::CreateDevice<Input::ButtonDevice>); | ||||
|     std::transform(Settings::values.debug_pad_analogs.begin(), | ||||
|                    Settings::values.debug_pad_analogs.end(), analogs.begin(), | ||||
|                    Input::CreateDevice<Input::AnalogDevice>); | ||||
| } | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -8,20 +8,15 @@ | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "common/swap.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class EmulatedController; | ||||
| struct DebugPadButton; | ||||
| struct AnalogStickState; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_DebugPad final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_DebugPad(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_DebugPad(Core::System& system_); | ||||
|     ~Controller_DebugPad() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -33,31 +28,66 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
| private: | ||||
|     // This is nn::hid::DebugPadAttribute | ||||
|     struct DebugPadAttribute { | ||||
|     struct AnalogStick { | ||||
|         s32_le x; | ||||
|         s32_le y; | ||||
|     }; | ||||
|     static_assert(sizeof(AnalogStick) == 0x8); | ||||
|  | ||||
|     struct PadState { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> a; | ||||
|             BitField<1, 1, u32> b; | ||||
|             BitField<2, 1, u32> x; | ||||
|             BitField<3, 1, u32> y; | ||||
|             BitField<4, 1, u32> l; | ||||
|             BitField<5, 1, u32> r; | ||||
|             BitField<6, 1, u32> zl; | ||||
|             BitField<7, 1, u32> zr; | ||||
|             BitField<8, 1, u32> plus; | ||||
|             BitField<9, 1, u32> minus; | ||||
|             BitField<10, 1, u32> d_left; | ||||
|             BitField<11, 1, u32> d_up; | ||||
|             BitField<12, 1, u32> d_right; | ||||
|             BitField<13, 1, u32> d_down; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(PadState) == 0x4, "PadState is an invalid size"); | ||||
|  | ||||
|     struct Attributes { | ||||
|         union { | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> connected; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(DebugPadAttribute) == 0x4, "DebugPadAttribute is an invalid size"); | ||||
|     static_assert(sizeof(Attributes) == 0x4, "Attributes is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::DebugPadState | ||||
|     struct DebugPadState { | ||||
|         s64 sampling_number; | ||||
|         DebugPadAttribute attribute; | ||||
|         Core::HID::DebugPadButton pad_state; | ||||
|         Core::HID::AnalogStickState r_stick; | ||||
|         Core::HID::AnalogStickState l_stick; | ||||
|     struct PadStates { | ||||
|         s64_le sampling_number; | ||||
|         s64_le sampling_number2; | ||||
|         Attributes attribute; | ||||
|         PadState pad_state; | ||||
|         AnalogStick r_stick; | ||||
|         AnalogStick l_stick; | ||||
|     }; | ||||
|     static_assert(sizeof(DebugPadState) == 0x20, "DebugPadState is an invalid state"); | ||||
|     static_assert(sizeof(PadStates) == 0x28, "PadStates is an invalid state"); | ||||
|  | ||||
|     // This is nn::hid::detail::DebugPadLifo | ||||
|     Lifo<DebugPadState> debug_pad_lifo{}; | ||||
|     static_assert(sizeof(debug_pad_lifo) == 0x2C8, "debug_pad_lifo is an invalid size"); | ||||
|     DebugPadState next_state{}; | ||||
|     struct SharedMemory { | ||||
|         CommonHeader header; | ||||
|         std::array<PadStates, 17> pad_states; | ||||
|         INSERT_PADDING_BYTES(0x138); | ||||
|     }; | ||||
|     static_assert(sizeof(SharedMemory) == 0x400, "SharedMemory is an invalid size"); | ||||
|     SharedMemory shared_memory{}; | ||||
|  | ||||
|     Core::HID::EmulatedController* controller; | ||||
|     std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeButton::NUM_BUTTONS_HID> | ||||
|         buttons; | ||||
|     std::array<std::unique_ptr<Input::AnalogDevice>, Settings::NativeAnalog::NUM_STICKS_HID> | ||||
|         analogs; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -7,7 +7,6 @@ | ||||
| #include "common/settings.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/hid/controllers/gesture.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| @@ -24,14 +23,16 @@ constexpr f32 Square(s32 num) { | ||||
|     return static_cast<f32>(num * num); | ||||
| } | ||||
|  | ||||
| Controller_Gesture::Controller_Gesture(Core::HID::HIDCore& hid_core_) : ControllerBase(hid_core_) { | ||||
|     console = hid_core.GetEmulatedConsole(); | ||||
| } | ||||
| Controller_Gesture::Controller_Gesture(Core::System& system_) : ControllerBase(system_) {} | ||||
| Controller_Gesture::~Controller_Gesture() = default; | ||||
|  | ||||
| void Controller_Gesture::OnInit() { | ||||
|     gesture_lifo.buffer_count = 0; | ||||
|     gesture_lifo.buffer_tail = 0; | ||||
|     for (std::size_t id = 0; id < MAX_FINGERS; ++id) { | ||||
|         mouse_finger_id[id] = MAX_POINTS; | ||||
|         keyboard_finger_id[id] = MAX_POINTS; | ||||
|         udp_finger_id[id] = MAX_POINTS; | ||||
|     } | ||||
|     shared_memory.header.entry_count = 0; | ||||
|     force_update = true; | ||||
| } | ||||
|  | ||||
| @@ -39,38 +40,50 @@ void Controller_Gesture::OnRelease() {} | ||||
|  | ||||
| void Controller_Gesture::OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                   std::size_t size) { | ||||
|     shared_memory.header.timestamp = core_timing.GetCPUTicks(); | ||||
|     shared_memory.header.total_entry_count = 17; | ||||
|  | ||||
|     if (!IsControllerActivated()) { | ||||
|         gesture_lifo.buffer_count = 0; | ||||
|         gesture_lifo.buffer_tail = 0; | ||||
|         std::memcpy(data + SHARED_MEMORY_OFFSET, &gesture_lifo, sizeof(gesture_lifo)); | ||||
|         shared_memory.header.entry_count = 0; | ||||
|         shared_memory.header.last_entry_index = 0; | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     ReadTouchInput(); | ||||
|  | ||||
|     GestureProperties gesture = GetGestureProperties(); | ||||
|     f32 time_difference = | ||||
|         static_cast<f32>(gesture_lifo.timestamp - last_update_timestamp) / (1000 * 1000 * 1000); | ||||
|     f32 time_difference = static_cast<f32>(shared_memory.header.timestamp - last_update_timestamp) / | ||||
|                           (1000 * 1000 * 1000); | ||||
|  | ||||
|     // Only update if necesary | ||||
|     if (!ShouldUpdateGesture(gesture, time_difference)) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     last_update_timestamp = gesture_lifo.timestamp; | ||||
|     last_update_timestamp = shared_memory.header.timestamp; | ||||
|     UpdateGestureSharedMemory(data, size, gesture, time_difference); | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::ReadTouchInput() { | ||||
|     const auto touch_status = console->GetTouch(); | ||||
|     for (std::size_t id = 0; id < fingers.size(); ++id) { | ||||
|         fingers[id] = touch_status[id]; | ||||
|     const Input::TouchStatus& mouse_status = touch_mouse_device->GetStatus(); | ||||
|     const Input::TouchStatus& udp_status = touch_udp_device->GetStatus(); | ||||
|     for (std::size_t id = 0; id < mouse_status.size(); ++id) { | ||||
|         mouse_finger_id[id] = UpdateTouchInputEvent(mouse_status[id], mouse_finger_id[id]); | ||||
|         udp_finger_id[id] = UpdateTouchInputEvent(udp_status[id], udp_finger_id[id]); | ||||
|     } | ||||
|  | ||||
|     if (Settings::values.use_touch_from_button) { | ||||
|         const Input::TouchStatus& keyboard_status = touch_btn_device->GetStatus(); | ||||
|         for (std::size_t id = 0; id < mouse_status.size(); ++id) { | ||||
|             keyboard_finger_id[id] = | ||||
|                 UpdateTouchInputEvent(keyboard_status[id], keyboard_finger_id[id]); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| bool Controller_Gesture::ShouldUpdateGesture(const GestureProperties& gesture, | ||||
|                                              f32 time_difference) { | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|     const auto& last_entry = shared_memory.gesture_states[shared_memory.header.last_entry_index]; | ||||
|     if (force_update) { | ||||
|         force_update = false; | ||||
|         return true; | ||||
| @@ -84,7 +97,7 @@ bool Controller_Gesture::ShouldUpdateGesture(const GestureProperties& gesture, | ||||
|     } | ||||
|  | ||||
|     // Update on press and hold event after 0.5 seconds | ||||
|     if (last_entry.type == GestureType::Touch && last_entry.point_count == 1 && | ||||
|     if (last_entry.type == TouchType::Touch && last_entry.point_count == 1 && | ||||
|         time_difference > press_delay) { | ||||
|         return enable_press_and_tap; | ||||
|     } | ||||
| @@ -95,19 +108,27 @@ bool Controller_Gesture::ShouldUpdateGesture(const GestureProperties& gesture, | ||||
| void Controller_Gesture::UpdateGestureSharedMemory(u8* data, std::size_t size, | ||||
|                                                    GestureProperties& gesture, | ||||
|                                                    f32 time_difference) { | ||||
|     GestureType type = GestureType::Idle; | ||||
|     GestureAttribute attributes{}; | ||||
|     TouchType type = TouchType::Idle; | ||||
|     Attribute attributes{}; | ||||
|  | ||||
|     const auto& last_entry = gesture_lifo.ReadCurrentEntry().state; | ||||
|     const auto& last_entry = shared_memory.gesture_states[shared_memory.header.last_entry_index]; | ||||
|     shared_memory.header.last_entry_index = (shared_memory.header.last_entry_index + 1) % 17; | ||||
|     auto& cur_entry = shared_memory.gesture_states[shared_memory.header.last_entry_index]; | ||||
|  | ||||
|     // Reset next state to default | ||||
|     next_state.sampling_number = last_entry.sampling_number + 1; | ||||
|     next_state.delta = {}; | ||||
|     next_state.vel_x = 0; | ||||
|     next_state.vel_y = 0; | ||||
|     next_state.direction = GestureDirection::None; | ||||
|     next_state.rotation_angle = 0; | ||||
|     next_state.scale = 0; | ||||
|     if (shared_memory.header.entry_count < 16) { | ||||
|         shared_memory.header.entry_count++; | ||||
|     } | ||||
|  | ||||
|     cur_entry.sampling_number = last_entry.sampling_number + 1; | ||||
|     cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|  | ||||
|     // Reset values to default | ||||
|     cur_entry.delta = {}; | ||||
|     cur_entry.vel_x = 0; | ||||
|     cur_entry.vel_y = 0; | ||||
|     cur_entry.direction = Direction::None; | ||||
|     cur_entry.rotation_angle = 0; | ||||
|     cur_entry.scale = 0; | ||||
|  | ||||
|     if (gesture.active_points > 0) { | ||||
|         if (last_gesture.active_points == 0) { | ||||
| @@ -120,47 +141,46 @@ void Controller_Gesture::UpdateGestureSharedMemory(u8* data, std::size_t size, | ||||
|     } | ||||
|  | ||||
|     // Apply attributes | ||||
|     next_state.detection_count = gesture.detection_count; | ||||
|     next_state.type = type; | ||||
|     next_state.attributes = attributes; | ||||
|     next_state.pos = gesture.mid_point; | ||||
|     next_state.point_count = static_cast<s32>(gesture.active_points); | ||||
|     next_state.points = gesture.points; | ||||
|     cur_entry.detection_count = gesture.detection_count; | ||||
|     cur_entry.type = type; | ||||
|     cur_entry.attributes = attributes; | ||||
|     cur_entry.pos = gesture.mid_point; | ||||
|     cur_entry.point_count = static_cast<s32>(gesture.active_points); | ||||
|     cur_entry.points = gesture.points; | ||||
|     last_gesture = gesture; | ||||
|  | ||||
|     gesture_lifo.WriteNextEntry(next_state); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &gesture_lifo, sizeof(gesture_lifo)); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &shared_memory, sizeof(SharedMemory)); | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::NewGesture(GestureProperties& gesture, GestureType& type, | ||||
|                                     GestureAttribute& attributes) { | ||||
| void Controller_Gesture::NewGesture(GestureProperties& gesture, TouchType& type, | ||||
|                                     Attribute& attributes) { | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|  | ||||
|     gesture.detection_count++; | ||||
|     type = GestureType::Touch; | ||||
|     type = TouchType::Touch; | ||||
|  | ||||
|     // New touch after cancel is not considered new | ||||
|     if (last_entry.type != GestureType::Cancel) { | ||||
|     if (last_entry.type != TouchType::Cancel) { | ||||
|         attributes.is_new_touch.Assign(1); | ||||
|         enable_press_and_tap = true; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::UpdateExistingGesture(GestureProperties& gesture, GestureType& type, | ||||
| void Controller_Gesture::UpdateExistingGesture(GestureProperties& gesture, TouchType& type, | ||||
|                                                f32 time_difference) { | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|  | ||||
|     // Promote to pan type if touch moved | ||||
|     for (size_t id = 0; id < MAX_POINTS; id++) { | ||||
|         if (gesture.points[id] != last_gesture.points[id]) { | ||||
|             type = GestureType::Pan; | ||||
|             type = TouchType::Pan; | ||||
|             break; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // Number of fingers changed cancel the last event and clear data | ||||
|     if (gesture.active_points != last_gesture.active_points) { | ||||
|         type = GestureType::Cancel; | ||||
|         type = TouchType::Cancel; | ||||
|         enable_press_and_tap = false; | ||||
|         gesture.active_points = 0; | ||||
|         gesture.mid_point = {}; | ||||
| @@ -169,41 +189,41 @@ void Controller_Gesture::UpdateExistingGesture(GestureProperties& gesture, Gestu | ||||
|     } | ||||
|  | ||||
|     // Calculate extra parameters of panning | ||||
|     if (type == GestureType::Pan) { | ||||
|     if (type == TouchType::Pan) { | ||||
|         UpdatePanEvent(gesture, last_gesture, type, time_difference); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Promote to press type | ||||
|     if (last_entry.type == GestureType::Touch) { | ||||
|         type = GestureType::Press; | ||||
|     if (last_entry.type == TouchType::Touch) { | ||||
|         type = TouchType::Press; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::EndGesture(GestureProperties& gesture, | ||||
|                                     GestureProperties& last_gesture_props, GestureType& type, | ||||
|                                     GestureAttribute& attributes, f32 time_difference) { | ||||
|                                     GestureProperties& last_gesture_props, TouchType& type, | ||||
|                                     Attribute& attributes, f32 time_difference) { | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|  | ||||
|     if (last_gesture_props.active_points != 0) { | ||||
|         switch (last_entry.type) { | ||||
|         case GestureType::Touch: | ||||
|         case TouchType::Touch: | ||||
|             if (enable_press_and_tap) { | ||||
|                 SetTapEvent(gesture, last_gesture_props, type, attributes); | ||||
|                 return; | ||||
|             } | ||||
|             type = GestureType::Cancel; | ||||
|             type = TouchType::Cancel; | ||||
|             force_update = true; | ||||
|             break; | ||||
|         case GestureType::Press: | ||||
|         case GestureType::Tap: | ||||
|         case GestureType::Swipe: | ||||
|         case GestureType::Pinch: | ||||
|         case GestureType::Rotate: | ||||
|             type = GestureType::Complete; | ||||
|         case TouchType::Press: | ||||
|         case TouchType::Tap: | ||||
|         case TouchType::Swipe: | ||||
|         case TouchType::Pinch: | ||||
|         case TouchType::Rotate: | ||||
|             type = TouchType::Complete; | ||||
|             force_update = true; | ||||
|             break; | ||||
|         case GestureType::Pan: | ||||
|         case TouchType::Pan: | ||||
|             EndPanEvent(gesture, last_gesture_props, type, time_difference); | ||||
|             break; | ||||
|         default: | ||||
| @@ -211,15 +231,15 @@ void Controller_Gesture::EndGesture(GestureProperties& gesture, | ||||
|         } | ||||
|         return; | ||||
|     } | ||||
|     if (last_entry.type == GestureType::Complete || last_entry.type == GestureType::Cancel) { | ||||
|     if (last_entry.type == TouchType::Complete || last_entry.type == TouchType::Cancel) { | ||||
|         gesture.detection_count++; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::SetTapEvent(GestureProperties& gesture, | ||||
|                                      GestureProperties& last_gesture_props, GestureType& type, | ||||
|                                      GestureAttribute& attributes) { | ||||
|     type = GestureType::Tap; | ||||
|                                      GestureProperties& last_gesture_props, TouchType& type, | ||||
|                                      Attribute& attributes) { | ||||
|     type = TouchType::Tap; | ||||
|     gesture = last_gesture_props; | ||||
|     force_update = true; | ||||
|     f32 tap_time_difference = | ||||
| @@ -231,42 +251,44 @@ void Controller_Gesture::SetTapEvent(GestureProperties& gesture, | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::UpdatePanEvent(GestureProperties& gesture, | ||||
|                                         GestureProperties& last_gesture_props, GestureType& type, | ||||
|                                         GestureProperties& last_gesture_props, TouchType& type, | ||||
|                                         f32 time_difference) { | ||||
|     auto& cur_entry = shared_memory.gesture_states[shared_memory.header.last_entry_index]; | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|  | ||||
|     next_state.delta = gesture.mid_point - last_entry.pos; | ||||
|     next_state.vel_x = static_cast<f32>(next_state.delta.x) / time_difference; | ||||
|     next_state.vel_y = static_cast<f32>(next_state.delta.y) / time_difference; | ||||
|     cur_entry.delta = gesture.mid_point - last_entry.pos; | ||||
|     cur_entry.vel_x = static_cast<f32>(cur_entry.delta.x) / time_difference; | ||||
|     cur_entry.vel_y = static_cast<f32>(cur_entry.delta.y) / time_difference; | ||||
|     last_pan_time_difference = time_difference; | ||||
|  | ||||
|     // Promote to pinch type | ||||
|     if (std::abs(gesture.average_distance - last_gesture_props.average_distance) > | ||||
|         pinch_threshold) { | ||||
|         type = GestureType::Pinch; | ||||
|         next_state.scale = gesture.average_distance / last_gesture_props.average_distance; | ||||
|         type = TouchType::Pinch; | ||||
|         cur_entry.scale = gesture.average_distance / last_gesture_props.average_distance; | ||||
|     } | ||||
|  | ||||
|     const f32 angle_between_two_lines = std::atan((gesture.angle - last_gesture_props.angle) / | ||||
|                                                   (1 + (gesture.angle * last_gesture_props.angle))); | ||||
|     // Promote to rotate type | ||||
|     if (std::abs(angle_between_two_lines) > angle_threshold) { | ||||
|         type = GestureType::Rotate; | ||||
|         next_state.scale = 0; | ||||
|         next_state.rotation_angle = angle_between_two_lines * 180.0f / Common::PI; | ||||
|         type = TouchType::Rotate; | ||||
|         cur_entry.scale = 0; | ||||
|         cur_entry.rotation_angle = angle_between_two_lines * 180.0f / Common::PI; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::EndPanEvent(GestureProperties& gesture, | ||||
|                                      GestureProperties& last_gesture_props, GestureType& type, | ||||
|                                      GestureProperties& last_gesture_props, TouchType& type, | ||||
|                                      f32 time_difference) { | ||||
|     auto& cur_entry = shared_memory.gesture_states[shared_memory.header.last_entry_index]; | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|     next_state.vel_x = | ||||
|     cur_entry.vel_x = | ||||
|         static_cast<f32>(last_entry.delta.x) / (last_pan_time_difference + time_difference); | ||||
|     next_state.vel_y = | ||||
|     cur_entry.vel_y = | ||||
|         static_cast<f32>(last_entry.delta.y) / (last_pan_time_difference + time_difference); | ||||
|     const f32 curr_vel = | ||||
|         std::sqrt((next_state.vel_x * next_state.vel_x) + (next_state.vel_y * next_state.vel_y)); | ||||
|         std::sqrt((cur_entry.vel_x * cur_entry.vel_x) + (cur_entry.vel_y * cur_entry.vel_y)); | ||||
|  | ||||
|     // Set swipe event with parameters | ||||
|     if (curr_vel > swipe_threshold) { | ||||
| @@ -275,50 +297,105 @@ void Controller_Gesture::EndPanEvent(GestureProperties& gesture, | ||||
|     } | ||||
|  | ||||
|     // End panning without swipe | ||||
|     type = GestureType::Complete; | ||||
|     next_state.vel_x = 0; | ||||
|     next_state.vel_y = 0; | ||||
|     type = TouchType::Complete; | ||||
|     cur_entry.vel_x = 0; | ||||
|     cur_entry.vel_y = 0; | ||||
|     force_update = true; | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::SetSwipeEvent(GestureProperties& gesture, | ||||
|                                        GestureProperties& last_gesture_props, GestureType& type) { | ||||
|                                        GestureProperties& last_gesture_props, TouchType& type) { | ||||
|     auto& cur_entry = shared_memory.gesture_states[shared_memory.header.last_entry_index]; | ||||
|     const auto& last_entry = GetLastGestureEntry(); | ||||
|  | ||||
|     type = GestureType::Swipe; | ||||
|     type = TouchType::Swipe; | ||||
|     gesture = last_gesture_props; | ||||
|     force_update = true; | ||||
|     next_state.delta = last_entry.delta; | ||||
|     cur_entry.delta = last_entry.delta; | ||||
|  | ||||
|     if (std::abs(next_state.delta.x) > std::abs(next_state.delta.y)) { | ||||
|         if (next_state.delta.x > 0) { | ||||
|             next_state.direction = GestureDirection::Right; | ||||
|     if (std::abs(cur_entry.delta.x) > std::abs(cur_entry.delta.y)) { | ||||
|         if (cur_entry.delta.x > 0) { | ||||
|             cur_entry.direction = Direction::Right; | ||||
|             return; | ||||
|         } | ||||
|         next_state.direction = GestureDirection::Left; | ||||
|         cur_entry.direction = Direction::Left; | ||||
|         return; | ||||
|     } | ||||
|     if (next_state.delta.y > 0) { | ||||
|         next_state.direction = GestureDirection::Down; | ||||
|     if (cur_entry.delta.y > 0) { | ||||
|         cur_entry.direction = Direction::Down; | ||||
|         return; | ||||
|     } | ||||
|     next_state.direction = GestureDirection::Up; | ||||
|     cur_entry.direction = Direction::Up; | ||||
| } | ||||
|  | ||||
| void Controller_Gesture::OnLoadInputDevices() { | ||||
|     touch_mouse_device = Input::CreateDevice<Input::TouchDevice>("engine:emu_window"); | ||||
|     touch_udp_device = Input::CreateDevice<Input::TouchDevice>("engine:cemuhookudp"); | ||||
|     touch_btn_device = Input::CreateDevice<Input::TouchDevice>("engine:touch_from_button"); | ||||
| } | ||||
|  | ||||
| std::optional<std::size_t> Controller_Gesture::GetUnusedFingerID() const { | ||||
|     // Dont assign any touch input to a point if disabled | ||||
|     if (!Settings::values.touchscreen.enabled) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     std::size_t first_free_id = 0; | ||||
|     while (first_free_id < MAX_POINTS) { | ||||
|         if (!fingers[first_free_id].pressed) { | ||||
|             return first_free_id; | ||||
|         } else { | ||||
|             first_free_id++; | ||||
|         } | ||||
|     } | ||||
|     return std::nullopt; | ||||
| } | ||||
|  | ||||
| Controller_Gesture::GestureState& Controller_Gesture::GetLastGestureEntry() { | ||||
|     return shared_memory.gesture_states[(shared_memory.header.last_entry_index + 16) % 17]; | ||||
| } | ||||
|  | ||||
| const Controller_Gesture::GestureState& Controller_Gesture::GetLastGestureEntry() const { | ||||
|     return gesture_lifo.ReadCurrentEntry().state; | ||||
|     return shared_memory.gesture_states[(shared_memory.header.last_entry_index + 16) % 17]; | ||||
| } | ||||
|  | ||||
| std::size_t Controller_Gesture::UpdateTouchInputEvent( | ||||
|     const std::tuple<float, float, bool>& touch_input, std::size_t finger_id) { | ||||
|     const auto& [x, y, pressed] = touch_input; | ||||
|     if (finger_id > MAX_POINTS) { | ||||
|         LOG_ERROR(Service_HID, "Invalid finger id {}", finger_id); | ||||
|         return MAX_POINTS; | ||||
|     } | ||||
|     if (pressed) { | ||||
|         if (finger_id == MAX_POINTS) { | ||||
|             const auto first_free_id = GetUnusedFingerID(); | ||||
|             if (!first_free_id) { | ||||
|                 // Invalid finger id do nothing | ||||
|                 return MAX_POINTS; | ||||
|             } | ||||
|             finger_id = first_free_id.value(); | ||||
|             fingers[finger_id].pressed = true; | ||||
|         } | ||||
|         fingers[finger_id].pos = {x, y}; | ||||
|         return finger_id; | ||||
|     } | ||||
|  | ||||
|     if (finger_id != MAX_POINTS) { | ||||
|         fingers[finger_id].pressed = false; | ||||
|     } | ||||
|  | ||||
|     return MAX_POINTS; | ||||
| } | ||||
|  | ||||
| Controller_Gesture::GestureProperties Controller_Gesture::GetGestureProperties() { | ||||
|     GestureProperties gesture; | ||||
|     std::array<Core::HID::TouchFinger, MAX_POINTS> active_fingers; | ||||
|     std::array<Finger, MAX_POINTS> active_fingers; | ||||
|     const auto end_iter = std::copy_if(fingers.begin(), fingers.end(), active_fingers.begin(), | ||||
|                                        [](const auto& finger) { return finger.pressed; }); | ||||
|     gesture.active_points = | ||||
|         static_cast<std::size_t>(std::distance(active_fingers.begin(), end_iter)); | ||||
|  | ||||
|     for (size_t id = 0; id < gesture.active_points; ++id) { | ||||
|         const auto& [active_x, active_y] = active_fingers[id].position; | ||||
|         const auto& [active_x, active_y] = active_fingers[id].pos; | ||||
|         gesture.points[id] = { | ||||
|             .x = static_cast<s32>(active_x * Layout::ScreenUndocked::Width), | ||||
|             .y = static_cast<s32>(active_y * Layout::ScreenUndocked::Height), | ||||
|   | ||||
| @@ -8,14 +8,13 @@ | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/point.h" | ||||
| #include "core/hid/emulated_console.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_Gesture final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_Gesture(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_Gesture(Core::System& system_); | ||||
|     ~Controller_Gesture() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -27,12 +26,14 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
| private: | ||||
|     static constexpr size_t MAX_FINGERS = 16; | ||||
|     static constexpr size_t MAX_POINTS = 4; | ||||
|  | ||||
|     // This is nn::hid::GestureType | ||||
|     enum class GestureType : u32 { | ||||
|     enum class TouchType : u32 { | ||||
|         Idle,     // Nothing touching the screen | ||||
|         Complete, // Set at the end of a touch event | ||||
|         Cancel,   // Set when the number of fingers change | ||||
| @@ -45,8 +46,7 @@ private: | ||||
|         Rotate,   // All points rotating from the midpoint | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::GestureDirection | ||||
|     enum class GestureDirection : u32 { | ||||
|     enum class Direction : u32 { | ||||
|         None, | ||||
|         Left, | ||||
|         Up, | ||||
| @@ -54,41 +54,51 @@ private: | ||||
|         Down, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::GestureAttribute | ||||
|     struct GestureAttribute { | ||||
|     struct Attribute { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u32_le raw{}; | ||||
|  | ||||
|             BitField<4, 1, u32> is_new_touch; | ||||
|             BitField<8, 1, u32> is_double_tap; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(GestureAttribute) == 4, "GestureAttribute is an invalid size"); | ||||
|     static_assert(sizeof(Attribute) == 4, "Attribute is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::GestureState | ||||
|     struct GestureState { | ||||
|         s64 sampling_number; | ||||
|         s64 detection_count; | ||||
|         GestureType type; | ||||
|         GestureDirection direction; | ||||
|         Common::Point<s32> pos; | ||||
|         Common::Point<s32> delta; | ||||
|         s64_le sampling_number; | ||||
|         s64_le sampling_number2; | ||||
|         s64_le detection_count; | ||||
|         TouchType type; | ||||
|         Direction direction; | ||||
|         Common::Point<s32_le> pos; | ||||
|         Common::Point<s32_le> delta; | ||||
|         f32 vel_x; | ||||
|         f32 vel_y; | ||||
|         GestureAttribute attributes; | ||||
|         Attribute attributes; | ||||
|         f32 scale; | ||||
|         f32 rotation_angle; | ||||
|         s32 point_count; | ||||
|         std::array<Common::Point<s32>, 4> points; | ||||
|         s32_le point_count; | ||||
|         std::array<Common::Point<s32_le>, 4> points; | ||||
|     }; | ||||
|     static_assert(sizeof(GestureState) == 0x68, "GestureState is an invalid size"); | ||||
|  | ||||
|     struct SharedMemory { | ||||
|         CommonHeader header; | ||||
|         std::array<GestureState, 17> gesture_states; | ||||
|     }; | ||||
|     static_assert(sizeof(SharedMemory) == 0x708, "SharedMemory is an invalid size"); | ||||
|  | ||||
|     struct Finger { | ||||
|         Common::Point<f32> pos{}; | ||||
|         bool pressed{}; | ||||
|     }; | ||||
|     static_assert(sizeof(GestureState) == 0x60, "GestureState is an invalid size"); | ||||
|  | ||||
|     struct GestureProperties { | ||||
|         std::array<Common::Point<s32>, MAX_POINTS> points{}; | ||||
|         std::array<Common::Point<s32_le>, MAX_POINTS> points{}; | ||||
|         std::size_t active_points{}; | ||||
|         Common::Point<s32> mid_point{}; | ||||
|         s64 detection_count{}; | ||||
|         u64 delta_time{}; | ||||
|         Common::Point<s32_le> mid_point{}; | ||||
|         s64_le detection_count{}; | ||||
|         u64_le delta_time{}; | ||||
|         f32 average_distance{}; | ||||
|         f32 angle{}; | ||||
|     }; | ||||
| @@ -104,48 +114,61 @@ private: | ||||
|                                    f32 time_difference); | ||||
|  | ||||
|     // Initializes new gesture | ||||
|     void NewGesture(GestureProperties& gesture, GestureType& type, GestureAttribute& attributes); | ||||
|     void NewGesture(GestureProperties& gesture, TouchType& type, Attribute& attributes); | ||||
|  | ||||
|     // Updates existing gesture state | ||||
|     void UpdateExistingGesture(GestureProperties& gesture, GestureType& type, f32 time_difference); | ||||
|     void UpdateExistingGesture(GestureProperties& gesture, TouchType& type, f32 time_difference); | ||||
|  | ||||
|     // Terminates exiting gesture | ||||
|     void EndGesture(GestureProperties& gesture, GestureProperties& last_gesture_props, | ||||
|                     GestureType& type, GestureAttribute& attributes, f32 time_difference); | ||||
|                     TouchType& type, Attribute& attributes, f32 time_difference); | ||||
|  | ||||
|     // Set current event to a tap event | ||||
|     void SetTapEvent(GestureProperties& gesture, GestureProperties& last_gesture_props, | ||||
|                      GestureType& type, GestureAttribute& attributes); | ||||
|                      TouchType& type, Attribute& attributes); | ||||
|  | ||||
|     // Calculates and set the extra parameters related to a pan event | ||||
|     void UpdatePanEvent(GestureProperties& gesture, GestureProperties& last_gesture_props, | ||||
|                         GestureType& type, f32 time_difference); | ||||
|                         TouchType& type, f32 time_difference); | ||||
|  | ||||
|     // Terminates the pan event | ||||
|     void EndPanEvent(GestureProperties& gesture, GestureProperties& last_gesture_props, | ||||
|                      GestureType& type, f32 time_difference); | ||||
|                      TouchType& type, f32 time_difference); | ||||
|  | ||||
|     // Set current event to a swipe event | ||||
|     void SetSwipeEvent(GestureProperties& gesture, GestureProperties& last_gesture_props, | ||||
|                        GestureType& type); | ||||
|                        TouchType& type); | ||||
|  | ||||
|     // Returns an unused finger id, if there is no fingers available std::nullopt is returned. | ||||
|     [[nodiscard]] std::optional<size_t> GetUnusedFingerID() const; | ||||
|  | ||||
|     // Retrieves the last gesture entry, as indicated by shared memory indices. | ||||
|     [[nodiscard]] GestureState& GetLastGestureEntry(); | ||||
|     [[nodiscard]] const GestureState& GetLastGestureEntry() const; | ||||
|  | ||||
|     /** | ||||
|      * If the touch is new it tries to assign a new finger id, if there is no fingers available no | ||||
|      * changes will be made. Updates the coordinates if the finger id it's already set. If the touch | ||||
|      * ends delays the output by one frame to set the end_touch flag before finally freeing the | ||||
|      * finger id | ||||
|      */ | ||||
|     size_t UpdateTouchInputEvent(const std::tuple<float, float, bool>& touch_input, | ||||
|                                  size_t finger_id); | ||||
|  | ||||
|     // Returns the average distance, angle and middle point of the active fingers | ||||
|     GestureProperties GetGestureProperties(); | ||||
|  | ||||
|     // This is nn::hid::detail::GestureLifo | ||||
|     Lifo<GestureState> gesture_lifo{}; | ||||
|     static_assert(sizeof(gesture_lifo) == 0x708, "gesture_lifo is an invalid size"); | ||||
|     GestureState next_state{}; | ||||
|  | ||||
|     Core::HID::EmulatedConsole* console; | ||||
|  | ||||
|     std::array<Core::HID::TouchFinger, MAX_POINTS> fingers{}; | ||||
|     SharedMemory shared_memory{}; | ||||
|     std::unique_ptr<Input::TouchDevice> touch_mouse_device; | ||||
|     std::unique_ptr<Input::TouchDevice> touch_udp_device; | ||||
|     std::unique_ptr<Input::TouchDevice> touch_btn_device; | ||||
|     std::array<size_t, MAX_FINGERS> mouse_finger_id{}; | ||||
|     std::array<size_t, MAX_FINGERS> keyboard_finger_id{}; | ||||
|     std::array<size_t, MAX_FINGERS> udp_finger_id{}; | ||||
|     std::array<Finger, MAX_POINTS> fingers{}; | ||||
|     GestureProperties last_gesture{}; | ||||
|     s64 last_update_timestamp{}; | ||||
|     s64 last_tap_timestamp{}; | ||||
|     s64_le last_update_timestamp{}; | ||||
|     s64_le last_tap_timestamp{}; | ||||
|     f32 last_pan_time_difference{}; | ||||
|     bool force_update{false}; | ||||
|     bool enable_press_and_tap{false}; | ||||
|   | ||||
| @@ -6,18 +6,13 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hid/emulated_devices.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/hid/controllers/keyboard.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| constexpr std::size_t SHARED_MEMORY_OFFSET = 0x3800; | ||||
| constexpr u8 KEYS_PER_BYTE = 8; | ||||
|  | ||||
| Controller_Keyboard::Controller_Keyboard(Core::HID::HIDCore& hid_core_) | ||||
|     : ControllerBase{hid_core_} { | ||||
|     emulated_devices = hid_core.GetEmulatedDevices(); | ||||
| } | ||||
|  | ||||
| Controller_Keyboard::Controller_Keyboard(Core::System& system_) : ControllerBase{system_} {} | ||||
| Controller_Keyboard::~Controller_Keyboard() = default; | ||||
|  | ||||
| void Controller_Keyboard::OnInit() {} | ||||
| @@ -26,28 +21,51 @@ void Controller_Keyboard::OnRelease() {} | ||||
|  | ||||
| void Controller_Keyboard::OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                    std::size_t size) { | ||||
|     shared_memory.header.timestamp = core_timing.GetCPUTicks(); | ||||
|     shared_memory.header.total_entry_count = 17; | ||||
|  | ||||
|     if (!IsControllerActivated()) { | ||||
|         keyboard_lifo.buffer_count = 0; | ||||
|         keyboard_lifo.buffer_tail = 0; | ||||
|         std::memcpy(data + SHARED_MEMORY_OFFSET, &keyboard_lifo, sizeof(keyboard_lifo)); | ||||
|         shared_memory.header.entry_count = 0; | ||||
|         shared_memory.header.last_entry_index = 0; | ||||
|         return; | ||||
|     } | ||||
|     shared_memory.header.entry_count = 16; | ||||
|  | ||||
|     const auto& last_entry = keyboard_lifo.ReadCurrentEntry().state; | ||||
|     next_state.sampling_number = last_entry.sampling_number + 1; | ||||
|     const auto& last_entry = shared_memory.pad_states[shared_memory.header.last_entry_index]; | ||||
|     shared_memory.header.last_entry_index = (shared_memory.header.last_entry_index + 1) % 17; | ||||
|     auto& cur_entry = shared_memory.pad_states[shared_memory.header.last_entry_index]; | ||||
|  | ||||
|     cur_entry.sampling_number = last_entry.sampling_number + 1; | ||||
|     cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|  | ||||
|     cur_entry.key.fill(0); | ||||
|     if (Settings::values.keyboard_enabled) { | ||||
|         const auto& keyboard_state = emulated_devices->GetKeyboard(); | ||||
|         const auto& keyboard_modifier_state = emulated_devices->GetKeyboardModifier(); | ||||
|         for (std::size_t i = 0; i < keyboard_keys.size(); ++i) { | ||||
|             auto& entry = cur_entry.key[i / KEYS_PER_BYTE]; | ||||
|             entry = static_cast<u8>(entry | (keyboard_keys[i]->GetStatus() << (i % KEYS_PER_BYTE))); | ||||
|         } | ||||
|  | ||||
|         next_state.key = keyboard_state; | ||||
|         next_state.modifier = keyboard_modifier_state; | ||||
|         // This is always enabled on HW. Check what it actually does | ||||
|         next_state.modifier.unknown.Assign(1); | ||||
|         using namespace Settings::NativeKeyboard; | ||||
|  | ||||
|         // TODO: Assign the correct key to all modifiers | ||||
|         cur_entry.modifier.control.Assign(keyboard_mods[LeftControl]->GetStatus()); | ||||
|         cur_entry.modifier.shift.Assign(keyboard_mods[LeftShift]->GetStatus()); | ||||
|         cur_entry.modifier.left_alt.Assign(keyboard_mods[LeftAlt]->GetStatus()); | ||||
|         cur_entry.modifier.right_alt.Assign(keyboard_mods[RightAlt]->GetStatus()); | ||||
|         cur_entry.modifier.gui.Assign(0); | ||||
|         cur_entry.modifier.caps_lock.Assign(keyboard_mods[CapsLock]->GetStatus()); | ||||
|         cur_entry.modifier.scroll_lock.Assign(keyboard_mods[ScrollLock]->GetStatus()); | ||||
|         cur_entry.modifier.num_lock.Assign(keyboard_mods[NumLock]->GetStatus()); | ||||
|         cur_entry.modifier.katakana.Assign(0); | ||||
|         cur_entry.modifier.hiragana.Assign(0); | ||||
|     } | ||||
|  | ||||
|     keyboard_lifo.WriteNextEntry(next_state); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &keyboard_lifo, sizeof(keyboard_lifo)); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &shared_memory, sizeof(SharedMemory)); | ||||
| } | ||||
|  | ||||
| void Controller_Keyboard::OnLoadInputDevices() { | ||||
|     std::transform(Settings::values.keyboard_keys.begin(), Settings::values.keyboard_keys.end(), | ||||
|                    keyboard_keys.begin(), Input::CreateDevice<Input::ButtonDevice>); | ||||
|     std::transform(Settings::values.keyboard_mods.begin(), Settings::values.keyboard_mods.end(), | ||||
|                    keyboard_mods.begin(), Input::CreateDevice<Input::ButtonDevice>); | ||||
| } | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -8,20 +8,15 @@ | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "common/swap.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class EmulatedDevices; | ||||
| struct KeyboardModifier; | ||||
| struct KeyboardKey; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_Keyboard final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_Keyboard(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_Keyboard(Core::System& system_); | ||||
|     ~Controller_Keyboard() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -33,20 +28,47 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
| private: | ||||
|     // This is nn::hid::detail::KeyboardState | ||||
|     struct KeyboardState { | ||||
|         s64 sampling_number; | ||||
|         Core::HID::KeyboardModifier modifier; | ||||
|         Core::HID::KeyboardKey key; | ||||
|     struct Modifiers { | ||||
|         union { | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> control; | ||||
|             BitField<1, 1, u32> shift; | ||||
|             BitField<2, 1, u32> left_alt; | ||||
|             BitField<3, 1, u32> right_alt; | ||||
|             BitField<4, 1, u32> gui; | ||||
|             BitField<8, 1, u32> caps_lock; | ||||
|             BitField<9, 1, u32> scroll_lock; | ||||
|             BitField<10, 1, u32> num_lock; | ||||
|             BitField<11, 1, u32> katakana; | ||||
|             BitField<12, 1, u32> hiragana; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(KeyboardState) == 0x30, "KeyboardState is an invalid size"); | ||||
|     static_assert(sizeof(Modifiers) == 0x4, "Modifiers is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::KeyboardLifo | ||||
|     Lifo<KeyboardState> keyboard_lifo{}; | ||||
|     static_assert(sizeof(keyboard_lifo) == 0x3D8, "keyboard_lifo is an invalid size"); | ||||
|     KeyboardState next_state{}; | ||||
|     struct KeyboardState { | ||||
|         s64_le sampling_number; | ||||
|         s64_le sampling_number2; | ||||
|  | ||||
|     Core::HID::EmulatedDevices* emulated_devices; | ||||
|         Modifiers modifier; | ||||
|         std::array<u8, 32> key; | ||||
|     }; | ||||
|     static_assert(sizeof(KeyboardState) == 0x38, "KeyboardState is an invalid size"); | ||||
|  | ||||
|     struct SharedMemory { | ||||
|         CommonHeader header; | ||||
|         std::array<KeyboardState, 17> pad_states; | ||||
|         INSERT_PADDING_BYTES(0x28); | ||||
|     }; | ||||
|     static_assert(sizeof(SharedMemory) == 0x400, "SharedMemory is an invalid size"); | ||||
|     SharedMemory shared_memory{}; | ||||
|  | ||||
|     std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeKeyboard::NumKeyboardKeys> | ||||
|         keyboard_keys; | ||||
|     std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeKeyboard::NumKeyboardMods> | ||||
|         keyboard_mods; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -6,17 +6,12 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "core/hid/emulated_devices.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/hid/controllers/mouse.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| constexpr std::size_t SHARED_MEMORY_OFFSET = 0x3400; | ||||
|  | ||||
| Controller_Mouse::Controller_Mouse(Core::HID::HIDCore& hid_core_) : ControllerBase{hid_core_} { | ||||
|     emulated_devices = hid_core.GetEmulatedDevices(); | ||||
| } | ||||
|  | ||||
| Controller_Mouse::Controller_Mouse(Core::System& system_) : ControllerBase{system_} {} | ||||
| Controller_Mouse::~Controller_Mouse() = default; | ||||
|  | ||||
| void Controller_Mouse::OnInit() {} | ||||
| @@ -24,33 +19,50 @@ void Controller_Mouse::OnRelease() {} | ||||
|  | ||||
| void Controller_Mouse::OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                 std::size_t size) { | ||||
|     shared_memory.header.timestamp = core_timing.GetCPUTicks(); | ||||
|     shared_memory.header.total_entry_count = 17; | ||||
|  | ||||
|     if (!IsControllerActivated()) { | ||||
|         mouse_lifo.buffer_count = 0; | ||||
|         mouse_lifo.buffer_tail = 0; | ||||
|         std::memcpy(data + SHARED_MEMORY_OFFSET, &mouse_lifo, sizeof(mouse_lifo)); | ||||
|         shared_memory.header.entry_count = 0; | ||||
|         shared_memory.header.last_entry_index = 0; | ||||
|         return; | ||||
|     } | ||||
|     shared_memory.header.entry_count = 16; | ||||
|  | ||||
|     const auto& last_entry = mouse_lifo.ReadCurrentEntry().state; | ||||
|     next_state.sampling_number = last_entry.sampling_number + 1; | ||||
|     auto& last_entry = shared_memory.mouse_states[shared_memory.header.last_entry_index]; | ||||
|     shared_memory.header.last_entry_index = (shared_memory.header.last_entry_index + 1) % 17; | ||||
|     auto& cur_entry = shared_memory.mouse_states[shared_memory.header.last_entry_index]; | ||||
|  | ||||
|     next_state.attribute.raw = 0; | ||||
|     cur_entry.sampling_number = last_entry.sampling_number + 1; | ||||
|     cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|  | ||||
|     cur_entry.attribute.raw = 0; | ||||
|     if (Settings::values.mouse_enabled) { | ||||
|         const auto& mouse_button_state = emulated_devices->GetMouseButtons(); | ||||
|         const auto& mouse_position_state = emulated_devices->GetMousePosition(); | ||||
|         next_state.attribute.is_connected.Assign(1); | ||||
|         next_state.x = mouse_position_state.x; | ||||
|         next_state.y = mouse_position_state.y; | ||||
|         next_state.delta_x = next_state.x - last_entry.x; | ||||
|         next_state.delta_y = next_state.y - last_entry.y; | ||||
|         next_state.delta_wheel_x = mouse_position_state.delta_wheel_x; | ||||
|         next_state.delta_wheel_y = mouse_position_state.delta_wheel_y; | ||||
|         const auto [px, py, sx, sy] = mouse_device->GetStatus(); | ||||
|         const auto x = static_cast<s32>(px * Layout::ScreenUndocked::Width); | ||||
|         const auto y = static_cast<s32>(py * Layout::ScreenUndocked::Height); | ||||
|         cur_entry.x = x; | ||||
|         cur_entry.y = y; | ||||
|         cur_entry.delta_x = x - last_entry.x; | ||||
|         cur_entry.delta_y = y - last_entry.y; | ||||
|         cur_entry.mouse_wheel_x = sx; | ||||
|         cur_entry.mouse_wheel_y = sy; | ||||
|         cur_entry.attribute.is_connected.Assign(1); | ||||
|  | ||||
|         next_state.button = mouse_button_state; | ||||
|         using namespace Settings::NativeMouseButton; | ||||
|         cur_entry.button.left.Assign(mouse_button_devices[Left]->GetStatus()); | ||||
|         cur_entry.button.right.Assign(mouse_button_devices[Right]->GetStatus()); | ||||
|         cur_entry.button.middle.Assign(mouse_button_devices[Middle]->GetStatus()); | ||||
|         cur_entry.button.forward.Assign(mouse_button_devices[Forward]->GetStatus()); | ||||
|         cur_entry.button.back.Assign(mouse_button_devices[Back]->GetStatus()); | ||||
|     } | ||||
|  | ||||
|     mouse_lifo.WriteNextEntry(next_state); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &mouse_lifo, sizeof(mouse_lifo)); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &shared_memory, sizeof(SharedMemory)); | ||||
| } | ||||
|  | ||||
| void Controller_Mouse::OnLoadInputDevices() { | ||||
|     mouse_device = Input::CreateDevice<Input::MouseDevice>(Settings::values.mouse_device); | ||||
|     std::transform(Settings::values.mouse_buttons.begin(), Settings::values.mouse_buttons.end(), | ||||
|                    mouse_button_devices.begin(), Input::CreateDevice<Input::ButtonDevice>); | ||||
| } | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -7,19 +7,15 @@ | ||||
| #include <array> | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "common/swap.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class EmulatedDevices; | ||||
| struct MouseState; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_Mouse final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_Mouse(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_Mouse(Core::System& system_); | ||||
|     ~Controller_Mouse() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -31,12 +27,53 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override; | ||||
|  | ||||
| private: | ||||
|     // This is nn::hid::detail::MouseLifo | ||||
|     Lifo<Core::HID::MouseState> mouse_lifo{}; | ||||
|     static_assert(sizeof(mouse_lifo) == 0x350, "mouse_lifo is an invalid size"); | ||||
|     Core::HID::MouseState next_state{}; | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
|     Core::HID::EmulatedDevices* emulated_devices; | ||||
| private: | ||||
|     struct Buttons { | ||||
|         union { | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> left; | ||||
|             BitField<1, 1, u32> right; | ||||
|             BitField<2, 1, u32> middle; | ||||
|             BitField<3, 1, u32> forward; | ||||
|             BitField<4, 1, u32> back; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(Buttons) == 0x4, "Buttons is an invalid size"); | ||||
|  | ||||
|     struct Attributes { | ||||
|         union { | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> transferable; | ||||
|             BitField<1, 1, u32> is_connected; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(Attributes) == 0x4, "Attributes is an invalid size"); | ||||
|  | ||||
|     struct MouseState { | ||||
|         s64_le sampling_number; | ||||
|         s64_le sampling_number2; | ||||
|         s32_le x; | ||||
|         s32_le y; | ||||
|         s32_le delta_x; | ||||
|         s32_le delta_y; | ||||
|         s32_le mouse_wheel_x; | ||||
|         s32_le mouse_wheel_y; | ||||
|         Buttons button; | ||||
|         Attributes attribute; | ||||
|     }; | ||||
|     static_assert(sizeof(MouseState) == 0x30, "MouseState is an invalid size"); | ||||
|  | ||||
|     struct SharedMemory { | ||||
|         CommonHeader header; | ||||
|         std::array<MouseState, 17> mouse_states; | ||||
|     }; | ||||
|     SharedMemory shared_memory{}; | ||||
|  | ||||
|     std::unique_ptr<Input::MouseDevice> mouse_device; | ||||
|     std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeMouseButton::NumMouseButtons> | ||||
|         mouse_button_devices; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -11,14 +11,9 @@ | ||||
| #include "common/bit_field.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/quaternion.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class EmulatedController; | ||||
| enum class ControllerTriggerType; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| namespace Kernel { | ||||
| class KEvent; | ||||
| @@ -31,9 +26,12 @@ class ServiceContext; | ||||
|  | ||||
| namespace Service::HID { | ||||
|  | ||||
| constexpr u32 NPAD_HANDHELD = 32; | ||||
| constexpr u32 NPAD_UNKNOWN = 16; // TODO(ogniK): What is this? | ||||
|  | ||||
| class Controller_NPad final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_NPad(Core::HID::HIDCore& hid_core_, | ||||
|     explicit Controller_NPad(Core::System& system_, | ||||
|                              KernelHelpers::ServiceContext& service_context_); | ||||
|     ~Controller_NPad() override; | ||||
|  | ||||
| @@ -50,39 +48,60 @@ public: | ||||
|     void OnMotionUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                         std::size_t size) override; | ||||
|  | ||||
|     // This is nn::hid::GyroscopeZeroDriftMode | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
|     enum class NPadControllerType { | ||||
|         None, | ||||
|         ProController, | ||||
|         Handheld, | ||||
|         JoyDual, | ||||
|         JoyLeft, | ||||
|         JoyRight, | ||||
|         GameCube, | ||||
|         Pokeball, | ||||
|     }; | ||||
|  | ||||
|     enum class NpadType : u8 { | ||||
|         ProController = 3, | ||||
|         Handheld = 4, | ||||
|         JoyconDual = 5, | ||||
|         JoyconLeft = 6, | ||||
|         JoyconRight = 7, | ||||
|         GameCube = 8, | ||||
|         Pokeball = 9, | ||||
|         MaxNpadType = 10, | ||||
|     }; | ||||
|  | ||||
|     enum class DeviceIndex : u8 { | ||||
|         Left = 0, | ||||
|         Right = 1, | ||||
|         None = 2, | ||||
|         MaxDeviceIndex = 3, | ||||
|     }; | ||||
|  | ||||
|     enum class GyroscopeZeroDriftMode : u32 { | ||||
|         Loose = 0, | ||||
|         Standard = 1, | ||||
|         Tight = 2, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadJoyHoldType | ||||
|     enum class NpadJoyHoldType : u64 { | ||||
|     enum class NpadHoldType : u64 { | ||||
|         Vertical = 0, | ||||
|         Horizontal = 1, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadJoyAssignmentMode | ||||
|     enum class NpadJoyAssignmentMode : u32 { | ||||
|     enum class NpadAssignments : u32 { | ||||
|         Dual = 0, | ||||
|         Single = 1, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadJoyDeviceType | ||||
|     enum class NpadJoyDeviceType : s64 { | ||||
|         Left = 0, | ||||
|         Right = 1, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadHandheldActivationMode | ||||
|     enum class NpadHandheldActivationMode : u64 { | ||||
|         Dual = 0, | ||||
|         Single = 1, | ||||
|         None = 2, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadCommunicationMode | ||||
|     enum class NpadCommunicationMode : u64 { | ||||
|         Mode_5ms = 0, | ||||
|         Mode_10ms = 1, | ||||
| @@ -90,22 +109,74 @@ public: | ||||
|         Default = 3, | ||||
|     }; | ||||
|  | ||||
|     static constexpr Core::HID::VibrationValue DEFAULT_VIBRATION_VALUE{ | ||||
|         .low_amplitude = 0.0f, | ||||
|         .low_frequency = 160.0f, | ||||
|         .high_amplitude = 0.0f, | ||||
|         .high_frequency = 320.0f, | ||||
|     struct DeviceHandle { | ||||
|         NpadType npad_type; | ||||
|         u8 npad_id; | ||||
|         DeviceIndex device_index; | ||||
|         INSERT_PADDING_BYTES_NOINIT(1); | ||||
|     }; | ||||
|     static_assert(sizeof(DeviceHandle) == 4, "DeviceHandle is an invalid size"); | ||||
|  | ||||
|     struct NpadStyleSet { | ||||
|         union { | ||||
|             u32_le raw{}; | ||||
|  | ||||
|             BitField<0, 1, u32> fullkey; | ||||
|             BitField<1, 1, u32> handheld; | ||||
|             BitField<2, 1, u32> joycon_dual; | ||||
|             BitField<3, 1, u32> joycon_left; | ||||
|             BitField<4, 1, u32> joycon_right; | ||||
|             BitField<5, 1, u32> gamecube; | ||||
|             BitField<6, 1, u32> palma; | ||||
|             BitField<7, 1, u32> lark; | ||||
|             BitField<8, 1, u32> handheld_lark; | ||||
|             BitField<9, 1, u32> lucia; | ||||
|             BitField<29, 1, u32> system_ext; | ||||
|             BitField<30, 1, u32> system; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(NpadStyleSet) == 4, "NpadStyleSet is an invalid size"); | ||||
|  | ||||
|     struct VibrationValue { | ||||
|         f32 amp_low; | ||||
|         f32 freq_low; | ||||
|         f32 amp_high; | ||||
|         f32 freq_high; | ||||
|     }; | ||||
|     static_assert(sizeof(VibrationValue) == 0x10, "Vibration is an invalid size"); | ||||
|  | ||||
|     static constexpr VibrationValue DEFAULT_VIBRATION_VALUE{ | ||||
|         .amp_low = 0.0f, | ||||
|         .freq_low = 160.0f, | ||||
|         .amp_high = 0.0f, | ||||
|         .freq_high = 320.0f, | ||||
|     }; | ||||
|  | ||||
|     void SetSupportedStyleSet(Core::HID::NpadStyleTag style_set); | ||||
|     Core::HID::NpadStyleTag GetSupportedStyleSet() const; | ||||
|     struct LedPattern { | ||||
|         explicit LedPattern(u64 light1, u64 light2, u64 light3, u64 light4) { | ||||
|             position1.Assign(light1); | ||||
|             position2.Assign(light2); | ||||
|             position3.Assign(light3); | ||||
|             position4.Assign(light4); | ||||
|         } | ||||
|         union { | ||||
|             u64 raw{}; | ||||
|             BitField<0, 1, u64> position1; | ||||
|             BitField<1, 1, u64> position2; | ||||
|             BitField<2, 1, u64> position3; | ||||
|             BitField<3, 1, u64> position4; | ||||
|         }; | ||||
|     }; | ||||
|  | ||||
|     void SetSupportedStyleSet(NpadStyleSet style_set); | ||||
|     NpadStyleSet GetSupportedStyleSet() const; | ||||
|  | ||||
|     void SetSupportedNpadIdTypes(u8* data, std::size_t length); | ||||
|     void GetSupportedNpadIdTypes(u32* data, std::size_t max_length); | ||||
|     std::size_t GetSupportedNpadIdTypesSize() const; | ||||
|  | ||||
|     void SetHoldType(NpadJoyHoldType joy_hold_type); | ||||
|     NpadJoyHoldType GetHoldType() const; | ||||
|     void SetHoldType(NpadHoldType joy_hold_type); | ||||
|     NpadHoldType GetHoldType() const; | ||||
|  | ||||
|     void SetNpadHandheldActivationMode(NpadHandheldActivationMode activation_mode); | ||||
|     NpadHandheldActivationMode GetNpadHandheldActivationMode() const; | ||||
| @@ -113,106 +184,162 @@ public: | ||||
|     void SetNpadCommunicationMode(NpadCommunicationMode communication_mode_); | ||||
|     NpadCommunicationMode GetNpadCommunicationMode() const; | ||||
|  | ||||
|     void SetNpadMode(Core::HID::NpadIdType npad_id, NpadJoyAssignmentMode assignment_mode); | ||||
|     void SetNpadMode(u32 npad_id, NpadAssignments assignment_mode); | ||||
|  | ||||
|     bool VibrateControllerAtIndex(Core::HID::NpadIdType npad_id, std::size_t device_index, | ||||
|                                   const Core::HID::VibrationValue& vibration_value); | ||||
|     bool VibrateControllerAtIndex(std::size_t npad_index, std::size_t device_index, | ||||
|                                   const VibrationValue& vibration_value); | ||||
|  | ||||
|     void VibrateController(const Core::HID::VibrationDeviceHandle& vibration_device_handle, | ||||
|                            const Core::HID::VibrationValue& vibration_value); | ||||
|     void VibrateController(const DeviceHandle& vibration_device_handle, | ||||
|                            const VibrationValue& vibration_value); | ||||
|  | ||||
|     void VibrateControllers( | ||||
|         const std::vector<Core::HID::VibrationDeviceHandle>& vibration_device_handles, | ||||
|         const std::vector<Core::HID::VibrationValue>& vibration_values); | ||||
|     void VibrateControllers(const std::vector<DeviceHandle>& vibration_device_handles, | ||||
|                             const std::vector<VibrationValue>& vibration_values); | ||||
|  | ||||
|     Core::HID::VibrationValue GetLastVibration( | ||||
|         const Core::HID::VibrationDeviceHandle& vibration_device_handle) const; | ||||
|     VibrationValue GetLastVibration(const DeviceHandle& vibration_device_handle) const; | ||||
|  | ||||
|     void InitializeVibrationDevice(const Core::HID::VibrationDeviceHandle& vibration_device_handle); | ||||
|     void InitializeVibrationDevice(const DeviceHandle& vibration_device_handle); | ||||
|  | ||||
|     void InitializeVibrationDeviceAtIndex(Core::HID::NpadIdType npad_id, std::size_t device_index); | ||||
|     void InitializeVibrationDeviceAtIndex(std::size_t npad_index, std::size_t device_index); | ||||
|  | ||||
|     void SetPermitVibrationSession(bool permit_vibration_session); | ||||
|  | ||||
|     bool IsVibrationDeviceMounted( | ||||
|         const Core::HID::VibrationDeviceHandle& vibration_device_handle) const; | ||||
|     bool IsVibrationDeviceMounted(const DeviceHandle& vibration_device_handle) const; | ||||
|  | ||||
|     Kernel::KReadableEvent& GetStyleSetChangedEvent(Core::HID::NpadIdType npad_id); | ||||
|     void SignalStyleSetChangedEvent(Core::HID::NpadIdType npad_id) const; | ||||
|     Kernel::KReadableEvent& GetStyleSetChangedEvent(u32 npad_id); | ||||
|     void SignalStyleSetChangedEvent(u32 npad_id) const; | ||||
|  | ||||
|     // Adds a new controller at an index. | ||||
|     void AddNewControllerAt(Core::HID::NpadStyleIndex controller, Core::HID::NpadIdType npad_id); | ||||
|     void AddNewControllerAt(NPadControllerType controller, std::size_t npad_index); | ||||
|     // Adds a new controller at an index with connection status. | ||||
|     void UpdateControllerAt(Core::HID::NpadStyleIndex controller, Core::HID::NpadIdType npad_id, | ||||
|                             bool connected); | ||||
|     void UpdateControllerAt(NPadControllerType controller, std::size_t npad_index, bool connected); | ||||
|  | ||||
|     void DisconnectNpad(Core::HID::NpadIdType npad_id); | ||||
|     void DisconnectNpad(u32 npad_id); | ||||
|     void DisconnectNpadAtIndex(std::size_t index); | ||||
|  | ||||
|     void SetGyroscopeZeroDriftMode(Core::HID::SixAxisSensorHandle sixaxis_handle, | ||||
|                                    GyroscopeZeroDriftMode drift_mode); | ||||
|     GyroscopeZeroDriftMode GetGyroscopeZeroDriftMode( | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle) const; | ||||
|     bool IsSixAxisSensorAtRest(Core::HID::SixAxisSensorHandle sixaxis_handle) const; | ||||
|     void SetSixAxisEnabled(Core::HID::SixAxisSensorHandle sixaxis_handle, bool sixaxis_status); | ||||
|     void SetSixAxisFusionEnabled(Core::HID::SixAxisSensorHandle sixaxis_handle, | ||||
|                                  bool sixaxis_fusion_status); | ||||
|     void SetSixAxisFusionParameters( | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle, | ||||
|         Core::HID::SixAxisSensorFusionParameters sixaxis_fusion_parameters); | ||||
|     Core::HID::SixAxisSensorFusionParameters GetSixAxisFusionParameters( | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle); | ||||
|     void ResetSixAxisFusionParameters(Core::HID::SixAxisSensorHandle sixaxis_handle); | ||||
|     Core::HID::LedPattern GetLedPattern(Core::HID::NpadIdType npad_id); | ||||
|     bool IsUnintendedHomeButtonInputProtectionEnabled(Core::HID::NpadIdType npad_id) const; | ||||
|     void SetUnintendedHomeButtonInputProtectionEnabled(bool is_protection_enabled, | ||||
|                                                        Core::HID::NpadIdType npad_id); | ||||
|     void SetGyroscopeZeroDriftMode(GyroscopeZeroDriftMode drift_mode); | ||||
|     GyroscopeZeroDriftMode GetGyroscopeZeroDriftMode() const; | ||||
|     bool IsSixAxisSensorAtRest() const; | ||||
|     void SetSixAxisEnabled(bool six_axis_status); | ||||
|     void SetSixAxisFusionParameters(f32 parameter1, f32 parameter2); | ||||
|     std::pair<f32, f32> GetSixAxisFusionParameters(); | ||||
|     void ResetSixAxisFusionParameters(); | ||||
|     LedPattern GetLedPattern(u32 npad_id); | ||||
|     bool IsUnintendedHomeButtonInputProtectionEnabled(u32 npad_id) const; | ||||
|     void SetUnintendedHomeButtonInputProtectionEnabled(bool is_protection_enabled, u32 npad_id); | ||||
|     void SetAnalogStickUseCenterClamp(bool use_center_clamp); | ||||
|     void ClearAllConnectedControllers(); | ||||
|     void DisconnectAllConnectedControllers(); | ||||
|     void ConnectAllDisconnectedControllers(); | ||||
|     void ClearAllControllers(); | ||||
|  | ||||
|     void MergeSingleJoyAsDualJoy(Core::HID::NpadIdType npad_id_1, Core::HID::NpadIdType npad_id_2); | ||||
|     void MergeSingleJoyAsDualJoy(u32 npad_id_1, u32 npad_id_2); | ||||
|     void StartLRAssignmentMode(); | ||||
|     void StopLRAssignmentMode(); | ||||
|     bool SwapNpadAssignment(Core::HID::NpadIdType npad_id_1, Core::HID::NpadIdType npad_id_2); | ||||
|     bool SwapNpadAssignment(u32 npad_id_1, u32 npad_id_2); | ||||
|  | ||||
|     // Logical OR for all buttons presses on all controllers | ||||
|     // Specifically for cheat engine and other features. | ||||
|     u32 GetAndResetPressState(); | ||||
|  | ||||
|     static bool IsNpadIdValid(Core::HID::NpadIdType npad_id); | ||||
|     static bool IsDeviceHandleValid(const Core::HID::SixAxisSensorHandle& device_handle); | ||||
|     static bool IsDeviceHandleValid(const Core::HID::VibrationDeviceHandle& device_handle); | ||||
|     static Controller_NPad::NPadControllerType MapSettingsTypeToNPad(Settings::ControllerType type); | ||||
|     static Settings::ControllerType MapNPadToSettingsType(Controller_NPad::NPadControllerType type); | ||||
|     static std::size_t NPadIdToIndex(u32 npad_id); | ||||
|     static u32 IndexToNPad(std::size_t index); | ||||
|     static bool IsNpadIdValid(u32 npad_id); | ||||
|     static bool IsDeviceHandleValid(const DeviceHandle& device_handle); | ||||
|  | ||||
| private: | ||||
|     // This is nn::hid::detail::ColorAttribute | ||||
|     enum class ColorAttribute : u32 { | ||||
|     struct CommonHeader { | ||||
|         s64_le timestamp; | ||||
|         s64_le total_entry_count; | ||||
|         s64_le last_entry_index; | ||||
|         s64_le entry_count; | ||||
|     }; | ||||
|     static_assert(sizeof(CommonHeader) == 0x20, "CommonHeader is an invalid size"); | ||||
|  | ||||
|     enum class ColorAttributes : u32_le { | ||||
|         Ok = 0, | ||||
|         ReadError = 1, | ||||
|         NoController = 2, | ||||
|     }; | ||||
|     static_assert(sizeof(ColorAttribute) == 4, "ColorAttribute is an invalid size"); | ||||
|     static_assert(sizeof(ColorAttributes) == 4, "ColorAttributes is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::NpadFullKeyColorState | ||||
|     struct NpadFullKeyColorState { | ||||
|         ColorAttribute attribute; | ||||
|         Core::HID::NpadControllerColor fullkey; | ||||
|     struct ControllerColor { | ||||
|         u32_le body; | ||||
|         u32_le button; | ||||
|     }; | ||||
|     static_assert(sizeof(NpadFullKeyColorState) == 0xC, "NpadFullKeyColorState is an invalid size"); | ||||
|     static_assert(sizeof(ControllerColor) == 8, "ControllerColor is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::NpadJoyColorState | ||||
|     struct NpadJoyColorState { | ||||
|         ColorAttribute attribute; | ||||
|         Core::HID::NpadControllerColor left; | ||||
|         Core::HID::NpadControllerColor right; | ||||
|     struct FullKeyColor { | ||||
|         ColorAttributes attribute; | ||||
|         ControllerColor fullkey; | ||||
|     }; | ||||
|     static_assert(sizeof(NpadJoyColorState) == 0x14, "NpadJoyColorState is an invalid size"); | ||||
|     static_assert(sizeof(FullKeyColor) == 0xC, "FullKeyColor is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::NpadAttribute | ||||
|     struct NpadAttribute { | ||||
|     struct JoyconColor { | ||||
|         ColorAttributes attribute; | ||||
|         ControllerColor left; | ||||
|         ControllerColor right; | ||||
|     }; | ||||
|     static_assert(sizeof(JoyconColor) == 0x14, "JoyconColor is an invalid size"); | ||||
|  | ||||
|     struct ControllerPadState { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u64_le raw{}; | ||||
|             // Button states | ||||
|             BitField<0, 1, u64> a; | ||||
|             BitField<1, 1, u64> b; | ||||
|             BitField<2, 1, u64> x; | ||||
|             BitField<3, 1, u64> y; | ||||
|             BitField<4, 1, u64> l_stick; | ||||
|             BitField<5, 1, u64> r_stick; | ||||
|             BitField<6, 1, u64> l; | ||||
|             BitField<7, 1, u64> r; | ||||
|             BitField<8, 1, u64> zl; | ||||
|             BitField<9, 1, u64> zr; | ||||
|             BitField<10, 1, u64> plus; | ||||
|             BitField<11, 1, u64> minus; | ||||
|  | ||||
|             // D-Pad | ||||
|             BitField<12, 1, u64> d_left; | ||||
|             BitField<13, 1, u64> d_up; | ||||
|             BitField<14, 1, u64> d_right; | ||||
|             BitField<15, 1, u64> d_down; | ||||
|  | ||||
|             // Left JoyStick | ||||
|             BitField<16, 1, u64> l_stick_left; | ||||
|             BitField<17, 1, u64> l_stick_up; | ||||
|             BitField<18, 1, u64> l_stick_right; | ||||
|             BitField<19, 1, u64> l_stick_down; | ||||
|  | ||||
|             // Right JoyStick | ||||
|             BitField<20, 1, u64> r_stick_left; | ||||
|             BitField<21, 1, u64> r_stick_up; | ||||
|             BitField<22, 1, u64> r_stick_right; | ||||
|             BitField<23, 1, u64> r_stick_down; | ||||
|  | ||||
|             // Not always active? | ||||
|             BitField<24, 1, u64> left_sl; | ||||
|             BitField<25, 1, u64> left_sr; | ||||
|  | ||||
|             BitField<26, 1, u64> right_sl; | ||||
|             BitField<27, 1, u64> right_sr; | ||||
|  | ||||
|             BitField<28, 1, u64> palma; | ||||
|             BitField<30, 1, u64> handheld_left_b; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(ControllerPadState) == 8, "ControllerPadState is an invalid size"); | ||||
|  | ||||
|     struct AnalogPosition { | ||||
|         s32_le x; | ||||
|         s32_le y; | ||||
|     }; | ||||
|     static_assert(sizeof(AnalogPosition) == 8, "AnalogPosition is an invalid size"); | ||||
|  | ||||
|     struct ConnectionState { | ||||
|         union { | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> is_connected; | ||||
|             BitField<1, 1, u32> is_wired; | ||||
|             BitField<2, 1, u32> is_left_connected; | ||||
| @@ -221,60 +348,79 @@ private: | ||||
|             BitField<5, 1, u32> is_right_wired; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(NpadAttribute) == 4, "NpadAttribute is an invalid size"); | ||||
|     static_assert(sizeof(ConnectionState) == 4, "ConnectionState is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::NpadFullKeyState | ||||
|     // This is nn::hid::NpadHandheldState | ||||
|     // This is nn::hid::NpadJoyDualState | ||||
|     // This is nn::hid::NpadJoyLeftState | ||||
|     // This is nn::hid::NpadJoyRightState | ||||
|     // This is nn::hid::NpadPalmaState | ||||
|     // This is nn::hid::NpadSystemExtState | ||||
|     struct NPadGenericState { | ||||
|         s64_le sampling_number; | ||||
|         Core::HID::NpadButtonState npad_buttons; | ||||
|         Core::HID::AnalogStickState l_stick; | ||||
|         Core::HID::AnalogStickState r_stick; | ||||
|         NpadAttribute connection_status; | ||||
|         INSERT_PADDING_BYTES(4); // Reserved | ||||
|     struct ControllerPad { | ||||
|         ControllerPadState pad_states; | ||||
|         AnalogPosition l_stick; | ||||
|         AnalogPosition r_stick; | ||||
|     }; | ||||
|     static_assert(sizeof(NPadGenericState) == 0x28, "NPadGenericState is an invalid size"); | ||||
|     static_assert(sizeof(ControllerPad) == 0x18, "ControllerPad is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::SixAxisSensorAttribute | ||||
|     struct SixAxisSensorAttribute { | ||||
|     struct GenericStates { | ||||
|         s64_le timestamp; | ||||
|         s64_le timestamp2; | ||||
|         ControllerPad pad; | ||||
|         ConnectionState connection_status; | ||||
|     }; | ||||
|     static_assert(sizeof(GenericStates) == 0x30, "NPadGenericStates is an invalid size"); | ||||
|  | ||||
|     struct NPadGeneric { | ||||
|         CommonHeader common; | ||||
|         std::array<GenericStates, 17> npad; | ||||
|     }; | ||||
|     static_assert(sizeof(NPadGeneric) == 0x350, "NPadGeneric is an invalid size"); | ||||
|  | ||||
|     struct SixAxisAttributes { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> is_connected; | ||||
|             BitField<1, 1, u32> is_interpolated; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(SixAxisSensorAttribute) == 4, "SixAxisSensorAttribute is an invalid size"); | ||||
|     static_assert(sizeof(SixAxisAttributes) == 4, "SixAxisAttributes is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::SixAxisSensorState | ||||
|     struct SixAxisSensorState { | ||||
|         s64 delta_time{}; | ||||
|         s64 sampling_number{}; | ||||
|     struct SixAxisStates { | ||||
|         s64_le timestamp{}; | ||||
|         INSERT_PADDING_WORDS(2); | ||||
|         s64_le timestamp2{}; | ||||
|         Common::Vec3f accel{}; | ||||
|         Common::Vec3f gyro{}; | ||||
|         Common::Vec3f rotation{}; | ||||
|         std::array<Common::Vec3f, 3> orientation{}; | ||||
|         SixAxisSensorAttribute attribute; | ||||
|         SixAxisAttributes attribute; | ||||
|         INSERT_PADDING_BYTES(4); // Reserved | ||||
|     }; | ||||
|     static_assert(sizeof(SixAxisSensorState) == 0x60, "SixAxisSensorState is an invalid size"); | ||||
|     static_assert(sizeof(SixAxisStates) == 0x68, "SixAxisStates is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::server::NpadGcTriggerState | ||||
|     struct NpadGcTriggerState { | ||||
|         s64 sampling_number{}; | ||||
|         s32 l_analog{}; | ||||
|         s32 r_analog{}; | ||||
|     struct SixAxisGeneric { | ||||
|         CommonHeader common{}; | ||||
|         std::array<SixAxisStates, 17> sixaxis{}; | ||||
|     }; | ||||
|     static_assert(sizeof(NpadGcTriggerState) == 0x10, "NpadGcTriggerState is an invalid size"); | ||||
|     static_assert(sizeof(SixAxisGeneric) == 0x708, "SixAxisGeneric is an invalid size"); | ||||
|  | ||||
|     struct TriggerState { | ||||
|         s64_le timestamp{}; | ||||
|         s64_le timestamp2{}; | ||||
|         s32_le l_analog{}; | ||||
|         s32_le r_analog{}; | ||||
|     }; | ||||
|     static_assert(sizeof(TriggerState) == 0x18, "TriggerState is an invalid size"); | ||||
|  | ||||
|     struct TriggerGeneric { | ||||
|         INSERT_PADDING_BYTES(0x4); | ||||
|         s64_le timestamp; | ||||
|         INSERT_PADDING_BYTES(0x4); | ||||
|         s64_le total_entry_count; | ||||
|         s64_le last_entry_index; | ||||
|         s64_le entry_count; | ||||
|         std::array<TriggerState, 17> trigger{}; | ||||
|     }; | ||||
|     static_assert(sizeof(TriggerGeneric) == 0x1C8, "TriggerGeneric is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::NpadSystemProperties | ||||
|     struct NPadSystemProperties { | ||||
|         union { | ||||
|             s64 raw{}; | ||||
|             s64_le raw{}; | ||||
|             BitField<0, 1, s64> is_charging_joy_dual; | ||||
|             BitField<1, 1, s64> is_charging_joy_left; | ||||
|             BitField<2, 1, s64> is_charging_joy_right; | ||||
| @@ -292,20 +438,17 @@ private: | ||||
|     }; | ||||
|     static_assert(sizeof(NPadSystemProperties) == 0x8, "NPadSystemProperties is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::NpadSystemButtonProperties | ||||
|     struct NpadSystemButtonProperties { | ||||
|     struct NPadButtonProperties { | ||||
|         union { | ||||
|             s32 raw{}; | ||||
|             s32_le raw{}; | ||||
|             BitField<0, 1, s32> is_home_button_protection_enabled; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(NpadSystemButtonProperties) == 0x4, | ||||
|                   "NPadButtonProperties is an invalid size"); | ||||
|     static_assert(sizeof(NPadButtonProperties) == 0x4, "NPadButtonProperties is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::system::DeviceType | ||||
|     struct DeviceType { | ||||
|     struct NPadDevice { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, s32> fullkey; | ||||
|             BitField<1, 1, s32> debug_pad; | ||||
|             BitField<2, 1, s32> handheld_left; | ||||
| @@ -322,49 +465,26 @@ private: | ||||
|             BitField<13, 1, s32> handheld_lark_nes_left; | ||||
|             BitField<14, 1, s32> handheld_lark_nes_right; | ||||
|             BitField<15, 1, s32> lucia; | ||||
|             BitField<16, 1, s32> lagon; | ||||
|             BitField<17, 1, s32> lager; | ||||
|             BitField<31, 1, s32> system; | ||||
|         }; | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::detail::NfcXcdDeviceHandleStateImpl | ||||
|     struct NfcXcdDeviceHandleStateImpl { | ||||
|         u64 handle; | ||||
|         bool is_available; | ||||
|         bool is_activated; | ||||
|         INSERT_PADDING_BYTES(0x6); // Reserved | ||||
|         u64 sampling_number; | ||||
|     struct MotionDevice { | ||||
|         Common::Vec3f accel; | ||||
|         Common::Vec3f gyro; | ||||
|         Common::Vec3f rotation; | ||||
|         std::array<Common::Vec3f, 3> orientation; | ||||
|         Common::Quaternion<f32> quaternion; | ||||
|     }; | ||||
|     static_assert(sizeof(NfcXcdDeviceHandleStateImpl) == 0x18, | ||||
|                   "NfcXcdDeviceHandleStateImpl is an invalid size"); | ||||
|  | ||||
|     // nn::hid::detail::NfcXcdDeviceHandleStateImplAtomicStorage | ||||
|     struct NfcXcdDeviceHandleStateImplAtomicStorage { | ||||
|         u64 sampling_number; | ||||
|         NfcXcdDeviceHandleStateImpl nfc_xcd_device_handle_state; | ||||
|     struct NfcXcdHandle { | ||||
|         INSERT_PADDING_BYTES(0x60); | ||||
|     }; | ||||
|     static_assert(sizeof(NfcXcdDeviceHandleStateImplAtomicStorage) == 0x20, | ||||
|                   "NfcXcdDeviceHandleStateImplAtomicStorage is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::NfcXcdDeviceHandleState | ||||
|     struct NfcXcdDeviceHandleState { | ||||
|         // TODO(german77): Make this struct a ring lifo object | ||||
|         INSERT_PADDING_BYTES(0x8); // Unused | ||||
|         s64 total_buffer_count = max_buffer_size; | ||||
|         s64 buffer_tail{}; | ||||
|         s64 buffer_count{}; | ||||
|         std::array<NfcXcdDeviceHandleStateImplAtomicStorage, 2> nfc_xcd_device_handle_storage; | ||||
|     }; | ||||
|     static_assert(sizeof(NfcXcdDeviceHandleState) == 0x60, | ||||
|                   "NfcXcdDeviceHandleState is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::system::AppletFooterUiAttributesSet | ||||
|     struct AppletFooterUiAttributes { | ||||
|         INSERT_PADDING_BYTES(0x4); | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::system::AppletFooterUiType | ||||
|     enum class AppletFooterUiType : u8 { | ||||
|         None = 0, | ||||
|         HandheldNone = 1, | ||||
| @@ -390,150 +510,95 @@ private: | ||||
|         Lagon = 21, | ||||
|     }; | ||||
|  | ||||
|     struct AppletFooterUi { | ||||
|         AppletFooterUiAttributes attributes; | ||||
|         AppletFooterUiType type; | ||||
|         INSERT_PADDING_BYTES(0x5B); // Reserved | ||||
|     }; | ||||
|     static_assert(sizeof(AppletFooterUi) == 0x60, "AppletFooterUi is an invalid size"); | ||||
|     struct NPadEntry { | ||||
|         NpadStyleSet style_set; | ||||
|         NpadAssignments assignment_mode; | ||||
|         FullKeyColor fullkey_color; | ||||
|         JoyconColor joycon_color; | ||||
|  | ||||
|     // This is nn::hid::NpadLarkType | ||||
|     enum class NpadLarkType : u32 { | ||||
|         Invalid, | ||||
|         H1, | ||||
|         H2, | ||||
|         NL, | ||||
|         NR, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadLuciaType | ||||
|     enum class NpadLuciaType : u32 { | ||||
|         Invalid, | ||||
|         J, | ||||
|         E, | ||||
|         U, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadLagonType | ||||
|     enum class NpadLagonType : u32 { | ||||
|         Invalid, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::NpadLagerType | ||||
|     enum class NpadLagerType : u32 { | ||||
|         Invalid, | ||||
|         J, | ||||
|         E, | ||||
|         U, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::detail::NpadInternalState | ||||
|     struct NpadInternalState { | ||||
|         Core::HID::NpadStyleTag style_tag; | ||||
|         NpadJoyAssignmentMode assignment_mode; | ||||
|         NpadFullKeyColorState fullkey_color; | ||||
|         NpadJoyColorState joycon_color; | ||||
|         Lifo<NPadGenericState> fullkey_lifo; | ||||
|         Lifo<NPadGenericState> handheld_lifo; | ||||
|         Lifo<NPadGenericState> joy_dual_lifo; | ||||
|         Lifo<NPadGenericState> joy_left_lifo; | ||||
|         Lifo<NPadGenericState> joy_right_lifo; | ||||
|         Lifo<NPadGenericState> palma_lifo; | ||||
|         Lifo<NPadGenericState> system_ext_lifo; | ||||
|         Lifo<SixAxisSensorState> sixaxis_fullkey_lifo; | ||||
|         Lifo<SixAxisSensorState> sixaxis_handheld_lifo; | ||||
|         Lifo<SixAxisSensorState> sixaxis_dual_left_lifo; | ||||
|         Lifo<SixAxisSensorState> sixaxis_dual_right_lifo; | ||||
|         Lifo<SixAxisSensorState> sixaxis_left_lifo; | ||||
|         Lifo<SixAxisSensorState> sixaxis_right_lifo; | ||||
|         DeviceType device_type; | ||||
|         INSERT_PADDING_BYTES(0x4); // Reserved | ||||
|         NPadGeneric fullkey_states; | ||||
|         NPadGeneric handheld_states; | ||||
|         NPadGeneric joy_dual_states; | ||||
|         NPadGeneric joy_left_states; | ||||
|         NPadGeneric joy_right_states; | ||||
|         NPadGeneric palma_states; | ||||
|         NPadGeneric system_ext_states; | ||||
|         SixAxisGeneric sixaxis_fullkey; | ||||
|         SixAxisGeneric sixaxis_handheld; | ||||
|         SixAxisGeneric sixaxis_dual_left; | ||||
|         SixAxisGeneric sixaxis_dual_right; | ||||
|         SixAxisGeneric sixaxis_left; | ||||
|         SixAxisGeneric sixaxis_right; | ||||
|         NPadDevice device_type; | ||||
|         INSERT_PADDING_BYTES(0x4); // reserved | ||||
|         NPadSystemProperties system_properties; | ||||
|         NpadSystemButtonProperties button_properties; | ||||
|         Core::HID::BatteryLevel battery_level_dual; | ||||
|         Core::HID::BatteryLevel battery_level_left; | ||||
|         Core::HID::BatteryLevel battery_level_right; | ||||
|         union { | ||||
|             NfcXcdDeviceHandleState nfc_xcd_device_handle; | ||||
|             AppletFooterUi applet_footer; | ||||
|         }; | ||||
|         INSERT_PADDING_BYTES(0x20); // Unknown | ||||
|         Lifo<NpadGcTriggerState> gc_trigger_lifo; | ||||
|         NpadLarkType lark_type_l_and_main; | ||||
|         NpadLarkType lark_type_r; | ||||
|         NpadLuciaType lucia_type; | ||||
|         NpadLagonType lagon_type; | ||||
|         NpadLagerType lager_type; | ||||
|         // FW 13.x Investigate there is some sort of bitflag related to joycons | ||||
|         INSERT_PADDING_BYTES(0x4); | ||||
|         INSERT_PADDING_BYTES(0xc08); // Unknown | ||||
|         NPadButtonProperties button_properties; | ||||
|         u32 battery_level_dual; | ||||
|         u32 battery_level_left; | ||||
|         u32 battery_level_right; | ||||
|         AppletFooterUiAttributes footer_attributes; | ||||
|         AppletFooterUiType footer_type; | ||||
|         // nfc_states needs to be checked switchbrew does not match with HW | ||||
|         NfcXcdHandle nfc_states; | ||||
|         INSERT_PADDING_BYTES(0x8); // Mutex | ||||
|         TriggerGeneric gc_trigger_states; | ||||
|         INSERT_PADDING_BYTES(0xc1f); | ||||
|     }; | ||||
|     static_assert(sizeof(NpadInternalState) == 0x5000, "NpadInternalState is an invalid size"); | ||||
|     static_assert(sizeof(NPadEntry) == 0x5000, "NPadEntry is an invalid size"); | ||||
|  | ||||
|     struct VibrationData { | ||||
|         bool device_mounted{}; | ||||
|         Core::HID::VibrationValue latest_vibration_value{}; | ||||
|         std::chrono::steady_clock::time_point last_vibration_timepoint{}; | ||||
|     struct ControllerHolder { | ||||
|         NPadControllerType type; | ||||
|         bool is_connected; | ||||
|     }; | ||||
|  | ||||
|     struct NpadControllerData { | ||||
|         Core::HID::EmulatedController* device; | ||||
|         Kernel::KEvent* styleset_changed_event{}; | ||||
|         NpadInternalState shared_memory_entry{}; | ||||
|  | ||||
|         std::array<VibrationData, 2> vibration{}; | ||||
|         bool unintended_home_button_input_protection{}; | ||||
|         bool is_connected{}; | ||||
|         Core::HID::NpadStyleIndex npad_type{Core::HID::NpadStyleIndex::None}; | ||||
|  | ||||
|         // Motion parameters | ||||
|         bool sixaxis_at_rest{true}; | ||||
|         bool sixaxis_sensor_enabled{true}; | ||||
|         bool sixaxis_fusion_enabled{false}; | ||||
|         Core::HID::SixAxisSensorFusionParameters sixaxis_fusion{}; | ||||
|         GyroscopeZeroDriftMode gyroscope_zero_drift_mode{GyroscopeZeroDriftMode::Standard}; | ||||
|  | ||||
|         // Current pad state | ||||
|         NPadGenericState npad_pad_state{}; | ||||
|         NPadGenericState npad_libnx_state{}; | ||||
|         NpadGcTriggerState npad_trigger_state{}; | ||||
|         SixAxisSensorState sixaxis_fullkey_state{}; | ||||
|         SixAxisSensorState sixaxis_handheld_state{}; | ||||
|         SixAxisSensorState sixaxis_dual_left_state{}; | ||||
|         SixAxisSensorState sixaxis_dual_right_state{}; | ||||
|         SixAxisSensorState sixaxis_left_lifo_state{}; | ||||
|         SixAxisSensorState sixaxis_right_lifo_state{}; | ||||
|         int callback_key; | ||||
|     }; | ||||
|  | ||||
|     void ControllerUpdate(Core::HID::ControllerTriggerType type, std::size_t controller_idx); | ||||
|     void InitNewlyAddedController(Core::HID::NpadIdType npad_id); | ||||
|     bool IsControllerSupported(Core::HID::NpadStyleIndex controller) const; | ||||
|     void RequestPadStateUpdate(Core::HID::NpadIdType npad_id); | ||||
|     void WriteEmptyEntry(NpadInternalState& npad); | ||||
|  | ||||
|     NpadControllerData& GetControllerFromHandle( | ||||
|         const Core::HID::SixAxisSensorHandle& device_handle); | ||||
|     const NpadControllerData& GetControllerFromHandle( | ||||
|         const Core::HID::SixAxisSensorHandle& device_handle) const; | ||||
|     NpadControllerData& GetControllerFromHandle( | ||||
|         const Core::HID::VibrationDeviceHandle& device_handle); | ||||
|     const NpadControllerData& GetControllerFromHandle( | ||||
|         const Core::HID::VibrationDeviceHandle& device_handle) const; | ||||
|     NpadControllerData& GetControllerFromNpadIdType(Core::HID::NpadIdType npad_id); | ||||
|     const NpadControllerData& GetControllerFromNpadIdType(Core::HID::NpadIdType npad_id) const; | ||||
|     void InitNewlyAddedController(std::size_t controller_idx); | ||||
|     bool IsControllerSupported(NPadControllerType controller) const; | ||||
|     void RequestPadStateUpdate(u32 npad_id); | ||||
|  | ||||
|     std::atomic<u32> press_state{}; | ||||
|  | ||||
|     std::array<NpadControllerData, 10> controller_data{}; | ||||
|     NpadStyleSet style{}; | ||||
|     std::array<NPadEntry, 10> shared_memory_entries{}; | ||||
|     using ButtonArray = std::array< | ||||
|         std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeButton::NUM_BUTTONS_HID>, | ||||
|         10>; | ||||
|     using StickArray = std::array< | ||||
|         std::array<std::unique_ptr<Input::AnalogDevice>, Settings::NativeAnalog::NUM_STICKS_HID>, | ||||
|         10>; | ||||
|     using VibrationArray = std::array<std::array<std::unique_ptr<Input::VibrationDevice>, | ||||
|                                                  Settings::NativeVibration::NUM_VIBRATIONS_HID>, | ||||
|                                       10>; | ||||
|     using MotionArray = std::array< | ||||
|         std::array<std::unique_ptr<Input::MotionDevice>, Settings::NativeMotion::NUM_MOTIONS_HID>, | ||||
|         10>; | ||||
|  | ||||
|     KernelHelpers::ServiceContext& service_context; | ||||
|     std::mutex mutex; | ||||
|     std::vector<Core::HID::NpadIdType> supported_npad_id_types{}; | ||||
|     NpadJoyHoldType hold_type{NpadJoyHoldType::Vertical}; | ||||
|     ButtonArray buttons; | ||||
|     StickArray sticks; | ||||
|     VibrationArray vibrations; | ||||
|     MotionArray motions; | ||||
|     std::vector<u32> supported_npad_id_types{}; | ||||
|     NpadHoldType hold_type{NpadHoldType::Vertical}; | ||||
|     NpadHandheldActivationMode handheld_activation_mode{NpadHandheldActivationMode::Dual}; | ||||
|     NpadCommunicationMode communication_mode{NpadCommunicationMode::Default}; | ||||
|     // Each controller should have their own styleset changed event | ||||
|     std::array<Kernel::KEvent*, 10> styleset_changed_events{}; | ||||
|     std::array<std::array<std::chrono::steady_clock::time_point, 2>, 10> | ||||
|         last_vibration_timepoints{}; | ||||
|     std::array<std::array<VibrationValue, 2>, 10> latest_vibration_values{}; | ||||
|     bool permit_vibration_session_enabled{false}; | ||||
|     std::array<std::array<bool, 2>, 10> vibration_devices_mounted{}; | ||||
|     std::array<ControllerHolder, 10> connected_controllers{}; | ||||
|     std::array<bool, 10> unintended_home_button_input_protection{}; | ||||
|     bool analog_stick_use_center_clamp{}; | ||||
|     GyroscopeZeroDriftMode gyroscope_zero_drift_mode{GyroscopeZeroDriftMode::Standard}; | ||||
|     bool sixaxis_sensors_enabled{true}; | ||||
|     f32 sixaxis_fusion_parameter1{}; | ||||
|     f32 sixaxis_fusion_parameter2{}; | ||||
|     bool sixaxis_at_rest{true}; | ||||
|     std::array<ControllerPad, 10> npad_pad_states{}; | ||||
|     std::array<TriggerState, 10> npad_trigger_states{}; | ||||
|     bool is_in_lr_assignment_mode{false}; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -5,12 +5,11 @@ | ||||
| #include <cstring> | ||||
| #include "common/common_types.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/hid/controllers/stubbed.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
|  | ||||
| Controller_Stubbed::Controller_Stubbed(Core::HID::HIDCore& hid_core_) : ControllerBase{hid_core_} {} | ||||
| Controller_Stubbed::Controller_Stubbed(Core::System& system_) : ControllerBase{system_} {} | ||||
| Controller_Stubbed::~Controller_Stubbed() = default; | ||||
|  | ||||
| void Controller_Stubbed::OnInit() {} | ||||
| @@ -32,9 +31,10 @@ void Controller_Stubbed::OnUpdate(const Core::Timing::CoreTiming& core_timing, u | ||||
|     std::memcpy(data + common_offset, &header, sizeof(CommonHeader)); | ||||
| } | ||||
|  | ||||
| void Controller_Stubbed::OnLoadInputDevices() {} | ||||
|  | ||||
| void Controller_Stubbed::SetCommonHeaderOffset(std::size_t off) { | ||||
|     common_offset = off; | ||||
|     smart_update = true; | ||||
| } | ||||
|  | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -10,7 +10,7 @@ | ||||
| namespace Service::HID { | ||||
| class Controller_Stubbed final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_Stubbed(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_Stubbed(Core::System& system_); | ||||
|     ~Controller_Stubbed() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -22,17 +22,12 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
|     void SetCommonHeaderOffset(std::size_t off); | ||||
|  | ||||
| private: | ||||
|     struct CommonHeader { | ||||
|         s64 timestamp; | ||||
|         s64 total_entry_count; | ||||
|         s64 last_entry_index; | ||||
|         s64 entry_count; | ||||
|     }; | ||||
|     static_assert(sizeof(CommonHeader) == 0x20, "CommonHeader is an invalid size"); | ||||
|  | ||||
|     bool smart_update{}; | ||||
|     std::size_t common_offset{}; | ||||
| }; | ||||
|   | ||||
| @@ -7,82 +7,72 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "common/logging/log.h" | ||||
| #include "common/settings.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/frontend/emu_window.h" | ||||
| #include "core/hid/emulated_console.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/touchscreen.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| constexpr std::size_t SHARED_MEMORY_OFFSET = 0x400; | ||||
|  | ||||
| Controller_Touchscreen::Controller_Touchscreen(Core::HID::HIDCore& hid_core_) | ||||
|     : ControllerBase{hid_core_} { | ||||
|     console = hid_core.GetEmulatedConsole(); | ||||
| } | ||||
|  | ||||
| Controller_Touchscreen::Controller_Touchscreen(Core::System& system_) : ControllerBase{system_} {} | ||||
| Controller_Touchscreen::~Controller_Touchscreen() = default; | ||||
|  | ||||
| void Controller_Touchscreen::OnInit() {} | ||||
| void Controller_Touchscreen::OnInit() { | ||||
|     for (std::size_t id = 0; id < MAX_FINGERS; ++id) { | ||||
|         mouse_finger_id[id] = MAX_FINGERS; | ||||
|         keyboard_finger_id[id] = MAX_FINGERS; | ||||
|         udp_finger_id[id] = MAX_FINGERS; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void Controller_Touchscreen::OnRelease() {} | ||||
|  | ||||
| void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                       std::size_t size) { | ||||
|     touch_screen_lifo.timestamp = core_timing.GetCPUTicks(); | ||||
|     shared_memory.header.timestamp = core_timing.GetCPUTicks(); | ||||
|     shared_memory.header.total_entry_count = 17; | ||||
|  | ||||
|     if (!IsControllerActivated()) { | ||||
|         touch_screen_lifo.buffer_count = 0; | ||||
|         touch_screen_lifo.buffer_tail = 0; | ||||
|         std::memcpy(data, &touch_screen_lifo, sizeof(touch_screen_lifo)); | ||||
|         shared_memory.header.entry_count = 0; | ||||
|         shared_memory.header.last_entry_index = 0; | ||||
|         return; | ||||
|     } | ||||
|     shared_memory.header.entry_count = 16; | ||||
|  | ||||
|     const auto touch_status = console->GetTouch(); | ||||
|     for (std::size_t id = 0; id < MAX_FINGERS; id++) { | ||||
|         const auto& current_touch = touch_status[id]; | ||||
|         auto& finger = fingers[id]; | ||||
|         finger.position = current_touch.position; | ||||
|         finger.id = current_touch.id; | ||||
|     const auto& last_entry = | ||||
|         shared_memory.shared_memory_entries[shared_memory.header.last_entry_index]; | ||||
|     shared_memory.header.last_entry_index = (shared_memory.header.last_entry_index + 1) % 17; | ||||
|     auto& cur_entry = shared_memory.shared_memory_entries[shared_memory.header.last_entry_index]; | ||||
|  | ||||
|         if (finger.attribute.start_touch) { | ||||
|             finger.attribute.raw = 0; | ||||
|             continue; | ||||
|         } | ||||
|     cur_entry.sampling_number = last_entry.sampling_number + 1; | ||||
|     cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|  | ||||
|         if (finger.attribute.end_touch) { | ||||
|             finger.attribute.raw = 0; | ||||
|             finger.pressed = false; | ||||
|             continue; | ||||
|         } | ||||
|     const Input::TouchStatus& mouse_status = touch_mouse_device->GetStatus(); | ||||
|     const Input::TouchStatus& udp_status = touch_udp_device->GetStatus(); | ||||
|     for (std::size_t id = 0; id < mouse_status.size(); ++id) { | ||||
|         mouse_finger_id[id] = UpdateTouchInputEvent(mouse_status[id], mouse_finger_id[id]); | ||||
|         udp_finger_id[id] = UpdateTouchInputEvent(udp_status[id], udp_finger_id[id]); | ||||
|     } | ||||
|  | ||||
|         if (!finger.pressed && current_touch.pressed) { | ||||
|             finger.attribute.start_touch.Assign(1); | ||||
|             finger.pressed = true; | ||||
|             continue; | ||||
|         } | ||||
|  | ||||
|         if (finger.pressed && !current_touch.pressed) { | ||||
|             finger.attribute.raw = 0; | ||||
|             finger.attribute.end_touch.Assign(1); | ||||
|     if (Settings::values.use_touch_from_button) { | ||||
|         const Input::TouchStatus& keyboard_status = touch_btn_device->GetStatus(); | ||||
|         for (std::size_t id = 0; id < mouse_status.size(); ++id) { | ||||
|             keyboard_finger_id[id] = | ||||
|                 UpdateTouchInputEvent(keyboard_status[id], keyboard_finger_id[id]); | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     std::array<Core::HID::TouchFinger, MAX_FINGERS> active_fingers; | ||||
|     std::array<Finger, 16> active_fingers; | ||||
|     const auto end_iter = std::copy_if(fingers.begin(), fingers.end(), active_fingers.begin(), | ||||
|                                        [](const auto& finger) { return finger.pressed; }); | ||||
|     const auto active_fingers_count = | ||||
|         static_cast<std::size_t>(std::distance(active_fingers.begin(), end_iter)); | ||||
|  | ||||
|     const u64 tick = core_timing.GetCPUTicks(); | ||||
|     const auto& last_entry = touch_screen_lifo.ReadCurrentEntry().state; | ||||
|  | ||||
|     next_state.sampling_number = last_entry.sampling_number + 1; | ||||
|     next_state.entry_count = static_cast<s32>(active_fingers_count); | ||||
|  | ||||
|     cur_entry.entry_count = static_cast<s32_le>(active_fingers_count); | ||||
|     for (std::size_t id = 0; id < MAX_FINGERS; ++id) { | ||||
|         auto& touch_entry = next_state.states[id]; | ||||
|         auto& touch_entry = cur_entry.states[id]; | ||||
|         if (id < active_fingers_count) { | ||||
|             const auto& [active_x, active_y] = active_fingers[id].position; | ||||
|             touch_entry.position = { | ||||
| @@ -107,9 +97,66 @@ void Controller_Touchscreen::OnUpdate(const Core::Timing::CoreTiming& core_timin | ||||
|             touch_entry.finger = 0; | ||||
|         } | ||||
|     } | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &shared_memory, sizeof(TouchScreenSharedMemory)); | ||||
| } | ||||
|  | ||||
|     touch_screen_lifo.WriteNextEntry(next_state); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &touch_screen_lifo, sizeof(touch_screen_lifo)); | ||||
| void Controller_Touchscreen::OnLoadInputDevices() { | ||||
|     touch_mouse_device = Input::CreateDevice<Input::TouchDevice>("engine:emu_window"); | ||||
|     touch_udp_device = Input::CreateDevice<Input::TouchDevice>("engine:cemuhookudp"); | ||||
|     touch_btn_device = Input::CreateDevice<Input::TouchDevice>("engine:touch_from_button"); | ||||
| } | ||||
|  | ||||
| std::optional<std::size_t> Controller_Touchscreen::GetUnusedFingerID() const { | ||||
|     // Dont assign any touch input to a finger if disabled | ||||
|     if (!Settings::values.touchscreen.enabled) { | ||||
|         return std::nullopt; | ||||
|     } | ||||
|     std::size_t first_free_id = 0; | ||||
|     while (first_free_id < MAX_FINGERS) { | ||||
|         if (!fingers[first_free_id].pressed) { | ||||
|             return first_free_id; | ||||
|         } else { | ||||
|             first_free_id++; | ||||
|         } | ||||
|     } | ||||
|     return std::nullopt; | ||||
| } | ||||
|  | ||||
| std::size_t Controller_Touchscreen::UpdateTouchInputEvent( | ||||
|     const std::tuple<float, float, bool>& touch_input, std::size_t finger_id) { | ||||
|     const auto& [x, y, pressed] = touch_input; | ||||
|     if (finger_id > MAX_FINGERS) { | ||||
|         LOG_ERROR(Service_HID, "Invalid finger id {}", finger_id); | ||||
|         return MAX_FINGERS; | ||||
|     } | ||||
|     if (pressed) { | ||||
|         Attributes attribute{}; | ||||
|         if (finger_id == MAX_FINGERS) { | ||||
|             const auto first_free_id = GetUnusedFingerID(); | ||||
|             if (!first_free_id) { | ||||
|                 // Invalid finger id do nothing | ||||
|                 return MAX_FINGERS; | ||||
|             } | ||||
|             finger_id = first_free_id.value(); | ||||
|             fingers[finger_id].pressed = true; | ||||
|             fingers[finger_id].id = static_cast<u32_le>(finger_id); | ||||
|             attribute.start_touch.Assign(1); | ||||
|         } | ||||
|         fingers[finger_id].position = {x, y}; | ||||
|         fingers[finger_id].attribute = attribute; | ||||
|         return finger_id; | ||||
|     } | ||||
|  | ||||
|     if (finger_id != MAX_FINGERS) { | ||||
|         if (!fingers[finger_id].attribute.end_touch) { | ||||
|             fingers[finger_id].attribute.end_touch.Assign(1); | ||||
|             fingers[finger_id].attribute.start_touch.Assign(0); | ||||
|             return finger_id; | ||||
|         } | ||||
|         fingers[finger_id].pressed = false; | ||||
|     } | ||||
|  | ||||
|     return MAX_FINGERS; | ||||
| } | ||||
|  | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -9,25 +9,18 @@ | ||||
| #include "common/common_types.h" | ||||
| #include "common/point.h" | ||||
| #include "common/swap.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Core::HID { | ||||
| class EmulatedConsole; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_Touchscreen final : public ControllerBase { | ||||
| public: | ||||
|     // This is nn::hid::TouchScreenModeForNx | ||||
|     enum class TouchScreenModeForNx : u8 { | ||||
|         UseSystemSetting, | ||||
|         Finger, | ||||
|         Heat2, | ||||
|     }; | ||||
|  | ||||
|     // This is nn::hid::TouchScreenConfigurationForNx | ||||
|     struct TouchScreenConfigurationForNx { | ||||
|         TouchScreenModeForNx mode; | ||||
|         INSERT_PADDING_BYTES_NOINIT(0x7); | ||||
| @@ -36,7 +29,7 @@ public: | ||||
|     static_assert(sizeof(TouchScreenConfigurationForNx) == 0x17, | ||||
|                   "TouchScreenConfigurationForNx is an invalid size"); | ||||
|  | ||||
|     explicit Controller_Touchscreen(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_Touchscreen(Core::System& system_); | ||||
|     ~Controller_Touchscreen() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -48,24 +41,73 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
| private: | ||||
|     static constexpr std::size_t MAX_FINGERS = 16; | ||||
|  | ||||
|     // This is nn::hid::TouchScreenState | ||||
|     struct TouchScreenState { | ||||
|         s64 sampling_number; | ||||
|         s32 entry_count; | ||||
|         INSERT_PADDING_BYTES(4); // Reserved | ||||
|         std::array<Core::HID::TouchState, MAX_FINGERS> states; | ||||
|     // Returns an unused finger id, if there is no fingers available std::nullopt will be returned | ||||
|     std::optional<std::size_t> GetUnusedFingerID() const; | ||||
|  | ||||
|     // If the touch is new it tries to assing a new finger id, if there is no fingers avaliable no | ||||
|     // changes will be made. Updates the coordinates if the finger id it's already set. If the touch | ||||
|     // ends delays the output by one frame to set the end_touch flag before finally freeing the | ||||
|     // finger id | ||||
|     std::size_t UpdateTouchInputEvent(const std::tuple<float, float, bool>& touch_input, | ||||
|                                       std::size_t finger_id); | ||||
|  | ||||
|     struct Attributes { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             BitField<0, 1, u32> start_touch; | ||||
|             BitField<1, 1, u32> end_touch; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(TouchScreenState) == 0x290, "TouchScreenState is an invalid size"); | ||||
|     static_assert(sizeof(Attributes) == 0x4, "Attributes is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::TouchScreenLifo | ||||
|     Lifo<TouchScreenState> touch_screen_lifo{}; | ||||
|     static_assert(sizeof(touch_screen_lifo) == 0x2C38, "touch_screen_lifo is an invalid size"); | ||||
|     TouchScreenState next_state{}; | ||||
|     struct TouchState { | ||||
|         u64_le delta_time; | ||||
|         Attributes attribute; | ||||
|         u32_le finger; | ||||
|         Common::Point<u32_le> position; | ||||
|         u32_le diameter_x; | ||||
|         u32_le diameter_y; | ||||
|         u32_le rotation_angle; | ||||
|     }; | ||||
|     static_assert(sizeof(TouchState) == 0x28, "Touchstate is an invalid size"); | ||||
|  | ||||
|     std::array<Core::HID::TouchFinger, MAX_FINGERS> fingers; | ||||
|     Core::HID::EmulatedConsole* console; | ||||
|     struct TouchScreenEntry { | ||||
|         s64_le sampling_number; | ||||
|         s64_le sampling_number2; | ||||
|         s32_le entry_count; | ||||
|         std::array<TouchState, MAX_FINGERS> states; | ||||
|     }; | ||||
|     static_assert(sizeof(TouchScreenEntry) == 0x298, "TouchScreenEntry is an invalid size"); | ||||
|  | ||||
|     struct TouchScreenSharedMemory { | ||||
|         CommonHeader header; | ||||
|         std::array<TouchScreenEntry, 17> shared_memory_entries{}; | ||||
|         INSERT_PADDING_BYTES(0x3c8); | ||||
|     }; | ||||
|     static_assert(sizeof(TouchScreenSharedMemory) == 0x3000, | ||||
|                   "TouchScreenSharedMemory is an invalid size"); | ||||
|  | ||||
|     struct Finger { | ||||
|         u64_le last_touch{}; | ||||
|         Common::Point<float> position; | ||||
|         u32_le id{}; | ||||
|         bool pressed{}; | ||||
|         Attributes attribute; | ||||
|     }; | ||||
|  | ||||
|     TouchScreenSharedMemory shared_memory{}; | ||||
|     std::unique_ptr<Input::TouchDevice> touch_mouse_device; | ||||
|     std::unique_ptr<Input::TouchDevice> touch_udp_device; | ||||
|     std::unique_ptr<Input::TouchDevice> touch_btn_device; | ||||
|     std::array<std::size_t, MAX_FINGERS> mouse_finger_id; | ||||
|     std::array<std::size_t, MAX_FINGERS> keyboard_finger_id; | ||||
|     std::array<std::size_t, MAX_FINGERS> udp_finger_id; | ||||
|     std::array<Finger, MAX_FINGERS> fingers; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -5,13 +5,12 @@ | ||||
| #include <cstring> | ||||
| #include "common/common_types.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/hid/controllers/xpad.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| constexpr std::size_t SHARED_MEMORY_OFFSET = 0x3C00; | ||||
|  | ||||
| Controller_XPad::Controller_XPad(Core::HID::HIDCore& hid_core_) : ControllerBase{hid_core_} {} | ||||
| Controller_XPad::Controller_XPad(Core::System& system_) : ControllerBase{system_} {} | ||||
| Controller_XPad::~Controller_XPad() = default; | ||||
|  | ||||
| void Controller_XPad::OnInit() {} | ||||
| @@ -20,19 +19,28 @@ void Controller_XPad::OnRelease() {} | ||||
|  | ||||
| void Controller_XPad::OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, | ||||
|                                std::size_t size) { | ||||
|     if (!IsControllerActivated()) { | ||||
|         basic_xpad_lifo.buffer_count = 0; | ||||
|         basic_xpad_lifo.buffer_tail = 0; | ||||
|         std::memcpy(data + SHARED_MEMORY_OFFSET, &basic_xpad_lifo, sizeof(basic_xpad_lifo)); | ||||
|         return; | ||||
|     } | ||||
|     for (auto& xpad_entry : shared_memory.shared_memory_entries) { | ||||
|         xpad_entry.header.timestamp = core_timing.GetCPUTicks(); | ||||
|         xpad_entry.header.total_entry_count = 17; | ||||
|  | ||||
|     const auto& last_entry = basic_xpad_lifo.ReadCurrentEntry().state; | ||||
|     next_state.sampling_number = last_entry.sampling_number + 1; | ||||
|         if (!IsControllerActivated()) { | ||||
|             xpad_entry.header.entry_count = 0; | ||||
|             xpad_entry.header.last_entry_index = 0; | ||||
|             return; | ||||
|         } | ||||
|         xpad_entry.header.entry_count = 16; | ||||
|  | ||||
|         const auto& last_entry = xpad_entry.pad_states[xpad_entry.header.last_entry_index]; | ||||
|         xpad_entry.header.last_entry_index = (xpad_entry.header.last_entry_index + 1) % 17; | ||||
|         auto& cur_entry = xpad_entry.pad_states[xpad_entry.header.last_entry_index]; | ||||
|  | ||||
|         cur_entry.sampling_number = last_entry.sampling_number + 1; | ||||
|         cur_entry.sampling_number2 = cur_entry.sampling_number; | ||||
|     } | ||||
|     // TODO(ogniK): Update xpad states | ||||
|  | ||||
|     basic_xpad_lifo.WriteNextEntry(next_state); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &basic_xpad_lifo, sizeof(basic_xpad_lifo)); | ||||
|     std::memcpy(data + SHARED_MEMORY_OFFSET, &shared_memory, sizeof(SharedMemory)); | ||||
| } | ||||
|  | ||||
| void Controller_XPad::OnLoadInputDevices() {} | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -8,14 +8,12 @@ | ||||
| #include "common/common_funcs.h" | ||||
| #include "common/common_types.h" | ||||
| #include "common/swap.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hle/service/hid/controllers/controller_base.h" | ||||
| #include "core/hle/service/hid/ring_lifo.h" | ||||
|  | ||||
| namespace Service::HID { | ||||
| class Controller_XPad final : public ControllerBase { | ||||
| public: | ||||
|     explicit Controller_XPad(Core::HID::HIDCore& hid_core_); | ||||
|     explicit Controller_XPad(Core::System& system_); | ||||
|     ~Controller_XPad() override; | ||||
|  | ||||
|     // Called when the controller is initialized | ||||
| @@ -27,11 +25,13 @@ public: | ||||
|     // When the controller is requesting an update for the shared memory | ||||
|     void OnUpdate(const Core::Timing::CoreTiming& core_timing, u8* data, std::size_t size) override; | ||||
|  | ||||
|     // Called when input devices should be loaded | ||||
|     void OnLoadInputDevices() override; | ||||
|  | ||||
| private: | ||||
|     // This is nn::hid::BasicXpadAttributeSet | ||||
|     struct BasicXpadAttributeSet { | ||||
|     struct Attributes { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u32_le raw{}; | ||||
|             BitField<0, 1, u32> is_connected; | ||||
|             BitField<1, 1, u32> is_wired; | ||||
|             BitField<2, 1, u32> is_left_connected; | ||||
| @@ -40,12 +40,11 @@ private: | ||||
|             BitField<5, 1, u32> is_right_wired; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(BasicXpadAttributeSet) == 4, "BasicXpadAttributeSet is an invalid size"); | ||||
|     static_assert(sizeof(Attributes) == 4, "Attributes is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::BasicXpadButtonSet | ||||
|     struct BasicXpadButtonSet { | ||||
|     struct Buttons { | ||||
|         union { | ||||
|             u32 raw{}; | ||||
|             u32_le raw{}; | ||||
|             // Button states | ||||
|             BitField<0, 1, u32> a; | ||||
|             BitField<1, 1, u32> b; | ||||
| @@ -89,21 +88,35 @@ private: | ||||
|             BitField<30, 1, u32> handheld_left_b; | ||||
|         }; | ||||
|     }; | ||||
|     static_assert(sizeof(BasicXpadButtonSet) == 4, "BasicXpadButtonSet is an invalid size"); | ||||
|     static_assert(sizeof(Buttons) == 4, "Buttons is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::BasicXpadState | ||||
|     struct BasicXpadState { | ||||
|         s64 sampling_number; | ||||
|         BasicXpadAttributeSet attributes; | ||||
|         BasicXpadButtonSet pad_states; | ||||
|         Core::HID::AnalogStickState l_stick; | ||||
|         Core::HID::AnalogStickState r_stick; | ||||
|     struct AnalogStick { | ||||
|         s32_le x; | ||||
|         s32_le y; | ||||
|     }; | ||||
|     static_assert(sizeof(BasicXpadState) == 0x20, "BasicXpadState is an invalid size"); | ||||
|     static_assert(sizeof(AnalogStick) == 0x8, "AnalogStick is an invalid size"); | ||||
|  | ||||
|     // This is nn::hid::detail::BasicXpadLifo | ||||
|     Lifo<BasicXpadState> basic_xpad_lifo{}; | ||||
|     static_assert(sizeof(basic_xpad_lifo) == 0x2C8, "basic_xpad_lifo is an invalid size"); | ||||
|     BasicXpadState next_state{}; | ||||
|     struct XPadState { | ||||
|         s64_le sampling_number; | ||||
|         s64_le sampling_number2; | ||||
|         Attributes attributes; | ||||
|         Buttons pad_states; | ||||
|         AnalogStick l_stick; | ||||
|         AnalogStick r_stick; | ||||
|     }; | ||||
|     static_assert(sizeof(XPadState) == 0x28, "XPadState is an invalid size"); | ||||
|  | ||||
|     struct XPadEntry { | ||||
|         CommonHeader header; | ||||
|         std::array<XPadState, 17> pad_states{}; | ||||
|         INSERT_PADDING_BYTES(0x138); | ||||
|     }; | ||||
|     static_assert(sizeof(XPadEntry) == 0x400, "XPadEntry is an invalid size"); | ||||
|  | ||||
|     struct SharedMemory { | ||||
|         std::array<XPadEntry, 4> shared_memory_entries{}; | ||||
|     }; | ||||
|     static_assert(sizeof(SharedMemory) == 0x1000, "SharedMemory is an invalid size"); | ||||
|     SharedMemory shared_memory{}; | ||||
| }; | ||||
| } // namespace Service::HID | ||||
|   | ||||
| @@ -8,7 +8,7 @@ | ||||
| #include "common/settings.h" | ||||
| #include "core/core.h" | ||||
| #include "core/core_timing.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "core/hle/ipc_helpers.h" | ||||
| #include "core/hle/kernel/k_readable_event.h" | ||||
| #include "core/hle/kernel/k_shared_memory.h" | ||||
| @@ -34,10 +34,10 @@ | ||||
| namespace Service::HID { | ||||
|  | ||||
| // Updating period for each HID device. | ||||
| // Period time is obtained by measuring the number of samples in a second on HW using a homebrew | ||||
| constexpr auto pad_update_ns = std::chrono::nanoseconds{4 * 1000 * 1000};      // (4ms, 250Hz) | ||||
| constexpr auto keyboard_update_ns = std::chrono::nanoseconds{8 * 1000 * 1000}; // (8ms, 125Hz) | ||||
| constexpr auto motion_update_ns = std::chrono::nanoseconds{5 * 1000 * 1000};   // (5ms, 200Hz) | ||||
| // HID is polled every 15ms, this value was derived from | ||||
| // https://github.com/dekuNukem/Nintendo_Switch_Reverse_Engineering#joy-con-status-data-packet | ||||
| constexpr auto pad_update_ns = std::chrono::nanoseconds{1000 * 1000};         // (1ms, 1000Hz) | ||||
| constexpr auto motion_update_ns = std::chrono::nanoseconds{15 * 1000 * 1000}; // (15ms, 66.666Hz) | ||||
| constexpr std::size_t SHARED_MEMORY_SIZE = 0x40000; | ||||
|  | ||||
| IAppletResource::IAppletResource(Core::System& system_, | ||||
| @@ -79,24 +79,17 @@ IAppletResource::IAppletResource(Core::System& system_, | ||||
|             const auto guard = LockService(); | ||||
|             UpdateControllers(user_data, ns_late); | ||||
|         }); | ||||
|     keyboard_update_event = Core::Timing::CreateEvent( | ||||
|         "HID::UpdatekeyboardCallback", | ||||
|         [this](std::uintptr_t user_data, std::chrono::nanoseconds ns_late) { | ||||
|             const auto guard = LockService(); | ||||
|             UpdateKeyboard(user_data, ns_late); | ||||
|         }); | ||||
|     motion_update_event = Core::Timing::CreateEvent( | ||||
|         "HID::UpdateMotionCallback", | ||||
|         "HID::MotionPadCallback", | ||||
|         [this](std::uintptr_t user_data, std::chrono::nanoseconds ns_late) { | ||||
|             const auto guard = LockService(); | ||||
|             UpdateMotion(user_data, ns_late); | ||||
|         }); | ||||
|  | ||||
|     system.CoreTiming().ScheduleEvent(pad_update_ns, pad_update_event); | ||||
|     system.CoreTiming().ScheduleEvent(keyboard_update_ns, keyboard_update_event); | ||||
|     system.CoreTiming().ScheduleEvent(motion_update_ns, motion_update_event); | ||||
|  | ||||
|     system.HIDCore().ReloadInputDevices(); | ||||
|     ReloadInputDevices(); | ||||
| } | ||||
|  | ||||
| void IAppletResource::ActivateController(HidController controller) { | ||||
| @@ -109,7 +102,6 @@ void IAppletResource::DeactivateController(HidController controller) { | ||||
|  | ||||
| IAppletResource::~IAppletResource() { | ||||
|     system.CoreTiming().UnscheduleEvent(pad_update_event, 0); | ||||
|     system.CoreTiming().UnscheduleEvent(keyboard_update_event, 0); | ||||
|     system.CoreTiming().UnscheduleEvent(motion_update_event, 0); | ||||
| } | ||||
|  | ||||
| @@ -125,37 +117,23 @@ void IAppletResource::UpdateControllers(std::uintptr_t user_data, | ||||
|                                         std::chrono::nanoseconds ns_late) { | ||||
|     auto& core_timing = system.CoreTiming(); | ||||
|  | ||||
|     const bool should_reload = Settings::values.is_device_reload_pending.exchange(false); | ||||
|     for (const auto& controller : controllers) { | ||||
|         // Keyboard has it's own update event | ||||
|         if (controller == controllers[static_cast<size_t>(HidController::Keyboard)]) { | ||||
|             continue; | ||||
|         if (should_reload) { | ||||
|             controller->OnLoadInputDevices(); | ||||
|         } | ||||
|         controller->OnUpdate(core_timing, system.Kernel().GetHidSharedMem().GetPointer(), | ||||
|                              SHARED_MEMORY_SIZE); | ||||
|     } | ||||
|  | ||||
|     // If ns_late is higher than the update rate ignore the delay | ||||
|     if (ns_late > pad_update_ns) { | ||||
|     if (ns_late > motion_update_ns) { | ||||
|         ns_late = {}; | ||||
|     } | ||||
|  | ||||
|     core_timing.ScheduleEvent(pad_update_ns - ns_late, pad_update_event); | ||||
| } | ||||
|  | ||||
| void IAppletResource::UpdateKeyboard(std::uintptr_t user_data, std::chrono::nanoseconds ns_late) { | ||||
|     auto& core_timing = system.CoreTiming(); | ||||
|  | ||||
|     controllers[static_cast<size_t>(HidController::Keyboard)]->OnUpdate( | ||||
|         core_timing, system.Kernel().GetHidSharedMem().GetPointer(), SHARED_MEMORY_SIZE); | ||||
|  | ||||
|     // If ns_late is higher than the update rate ignore the delay | ||||
|     if (ns_late > keyboard_update_ns) { | ||||
|         ns_late = {}; | ||||
|     } | ||||
|  | ||||
|     core_timing.ScheduleEvent(keyboard_update_ns - ns_late, keyboard_update_event); | ||||
| } | ||||
|  | ||||
| void IAppletResource::UpdateMotion(std::uintptr_t user_data, std::chrono::nanoseconds ns_late) { | ||||
|     auto& core_timing = system.CoreTiming(); | ||||
|  | ||||
| @@ -188,7 +166,7 @@ public: | ||||
| private: | ||||
|     void InitializeVibrationDevice(Kernel::HLERequestContext& ctx) { | ||||
|         IPC::RequestParser rp{ctx}; | ||||
|         const auto vibration_device_handle{rp.PopRaw<Core::HID::VibrationDeviceHandle>()}; | ||||
|         const auto vibration_device_handle{rp.PopRaw<Controller_NPad::DeviceHandle>()}; | ||||
|  | ||||
|         if (applet_resource != nullptr) { | ||||
|             applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
| @@ -444,7 +422,6 @@ void Hid::ActivateXpad(Kernel::HLERequestContext& ctx) { | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -471,18 +448,19 @@ void Hid::GetXpadIDs(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::ActivateSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         u32 basic_xpad_id; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     // This function does nothing on 10.0.0+ | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad).SetSixAxisEnabled(true); | ||||
|  | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called, basic_xpad_id={}, applet_resource_user_id={}", | ||||
|                 parameters.basic_xpad_id, parameters.applet_resource_user_id); | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|               parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|               parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -491,18 +469,19 @@ void Hid::ActivateSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::DeactivateSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         u32 basic_xpad_id; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     // This function does nothing on 10.0.0+ | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad).SetSixAxisEnabled(false); | ||||
|  | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called, basic_xpad_id={}, applet_resource_user_id={}", | ||||
|                 parameters.basic_xpad_id, parameters.applet_resource_user_id); | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|               parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|               parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -511,16 +490,14 @@ void Hid::DeactivateSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::StartSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetSixAxisEnabled(parameters.sixaxis_handle, true); | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad).SetSixAxisEnabled(true); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
| @@ -534,16 +511,14 @@ void Hid::StartSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::StopSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetSixAxisEnabled(parameters.sixaxis_handle, false); | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad).SetSixAxisEnabled(false); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
| @@ -559,23 +534,19 @@ void Hid::EnableSixAxisSensorFusion(Kernel::HLERequestContext& ctx) { | ||||
|     struct Parameters { | ||||
|         bool enable_sixaxis_sensor_fusion; | ||||
|         INSERT_PADDING_BYTES_NOINIT(3); | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetSixAxisFusionEnabled(parameters.sixaxis_handle, | ||||
|                                  parameters.enable_sixaxis_sensor_fusion); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, enable_sixaxis_sensor_fusion={}, npad_type={}, npad_id={}, " | ||||
|               "device_index={}, applet_resource_user_id={}", | ||||
|               parameters.enable_sixaxis_sensor_fusion, parameters.sixaxis_handle.npad_type, | ||||
|               parameters.sixaxis_handle.npad_id, parameters.sixaxis_handle.device_index, | ||||
|               parameters.applet_resource_user_id); | ||||
|     LOG_WARNING(Service_HID, | ||||
|                 "(STUBBED) called, enable_sixaxis_sensor_fusion={}, npad_type={}, npad_id={}, " | ||||
|                 "device_index={}, applet_resource_user_id={}", | ||||
|                 parameters.enable_sixaxis_sensor_fusion, parameters.sixaxis_handle.npad_type, | ||||
|                 parameters.sixaxis_handle.npad_id, parameters.sixaxis_handle.device_index, | ||||
|                 parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -584,9 +555,9 @@ void Hid::EnableSixAxisSensorFusion(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::SetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Core::HID::SixAxisSensorFusionParameters sixaxis_fusion; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         f32 parameter1; | ||||
|         f32 parameter2; | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x18, "Parameters has incorrect size."); | ||||
| @@ -594,14 +565,14 @@ void Hid::SetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetSixAxisFusionParameters(parameters.sixaxis_handle, parameters.sixaxis_fusion); | ||||
|         .SetSixAxisFusionParameters(parameters.parameter1, parameters.parameter2); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, parameter1={}, " | ||||
|               "parameter2={}, applet_resource_user_id={}", | ||||
|               parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|               parameters.sixaxis_handle.device_index, parameters.sixaxis_fusion.parameter1, | ||||
|               parameters.sixaxis_fusion.parameter2, parameters.applet_resource_user_id); | ||||
|     LOG_WARNING(Service_HID, | ||||
|                 "(STUBBED) called, npad_type={}, npad_id={}, device_index={}, parameter1={}, " | ||||
|                 "parameter2={}, applet_resource_user_id={}", | ||||
|                 parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|                 parameters.sixaxis_handle.device_index, parameters.parameter1, | ||||
|                 parameters.parameter2, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -610,33 +581,35 @@ void Hid::SetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::GetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     f32 parameter1 = 0; | ||||
|     f32 parameter2 = 0; | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     const auto sixaxis_fusion_parameters = | ||||
|     std::tie(parameter1, parameter2) = | ||||
|         applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|             .GetSixAxisFusionParameters(parameters.sixaxis_handle); | ||||
|             .GetSixAxisFusionParameters(); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|               parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|               parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|     LOG_WARNING( | ||||
|         Service_HID, | ||||
|         "(STUBBED) called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|         parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|         parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 4}; | ||||
|     rb.Push(ResultSuccess); | ||||
|     rb.PushRaw(sixaxis_fusion_parameters); | ||||
|     rb.Push(parameter1); | ||||
|     rb.Push(parameter2); | ||||
| } | ||||
|  | ||||
| void Hid::ResetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
| @@ -644,12 +617,13 @@ void Hid::ResetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .ResetSixAxisFusionParameters(parameters.sixaxis_handle); | ||||
|         .ResetSixAxisFusionParameters(); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|               parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|               parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|     LOG_WARNING( | ||||
|         Service_HID, | ||||
|         "(STUBBED) called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|         parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|         parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -657,12 +631,12 @@ void Hid::ResetSixAxisSensorFusionParameters(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::SetGyroscopeZeroDriftMode(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto sixaxis_handle{rp.PopRaw<Core::HID::SixAxisSensorHandle>()}; | ||||
|     const auto sixaxis_handle{rp.PopRaw<Controller_NPad::DeviceHandle>()}; | ||||
|     const auto drift_mode{rp.PopEnum<Controller_NPad::GyroscopeZeroDriftMode>()}; | ||||
|     const auto applet_resource_user_id{rp.Pop<u64>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetGyroscopeZeroDriftMode(sixaxis_handle, drift_mode); | ||||
|         .SetGyroscopeZeroDriftMode(drift_mode); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, drift_mode={}, " | ||||
| @@ -677,11 +651,10 @@ void Hid::SetGyroscopeZeroDriftMode(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::GetGyroscopeZeroDriftMode(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -693,23 +666,21 @@ void Hid::GetGyroscopeZeroDriftMode(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::ResponseBuilder rb{ctx, 3}; | ||||
|     rb.Push(ResultSuccess); | ||||
|     rb.PushEnum(applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|                     .GetGyroscopeZeroDriftMode(parameters.sixaxis_handle)); | ||||
|                     .GetGyroscopeZeroDriftMode()); | ||||
| } | ||||
|  | ||||
| void Hid::ResetGyroscopeZeroDriftMode(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|     const auto drift_mode{Controller_NPad::GyroscopeZeroDriftMode::Standard}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetGyroscopeZeroDriftMode(parameters.sixaxis_handle, drift_mode); | ||||
|         .SetGyroscopeZeroDriftMode(Controller_NPad::GyroscopeZeroDriftMode::Standard); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
|               "called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
| @@ -723,11 +694,10 @@ void Hid::ResetGyroscopeZeroDriftMode(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::IsSixAxisSensorAtRest(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -739,17 +709,16 @@ void Hid::IsSixAxisSensorAtRest(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::ResponseBuilder rb{ctx, 3}; | ||||
|     rb.Push(ResultSuccess); | ||||
|     rb.Push(applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|                 .IsSixAxisSensorAtRest(parameters.sixaxis_handle)); | ||||
|                 .IsSixAxisSensorAtRest()); | ||||
| } | ||||
|  | ||||
| void Hid::IsFirmwareUpdateAvailableForSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::SixAxisSensorHandle sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -771,14 +740,13 @@ void Hid::ActivateGesture(Kernel::HLERequestContext& ctx) { | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->ActivateController(HidController::Gesture); | ||||
|  | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called, unknown={}, applet_resource_user_id={}", | ||||
|                 parameters.unknown, parameters.applet_resource_user_id); | ||||
|     LOG_DEBUG(Service_HID, "called, unknown={}, applet_resource_user_id={}", parameters.unknown, | ||||
|               parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -786,20 +754,12 @@ void Hid::ActivateGesture(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::SetSupportedNpadStyleSet(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadStyleSet supported_styleset; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|     const auto supported_styleset{rp.Pop<u32>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetSupportedStyleSet({parameters.supported_styleset}); | ||||
|         .SetSupportedStyleSet({supported_styleset}); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, "called, supported_styleset={}, applet_resource_user_id={}", | ||||
|               parameters.supported_styleset, parameters.applet_resource_user_id); | ||||
|     LOG_DEBUG(Service_HID, "called, supported_styleset={}", supported_styleset); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -813,9 +773,9 @@ void Hid::GetSupportedNpadStyleSet(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 3}; | ||||
|     rb.Push(ResultSuccess); | ||||
|     rb.PushEnum(applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|                     .GetSupportedStyleSet() | ||||
|                     .raw); | ||||
|     rb.Push(applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|                 .GetSupportedStyleSet() | ||||
|                 .raw); | ||||
| } | ||||
|  | ||||
| void Hid::SetSupportedNpadIdType(Kernel::HLERequestContext& ctx) { | ||||
| @@ -858,12 +818,11 @@ void Hid::DeactivateNpad(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::AcquireNpadStyleSetUpdateEventHandle(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|         u64 unknown; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x18, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -879,11 +838,10 @@ void Hid::AcquireNpadStyleSetUpdateEventHandle(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::DisconnectNpad(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -899,7 +857,7 @@ void Hid::DisconnectNpad(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::GetPlayerLedPattern(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto npad_id{rp.PopEnum<Core::HID::NpadIdType>()}; | ||||
|     const auto npad_id{rp.Pop<u32>()}; | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, "called, npad_id={}", npad_id); | ||||
|  | ||||
| @@ -914,17 +872,16 @@ void Hid::ActivateNpadWithRevision(Kernel::HLERequestContext& ctx) { | ||||
|     // Should have no effect with how our npad sets up the data | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         s32 revision; | ||||
|         u32 unknown; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->ActivateController(HidController::NPad); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, "called, revision={}, applet_resource_user_id={}", parameters.revision, | ||||
|     LOG_DEBUG(Service_HID, "called, unknown={}, applet_resource_user_id={}", parameters.unknown, | ||||
|               parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
| @@ -934,7 +891,7 @@ void Hid::ActivateNpadWithRevision(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::SetNpadJoyHoldType(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto applet_resource_user_id{rp.Pop<u64>()}; | ||||
|     const auto hold_type{rp.PopEnum<Controller_NPad::NpadJoyHoldType>()}; | ||||
|     const auto hold_type{rp.PopEnum<Controller_NPad::NpadHoldType>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad).SetHoldType(hold_type); | ||||
|  | ||||
| @@ -959,16 +916,15 @@ void Hid::GetNpadJoyHoldType(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::SetNpadJoyAssignmentModeSingleByDefault(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetNpadMode(parameters.npad_id, Controller_NPad::NpadJoyAssignmentMode::Single); | ||||
|         .SetNpadMode(parameters.npad_id, Controller_NPad::NpadAssignments::Single); | ||||
|  | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called, npad_id={}, applet_resource_user_id={}", | ||||
|                 parameters.npad_id, parameters.applet_resource_user_id); | ||||
| @@ -981,17 +937,16 @@ void Hid::SetNpadJoyAssignmentModeSingle(Kernel::HLERequestContext& ctx) { | ||||
|     // TODO: Check the differences between this and SetNpadJoyAssignmentModeSingleByDefault | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|         u64 npad_joy_device_type; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x18, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetNpadMode(parameters.npad_id, Controller_NPad::NpadJoyAssignmentMode::Single); | ||||
|         .SetNpadMode(parameters.npad_id, Controller_NPad::NpadAssignments::Single); | ||||
|  | ||||
|     LOG_WARNING(Service_HID, | ||||
|                 "(STUBBED) called, npad_id={}, applet_resource_user_id={}, npad_joy_device_type={}", | ||||
| @@ -1005,16 +960,15 @@ void Hid::SetNpadJoyAssignmentModeSingle(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::SetNpadJoyAssignmentModeDual(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
|         .SetNpadMode(parameters.npad_id, Controller_NPad::NpadJoyAssignmentMode::Dual); | ||||
|         .SetNpadMode(parameters.npad_id, Controller_NPad::NpadAssignments::Dual); | ||||
|  | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called, npad_id={}, applet_resource_user_id={}", | ||||
|                 parameters.npad_id, parameters.applet_resource_user_id); | ||||
| @@ -1025,8 +979,8 @@ void Hid::SetNpadJoyAssignmentModeDual(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::MergeSingleJoyAsDualJoy(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto npad_id_1{rp.PopEnum<Core::HID::NpadIdType>()}; | ||||
|     const auto npad_id_2{rp.PopEnum<Core::HID::NpadIdType>()}; | ||||
|     const auto npad_id_1{rp.Pop<u32>()}; | ||||
|     const auto npad_id_2{rp.Pop<u32>()}; | ||||
|     const auto applet_resource_user_id{rp.Pop<u64>()}; | ||||
|  | ||||
|     applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
| @@ -1092,8 +1046,8 @@ void Hid::GetNpadHandheldActivationMode(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::SwapNpadAssignment(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto npad_id_1{rp.PopEnum<Core::HID::NpadIdType>()}; | ||||
|     const auto npad_id_2{rp.PopEnum<Core::HID::NpadIdType>()}; | ||||
|     const auto npad_id_1{rp.Pop<u32>()}; | ||||
|     const auto npad_id_2{rp.Pop<u32>()}; | ||||
|     const auto applet_resource_user_id{rp.Pop<u64>()}; | ||||
|  | ||||
|     const bool res = applet_resource->GetController<Controller_NPad>(HidController::NPad) | ||||
| @@ -1114,11 +1068,10 @@ void Hid::SwapNpadAssignment(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::IsUnintendedHomeButtonInputProtectionEnabled(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -1136,10 +1089,9 @@ void Hid::EnableUnintendedHomeButtonInputProtection(Kernel::HLERequestContext& c | ||||
|     struct Parameters { | ||||
|         bool unintended_home_button_input_protection; | ||||
|         INSERT_PADDING_BYTES_NOINIT(3); | ||||
|         Core::HID::NpadIdType npad_id; | ||||
|         u32 npad_id; | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -1161,7 +1113,6 @@ void Hid::SetNpadAnalogStickUseCenterClamp(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         bool analog_stick_use_center_clamp; | ||||
|         INSERT_PADDING_BYTES_NOINIT(7); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
| @@ -1181,38 +1132,38 @@ void Hid::SetNpadAnalogStickUseCenterClamp(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::GetVibrationDeviceInfo(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto vibration_device_handle{rp.PopRaw<Core::HID::VibrationDeviceHandle>()}; | ||||
|     const auto vibration_device_handle{rp.PopRaw<Controller_NPad::DeviceHandle>()}; | ||||
|  | ||||
|     Core::HID::VibrationDeviceInfo vibration_device_info; | ||||
|     VibrationDeviceInfo vibration_device_info; | ||||
|  | ||||
|     switch (vibration_device_handle.npad_type) { | ||||
|     case Core::HID::NpadStyleIndex::ProController: | ||||
|     case Core::HID::NpadStyleIndex::Handheld: | ||||
|     case Core::HID::NpadStyleIndex::JoyconDual: | ||||
|     case Core::HID::NpadStyleIndex::JoyconLeft: | ||||
|     case Core::HID::NpadStyleIndex::JoyconRight: | ||||
|     case Controller_NPad::NpadType::ProController: | ||||
|     case Controller_NPad::NpadType::Handheld: | ||||
|     case Controller_NPad::NpadType::JoyconDual: | ||||
|     case Controller_NPad::NpadType::JoyconLeft: | ||||
|     case Controller_NPad::NpadType::JoyconRight: | ||||
|     default: | ||||
|         vibration_device_info.type = Core::HID::VibrationDeviceType::LinearResonantActuator; | ||||
|         vibration_device_info.type = VibrationDeviceType::LinearResonantActuator; | ||||
|         break; | ||||
|     case Core::HID::NpadStyleIndex::GameCube: | ||||
|         vibration_device_info.type = Core::HID::VibrationDeviceType::GcErm; | ||||
|     case Controller_NPad::NpadType::GameCube: | ||||
|         vibration_device_info.type = VibrationDeviceType::GcErm; | ||||
|         break; | ||||
|     case Core::HID::NpadStyleIndex::Pokeball: | ||||
|         vibration_device_info.type = Core::HID::VibrationDeviceType::Unknown; | ||||
|     case Controller_NPad::NpadType::Pokeball: | ||||
|         vibration_device_info.type = VibrationDeviceType::Unknown; | ||||
|         break; | ||||
|     } | ||||
|  | ||||
|     switch (vibration_device_handle.device_index) { | ||||
|     case Core::HID::DeviceIndex::Left: | ||||
|         vibration_device_info.position = Core::HID::VibrationDevicePosition::Left; | ||||
|     case Controller_NPad::DeviceIndex::Left: | ||||
|         vibration_device_info.position = VibrationDevicePosition::Left; | ||||
|         break; | ||||
|     case Core::HID::DeviceIndex::Right: | ||||
|         vibration_device_info.position = Core::HID::VibrationDevicePosition::Right; | ||||
|     case Controller_NPad::DeviceIndex::Right: | ||||
|         vibration_device_info.position = VibrationDevicePosition::Right; | ||||
|         break; | ||||
|     case Core::HID::DeviceIndex::None: | ||||
|     case Controller_NPad::DeviceIndex::None: | ||||
|     default: | ||||
|         UNREACHABLE_MSG("DeviceIndex should never be None!"); | ||||
|         vibration_device_info.position = Core::HID::VibrationDevicePosition::None; | ||||
|         vibration_device_info.position = VibrationDevicePosition::None; | ||||
|         break; | ||||
|     } | ||||
|  | ||||
| @@ -1227,12 +1178,11 @@ void Hid::GetVibrationDeviceInfo(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::SendVibrationValue(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::VibrationDeviceHandle vibration_device_handle; | ||||
|         Core::HID::VibrationValue vibration_value; | ||||
|         Controller_NPad::DeviceHandle vibration_device_handle; | ||||
|         Controller_NPad::VibrationValue vibration_value; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x20, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -1252,11 +1202,10 @@ void Hid::SendVibrationValue(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::GetActualVibrationValue(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::VibrationDeviceHandle vibration_device_handle; | ||||
|         Controller_NPad::DeviceHandle vibration_device_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -1307,10 +1256,10 @@ void Hid::SendVibrationValues(Kernel::HLERequestContext& ctx) { | ||||
|     const auto handles = ctx.ReadBuffer(0); | ||||
|     const auto vibrations = ctx.ReadBuffer(1); | ||||
|  | ||||
|     std::vector<Core::HID::VibrationDeviceHandle> vibration_device_handles( | ||||
|         handles.size() / sizeof(Core::HID::VibrationDeviceHandle)); | ||||
|     std::vector<Core::HID::VibrationValue> vibration_values(vibrations.size() / | ||||
|                                                             sizeof(Core::HID::VibrationValue)); | ||||
|     std::vector<Controller_NPad::DeviceHandle> vibration_device_handles( | ||||
|         handles.size() / sizeof(Controller_NPad::DeviceHandle)); | ||||
|     std::vector<Controller_NPad::VibrationValue> vibration_values( | ||||
|         vibrations.size() / sizeof(Controller_NPad::VibrationValue)); | ||||
|  | ||||
|     std::memcpy(vibration_device_handles.data(), handles.data(), handles.size()); | ||||
|     std::memcpy(vibration_values.data(), vibrations.data(), vibrations.size()); | ||||
| @@ -1327,10 +1276,9 @@ void Hid::SendVibrationValues(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::SendVibrationGcErmCommand(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::VibrationDeviceHandle vibration_device_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         Controller_NPad::DeviceHandle vibration_device_handle; | ||||
|         u64 applet_resource_user_id; | ||||
|         Core::HID::VibrationGcErmCommand gc_erm_command; | ||||
|         VibrationGcErmCommand gc_erm_command; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x18, "Parameters has incorrect size."); | ||||
|  | ||||
| @@ -1344,26 +1292,26 @@ void Hid::SendVibrationGcErmCommand(Kernel::HLERequestContext& ctx) { | ||||
|      */ | ||||
|     const auto vibration_value = [parameters] { | ||||
|         switch (parameters.gc_erm_command) { | ||||
|         case Core::HID::VibrationGcErmCommand::Stop: | ||||
|             return Core::HID::VibrationValue{ | ||||
|                 .low_amplitude = 0.0f, | ||||
|                 .low_frequency = 160.0f, | ||||
|                 .high_amplitude = 0.0f, | ||||
|                 .high_frequency = 320.0f, | ||||
|         case VibrationGcErmCommand::Stop: | ||||
|             return Controller_NPad::VibrationValue{ | ||||
|                 .amp_low = 0.0f, | ||||
|                 .freq_low = 160.0f, | ||||
|                 .amp_high = 0.0f, | ||||
|                 .freq_high = 320.0f, | ||||
|             }; | ||||
|         case Core::HID::VibrationGcErmCommand::Start: | ||||
|             return Core::HID::VibrationValue{ | ||||
|                 .low_amplitude = 1.0f, | ||||
|                 .low_frequency = 160.0f, | ||||
|                 .high_amplitude = 1.0f, | ||||
|                 .high_frequency = 320.0f, | ||||
|         case VibrationGcErmCommand::Start: | ||||
|             return Controller_NPad::VibrationValue{ | ||||
|                 .amp_low = 1.0f, | ||||
|                 .freq_low = 160.0f, | ||||
|                 .amp_high = 1.0f, | ||||
|                 .freq_high = 320.0f, | ||||
|             }; | ||||
|         case Core::HID::VibrationGcErmCommand::StopHard: | ||||
|             return Core::HID::VibrationValue{ | ||||
|                 .low_amplitude = 0.0f, | ||||
|                 .low_frequency = 0.0f, | ||||
|                 .high_amplitude = 0.0f, | ||||
|                 .high_frequency = 0.0f, | ||||
|         case VibrationGcErmCommand::StopHard: | ||||
|             return Controller_NPad::VibrationValue{ | ||||
|                 .amp_low = 0.0f, | ||||
|                 .freq_low = 0.0f, | ||||
|                 .amp_high = 0.0f, | ||||
|                 .freq_high = 0.0f, | ||||
|             }; | ||||
|         default: | ||||
|             return Controller_NPad::DEFAULT_VIBRATION_VALUE; | ||||
| @@ -1388,7 +1336,7 @@ void Hid::SendVibrationGcErmCommand(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::GetActualVibrationGcErmCommand(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::VibrationDeviceHandle vibration_device_handle; | ||||
|         Controller_NPad::DeviceHandle vibration_device_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
| @@ -1399,8 +1347,8 @@ void Hid::GetActualVibrationGcErmCommand(Kernel::HLERequestContext& ctx) { | ||||
|                                     .GetLastVibration(parameters.vibration_device_handle); | ||||
|  | ||||
|     const auto gc_erm_command = [last_vibration] { | ||||
|         if (last_vibration.low_amplitude != 0.0f || last_vibration.high_amplitude != 0.0f) { | ||||
|             return Core::HID::VibrationGcErmCommand::Start; | ||||
|         if (last_vibration.amp_low != 0.0f || last_vibration.amp_high != 0.0f) { | ||||
|             return VibrationGcErmCommand::Start; | ||||
|         } | ||||
|  | ||||
|         /** | ||||
| @@ -1409,11 +1357,11 @@ void Hid::GetActualVibrationGcErmCommand(Kernel::HLERequestContext& ctx) { | ||||
|          * SendVibrationGcErmCommand, in order to differentiate between Stop and StopHard commands. | ||||
|          * This is done to reuse the controller vibration functions made for regular controllers. | ||||
|          */ | ||||
|         if (last_vibration.low_frequency == 0.0f && last_vibration.high_frequency == 0.0f) { | ||||
|             return Core::HID::VibrationGcErmCommand::StopHard; | ||||
|         if (last_vibration.freq_low == 0.0f && last_vibration.freq_high == 0.0f) { | ||||
|             return VibrationGcErmCommand::StopHard; | ||||
|         } | ||||
|  | ||||
|         return Core::HID::VibrationGcErmCommand::Stop; | ||||
|         return VibrationGcErmCommand::Stop; | ||||
|     }(); | ||||
|  | ||||
|     LOG_DEBUG(Service_HID, | ||||
| @@ -1453,11 +1401,10 @@ void Hid::EndPermitVibrationSession(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::IsVibrationDeviceMounted(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::VibrationDeviceHandle vibration_device_handle; | ||||
|         Controller_NPad::DeviceHandle vibration_device_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
| @@ -1488,18 +1435,18 @@ void Hid::ActivateConsoleSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::StartConsoleSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::ConsoleSixAxisSensorHandle console_sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     LOG_WARNING(Service_HID, | ||||
|                 "(STUBBED) called, unknown_1={}, unknown_2={}, applet_resource_user_id={}", | ||||
|                 parameters.console_sixaxis_handle.unknown_1, | ||||
|                 parameters.console_sixaxis_handle.unknown_2, parameters.applet_resource_user_id); | ||||
|     LOG_WARNING( | ||||
|         Service_HID, | ||||
|         "(STUBBED) called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|         parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|         parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -1508,18 +1455,18 @@ void Hid::StartConsoleSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
| void Hid::StopConsoleSixAxisSensor(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     struct Parameters { | ||||
|         Core::HID::ConsoleSixAxisSensorHandle console_sixaxis_handle; | ||||
|         Controller_NPad::DeviceHandle sixaxis_handle; | ||||
|         INSERT_PADDING_WORDS_NOINIT(1); | ||||
|         u64 applet_resource_user_id; | ||||
|     }; | ||||
|     static_assert(sizeof(Parameters) == 0x10, "Parameters has incorrect size."); | ||||
|  | ||||
|     const auto parameters{rp.PopRaw<Parameters>()}; | ||||
|  | ||||
|     LOG_WARNING(Service_HID, | ||||
|                 "(STUBBED) called, unknown_1={}, unknown_2={}, applet_resource_user_id={}", | ||||
|                 parameters.console_sixaxis_handle.unknown_1, | ||||
|                 parameters.console_sixaxis_handle.unknown_2, parameters.applet_resource_user_id); | ||||
|     LOG_WARNING( | ||||
|         Service_HID, | ||||
|         "(STUBBED) called, npad_type={}, npad_id={}, device_index={}, applet_resource_user_id={}", | ||||
|         parameters.sixaxis_handle.npad_type, parameters.sixaxis_handle.npad_id, | ||||
|         parameters.sixaxis_handle.device_index, parameters.applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 2}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -1673,8 +1620,10 @@ void Hid::SetNpadCommunicationMode(Kernel::HLERequestContext& ctx) { | ||||
|  | ||||
| void Hid::GetNpadCommunicationMode(Kernel::HLERequestContext& ctx) { | ||||
|     IPC::RequestParser rp{ctx}; | ||||
|     const auto applet_resource_user_id{rp.Pop<u64>()}; | ||||
|  | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called"); | ||||
|     LOG_WARNING(Service_HID, "(STUBBED) called, applet_resource_user_id={}", | ||||
|                 applet_resource_user_id); | ||||
|  | ||||
|     IPC::ResponseBuilder rb{ctx, 4}; | ||||
|     rb.Push(ResultSuccess); | ||||
| @@ -2088,6 +2037,10 @@ public: | ||||
|     } | ||||
| }; | ||||
|  | ||||
| void ReloadInputDevices() { | ||||
|     Settings::values.is_device_reload_pending.store(true); | ||||
| } | ||||
|  | ||||
| void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system) { | ||||
|     std::make_shared<Hid>(system)->InstallAsService(service_manager); | ||||
|     std::make_shared<HidBus>(system)->InstallAsService(service_manager); | ||||
|   | ||||
| @@ -60,23 +60,21 @@ public: | ||||
| private: | ||||
|     template <typename T> | ||||
|     void MakeController(HidController controller) { | ||||
|         controllers[static_cast<std::size_t>(controller)] = std::make_unique<T>(system.HIDCore()); | ||||
|         controllers[static_cast<std::size_t>(controller)] = std::make_unique<T>(system); | ||||
|     } | ||||
|     template <typename T> | ||||
|     void MakeControllerWithServiceContext(HidController controller) { | ||||
|         controllers[static_cast<std::size_t>(controller)] = | ||||
|             std::make_unique<T>(system.HIDCore(), service_context); | ||||
|             std::make_unique<T>(system, service_context); | ||||
|     } | ||||
|  | ||||
|     void GetSharedMemoryHandle(Kernel::HLERequestContext& ctx); | ||||
|     void UpdateControllers(std::uintptr_t user_data, std::chrono::nanoseconds ns_late); | ||||
|     void UpdateKeyboard(std::uintptr_t user_data, std::chrono::nanoseconds ns_late); | ||||
|     void UpdateMotion(std::uintptr_t user_data, std::chrono::nanoseconds ns_late); | ||||
|  | ||||
|     KernelHelpers::ServiceContext& service_context; | ||||
|  | ||||
|     std::shared_ptr<Core::Timing::EventType> pad_update_event; | ||||
|     std::shared_ptr<Core::Timing::EventType> keyboard_update_event; | ||||
|     std::shared_ptr<Core::Timing::EventType> motion_update_event; | ||||
|  | ||||
|     std::array<std::unique_ptr<ControllerBase>, static_cast<size_t>(HidController::MaxControllers)> | ||||
| @@ -163,11 +161,38 @@ private: | ||||
|     void GetNpadCommunicationMode(Kernel::HLERequestContext& ctx); | ||||
|     void SetTouchScreenConfiguration(Kernel::HLERequestContext& ctx); | ||||
|  | ||||
|     enum class VibrationDeviceType : u32 { | ||||
|         Unknown = 0, | ||||
|         LinearResonantActuator = 1, | ||||
|         GcErm = 2, | ||||
|     }; | ||||
|  | ||||
|     enum class VibrationDevicePosition : u32 { | ||||
|         None = 0, | ||||
|         Left = 1, | ||||
|         Right = 2, | ||||
|     }; | ||||
|  | ||||
|     enum class VibrationGcErmCommand : u64 { | ||||
|         Stop = 0, | ||||
|         Start = 1, | ||||
|         StopHard = 2, | ||||
|     }; | ||||
|  | ||||
|     struct VibrationDeviceInfo { | ||||
|         VibrationDeviceType type{}; | ||||
|         VibrationDevicePosition position{}; | ||||
|     }; | ||||
|     static_assert(sizeof(VibrationDeviceInfo) == 0x8, "VibrationDeviceInfo has incorrect size."); | ||||
|  | ||||
|     std::shared_ptr<IAppletResource> applet_resource; | ||||
|  | ||||
|     KernelHelpers::ServiceContext service_context; | ||||
| }; | ||||
|  | ||||
| /// Reload input devices. Used when input configuration changed | ||||
| void ReloadInputDevices(); | ||||
|  | ||||
| /// Registers all HID services with the specified service manager. | ||||
| void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system); | ||||
|  | ||||
|   | ||||
| @@ -1,32 +1,36 @@ | ||||
| add_library(input_common STATIC | ||||
|     drivers/gc_adapter.cpp | ||||
|     drivers/gc_adapter.h | ||||
|     drivers/keyboard.cpp | ||||
|     drivers/keyboard.h | ||||
|     drivers/mouse.cpp | ||||
|     drivers/mouse.h | ||||
|     drivers/sdl_driver.cpp | ||||
|     drivers/sdl_driver.h | ||||
|     drivers/tas_input.cpp | ||||
|     drivers/tas_input.h | ||||
|     drivers/touch_screen.cpp | ||||
|     drivers/touch_screen.h | ||||
|     drivers/udp_client.cpp | ||||
|     drivers/udp_client.h | ||||
|     helpers/stick_from_buttons.cpp | ||||
|     helpers/stick_from_buttons.h | ||||
|     helpers/touch_from_buttons.cpp | ||||
|     helpers/touch_from_buttons.h | ||||
|     helpers/udp_protocol.cpp | ||||
|     helpers/udp_protocol.h | ||||
|     input_engine.cpp | ||||
|     input_engine.h | ||||
|     input_mapping.cpp | ||||
|     input_mapping.h | ||||
|     input_poller.cpp | ||||
|     input_poller.h | ||||
|     analog_from_button.cpp | ||||
|     analog_from_button.h | ||||
|     keyboard.cpp | ||||
|     keyboard.h | ||||
|     main.cpp | ||||
|     main.h | ||||
|     motion_from_button.cpp | ||||
|     motion_from_button.h | ||||
|     motion_input.cpp | ||||
|     motion_input.h | ||||
|     touch_from_button.cpp | ||||
|     touch_from_button.h | ||||
|     gcadapter/gc_adapter.cpp | ||||
|     gcadapter/gc_adapter.h | ||||
|     gcadapter/gc_poller.cpp | ||||
|     gcadapter/gc_poller.h | ||||
|     mouse/mouse_input.cpp | ||||
|     mouse/mouse_input.h | ||||
|     mouse/mouse_poller.cpp | ||||
|     mouse/mouse_poller.h | ||||
|     sdl/sdl.cpp | ||||
|     sdl/sdl.h | ||||
|     tas/tas_input.cpp | ||||
|     tas/tas_input.h | ||||
|     tas/tas_poller.cpp | ||||
|     tas/tas_poller.h | ||||
|     udp/client.cpp | ||||
|     udp/client.h | ||||
|     udp/protocol.cpp | ||||
|     udp/protocol.h | ||||
|     udp/udp.cpp | ||||
|     udp/udp.h | ||||
| ) | ||||
|  | ||||
| if (MSVC) | ||||
| @@ -53,8 +57,8 @@ endif() | ||||
|  | ||||
| if (ENABLE_SDL2) | ||||
|     target_sources(input_common PRIVATE | ||||
|         drivers/sdl_driver.cpp | ||||
|         drivers/sdl_driver.h | ||||
|         sdl/sdl_impl.cpp | ||||
|         sdl/sdl_impl.h | ||||
|     ) | ||||
|     target_link_libraries(input_common PRIVATE SDL2) | ||||
|     target_compile_definitions(input_common PRIVATE HAVE_SDL2) | ||||
|   | ||||
| @@ -4,164 +4,146 @@ | ||||
|  | ||||
| #include <memory> | ||||
| #include <thread> | ||||
| #include "common/input.h" | ||||
| #include "common/param_package.h" | ||||
| #include "input_common/drivers/gc_adapter.h" | ||||
| #include "input_common/drivers/keyboard.h" | ||||
| #include "input_common/drivers/mouse.h" | ||||
| #include "input_common/drivers/tas_input.h" | ||||
| #include "input_common/drivers/touch_screen.h" | ||||
| #include "input_common/drivers/udp_client.h" | ||||
| #include "input_common/helpers/stick_from_buttons.h" | ||||
| #include "input_common/helpers/touch_from_buttons.h" | ||||
| #include "input_common/input_engine.h" | ||||
| #include "input_common/input_mapping.h" | ||||
| #include "input_common/input_poller.h" | ||||
| #include "common/settings.h" | ||||
| #include "input_common/analog_from_button.h" | ||||
| #include "input_common/gcadapter/gc_adapter.h" | ||||
| #include "input_common/gcadapter/gc_poller.h" | ||||
| #include "input_common/keyboard.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/motion_from_button.h" | ||||
| #include "input_common/mouse/mouse_input.h" | ||||
| #include "input_common/mouse/mouse_poller.h" | ||||
| #include "input_common/tas/tas_input.h" | ||||
| #include "input_common/tas/tas_poller.h" | ||||
| #include "input_common/touch_from_button.h" | ||||
| #include "input_common/udp/client.h" | ||||
| #include "input_common/udp/udp.h" | ||||
| #ifdef HAVE_SDL2 | ||||
| #include "input_common/drivers/sdl_driver.h" | ||||
| #include "input_common/sdl/sdl.h" | ||||
| #endif | ||||
|  | ||||
| namespace InputCommon { | ||||
|  | ||||
| struct InputSubsystem::Impl { | ||||
|     void Initialize() { | ||||
|         mapping_factory = std::make_shared<MappingFactory>(); | ||||
|         MappingCallback mapping_callback{[this](MappingData data) { RegisterInput(data); }}; | ||||
|         gcadapter = std::make_shared<GCAdapter::Adapter>(); | ||||
|         gcbuttons = std::make_shared<GCButtonFactory>(gcadapter); | ||||
|         Input::RegisterFactory<Input::ButtonDevice>("gcpad", gcbuttons); | ||||
|         gcanalog = std::make_shared<GCAnalogFactory>(gcadapter); | ||||
|         Input::RegisterFactory<Input::AnalogDevice>("gcpad", gcanalog); | ||||
|         gcvibration = std::make_shared<GCVibrationFactory>(gcadapter); | ||||
|         Input::RegisterFactory<Input::VibrationDevice>("gcpad", gcvibration); | ||||
|  | ||||
|         keyboard = std::make_shared<Keyboard>("keyboard"); | ||||
|         keyboard->SetMappingCallback(mapping_callback); | ||||
|         keyboard_factory = std::make_shared<InputFactory>(keyboard); | ||||
|         keyboard_output_factory = std::make_shared<OutputFactory>(keyboard); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(keyboard->GetEngineName(), | ||||
|                                                                    keyboard_factory); | ||||
|         Common::Input::RegisterFactory<Common::Input::OutputDevice>(keyboard->GetEngineName(), | ||||
|                                                                     keyboard_output_factory); | ||||
|  | ||||
|         mouse = std::make_shared<Mouse>("mouse"); | ||||
|         mouse->SetMappingCallback(mapping_callback); | ||||
|         mouse_factory = std::make_shared<InputFactory>(mouse); | ||||
|         mouse_output_factory = std::make_shared<OutputFactory>(mouse); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(mouse->GetEngineName(), | ||||
|                                                                    mouse_factory); | ||||
|         Common::Input::RegisterFactory<Common::Input::OutputDevice>(mouse->GetEngineName(), | ||||
|                                                                     mouse_output_factory); | ||||
|  | ||||
|         touch_screen = std::make_shared<TouchScreen>("touch"); | ||||
|         touch_screen_factory = std::make_shared<InputFactory>(touch_screen); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(touch_screen->GetEngineName(), | ||||
|                                                                    touch_screen_factory); | ||||
|  | ||||
|         gcadapter = std::make_shared<GCAdapter>("gcpad"); | ||||
|         gcadapter->SetMappingCallback(mapping_callback); | ||||
|         gcadapter_input_factory = std::make_shared<InputFactory>(gcadapter); | ||||
|         gcadapter_output_factory = std::make_shared<OutputFactory>(gcadapter); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(gcadapter->GetEngineName(), | ||||
|                                                                    gcadapter_input_factory); | ||||
|         Common::Input::RegisterFactory<Common::Input::OutputDevice>(gcadapter->GetEngineName(), | ||||
|                                                                     gcadapter_output_factory); | ||||
|  | ||||
|         udp_client = std::make_shared<CemuhookUDP::UDPClient>("cemuhookudp"); | ||||
|         udp_client->SetMappingCallback(mapping_callback); | ||||
|         udp_client_factory = std::make_shared<InputFactory>(udp_client); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(udp_client->GetEngineName(), | ||||
|                                                                    udp_client_factory); | ||||
|  | ||||
|         tas_input = std::make_shared<TasInput::Tas>("tas"); | ||||
|         tas_input->SetMappingCallback(mapping_callback); | ||||
|         tas_input_factory = std::make_shared<InputFactory>(tas_input); | ||||
|         tas_output_factory = std::make_shared<OutputFactory>(tas_input); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(tas_input->GetEngineName(), | ||||
|                                                                    tas_input_factory); | ||||
|         Common::Input::RegisterFactory<Common::Input::OutputDevice>(tas_input->GetEngineName(), | ||||
|                                                                     tas_output_factory); | ||||
|         keyboard = std::make_shared<Keyboard>(); | ||||
|         Input::RegisterFactory<Input::ButtonDevice>("keyboard", keyboard); | ||||
|         Input::RegisterFactory<Input::AnalogDevice>("analog_from_button", | ||||
|                                                     std::make_shared<AnalogFromButton>()); | ||||
|         Input::RegisterFactory<Input::MotionDevice>("keyboard", | ||||
|                                                     std::make_shared<MotionFromButton>()); | ||||
|         Input::RegisterFactory<Input::TouchDevice>("touch_from_button", | ||||
|                                                    std::make_shared<TouchFromButtonFactory>()); | ||||
|  | ||||
| #ifdef HAVE_SDL2 | ||||
|         sdl = std::make_shared<SDLDriver>("sdl"); | ||||
|         sdl->SetMappingCallback(mapping_callback); | ||||
|         sdl_input_factory = std::make_shared<InputFactory>(sdl); | ||||
|         sdl_output_factory = std::make_shared<OutputFactory>(sdl); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>(sdl->GetEngineName(), | ||||
|                                                                    sdl_input_factory); | ||||
|         Common::Input::RegisterFactory<Common::Input::OutputDevice>(sdl->GetEngineName(), | ||||
|                                                                     sdl_output_factory); | ||||
|         sdl = SDL::Init(); | ||||
| #endif | ||||
|  | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>( | ||||
|             "touch_from_button", std::make_shared<TouchFromButton>()); | ||||
|         Common::Input::RegisterFactory<Common::Input::InputDevice>( | ||||
|             "analog_from_button", std::make_shared<StickFromButton>()); | ||||
|         udp = std::make_shared<InputCommon::CemuhookUDP::Client>(); | ||||
|         udpmotion = std::make_shared<UDPMotionFactory>(udp); | ||||
|         Input::RegisterFactory<Input::MotionDevice>("cemuhookudp", udpmotion); | ||||
|         udptouch = std::make_shared<UDPTouchFactory>(udp); | ||||
|         Input::RegisterFactory<Input::TouchDevice>("cemuhookudp", udptouch); | ||||
|  | ||||
|         mouse = std::make_shared<MouseInput::Mouse>(); | ||||
|         mousebuttons = std::make_shared<MouseButtonFactory>(mouse); | ||||
|         Input::RegisterFactory<Input::ButtonDevice>("mouse", mousebuttons); | ||||
|         mouseanalog = std::make_shared<MouseAnalogFactory>(mouse); | ||||
|         Input::RegisterFactory<Input::AnalogDevice>("mouse", mouseanalog); | ||||
|         mousemotion = std::make_shared<MouseMotionFactory>(mouse); | ||||
|         Input::RegisterFactory<Input::MotionDevice>("mouse", mousemotion); | ||||
|         mousetouch = std::make_shared<MouseTouchFactory>(mouse); | ||||
|         Input::RegisterFactory<Input::TouchDevice>("mouse", mousetouch); | ||||
|  | ||||
|         tas = std::make_shared<TasInput::Tas>(); | ||||
|         tasbuttons = std::make_shared<TasButtonFactory>(tas); | ||||
|         Input::RegisterFactory<Input::ButtonDevice>("tas", tasbuttons); | ||||
|         tasanalog = std::make_shared<TasAnalogFactory>(tas); | ||||
|         Input::RegisterFactory<Input::AnalogDevice>("tas", tasanalog); | ||||
|     } | ||||
|  | ||||
|     void Shutdown() { | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(keyboard->GetEngineName()); | ||||
|         Common::Input::UnregisterFactory<Common::Input::OutputDevice>(keyboard->GetEngineName()); | ||||
|         Input::UnregisterFactory<Input::ButtonDevice>("keyboard"); | ||||
|         Input::UnregisterFactory<Input::MotionDevice>("keyboard"); | ||||
|         keyboard.reset(); | ||||
|  | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(mouse->GetEngineName()); | ||||
|         Common::Input::UnregisterFactory<Common::Input::OutputDevice>(mouse->GetEngineName()); | ||||
|         mouse.reset(); | ||||
|  | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(touch_screen->GetEngineName()); | ||||
|         touch_screen.reset(); | ||||
|  | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(gcadapter->GetEngineName()); | ||||
|         Common::Input::UnregisterFactory<Common::Input::OutputDevice>(gcadapter->GetEngineName()); | ||||
|         gcadapter.reset(); | ||||
|  | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(udp_client->GetEngineName()); | ||||
|         udp_client.reset(); | ||||
|  | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(tas_input->GetEngineName()); | ||||
|         Common::Input::UnregisterFactory<Common::Input::OutputDevice>(tas_input->GetEngineName()); | ||||
|         tas_input.reset(); | ||||
|  | ||||
|         Input::UnregisterFactory<Input::AnalogDevice>("analog_from_button"); | ||||
|         Input::UnregisterFactory<Input::TouchDevice>("touch_from_button"); | ||||
| #ifdef HAVE_SDL2 | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>(sdl->GetEngineName()); | ||||
|         Common::Input::UnregisterFactory<Common::Input::OutputDevice>(sdl->GetEngineName()); | ||||
|         sdl.reset(); | ||||
| #endif | ||||
|         Input::UnregisterFactory<Input::ButtonDevice>("gcpad"); | ||||
|         Input::UnregisterFactory<Input::AnalogDevice>("gcpad"); | ||||
|         Input::UnregisterFactory<Input::VibrationDevice>("gcpad"); | ||||
|  | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>("touch_from_button"); | ||||
|         Common::Input::UnregisterFactory<Common::Input::InputDevice>("analog_from_button"); | ||||
|         gcbuttons.reset(); | ||||
|         gcanalog.reset(); | ||||
|         gcvibration.reset(); | ||||
|  | ||||
|         Input::UnregisterFactory<Input::MotionDevice>("cemuhookudp"); | ||||
|         Input::UnregisterFactory<Input::TouchDevice>("cemuhookudp"); | ||||
|  | ||||
|         udpmotion.reset(); | ||||
|         udptouch.reset(); | ||||
|  | ||||
|         Input::UnregisterFactory<Input::ButtonDevice>("mouse"); | ||||
|         Input::UnregisterFactory<Input::AnalogDevice>("mouse"); | ||||
|         Input::UnregisterFactory<Input::MotionDevice>("mouse"); | ||||
|         Input::UnregisterFactory<Input::TouchDevice>("mouse"); | ||||
|  | ||||
|         mousebuttons.reset(); | ||||
|         mouseanalog.reset(); | ||||
|         mousemotion.reset(); | ||||
|         mousetouch.reset(); | ||||
|  | ||||
|         Input::UnregisterFactory<Input::ButtonDevice>("tas"); | ||||
|         Input::UnregisterFactory<Input::AnalogDevice>("tas"); | ||||
|  | ||||
|         tasbuttons.reset(); | ||||
|         tasanalog.reset(); | ||||
|     } | ||||
|  | ||||
|     [[nodiscard]] std::vector<Common::ParamPackage> GetInputDevices() const { | ||||
|         std::vector<Common::ParamPackage> devices = { | ||||
|             Common::ParamPackage{{"display", "Any"}, {"engine", "any"}}, | ||||
|             Common::ParamPackage{{"display", "Any"}, {"class", "any"}}, | ||||
|             Common::ParamPackage{{"display", "Keyboard/Mouse"}, {"class", "keyboard"}}, | ||||
|         }; | ||||
|  | ||||
|         auto keyboard_devices = keyboard->GetInputDevices(); | ||||
|         devices.insert(devices.end(), keyboard_devices.begin(), keyboard_devices.end()); | ||||
|         auto mouse_devices = mouse->GetInputDevices(); | ||||
|         devices.insert(devices.end(), mouse_devices.begin(), mouse_devices.end()); | ||||
|         auto gcadapter_devices = gcadapter->GetInputDevices(); | ||||
|         devices.insert(devices.end(), gcadapter_devices.begin(), gcadapter_devices.end()); | ||||
|         if (Settings::values.tas_enable) { | ||||
|             devices.emplace_back( | ||||
|                 Common::ParamPackage{{"display", "TAS Controller"}, {"class", "tas"}}); | ||||
|         } | ||||
| #ifdef HAVE_SDL2 | ||||
|         auto sdl_devices = sdl->GetInputDevices(); | ||||
|         devices.insert(devices.end(), sdl_devices.begin(), sdl_devices.end()); | ||||
| #endif | ||||
|  | ||||
|         auto udp_devices = udp->GetInputDevices(); | ||||
|         devices.insert(devices.end(), udp_devices.begin(), udp_devices.end()); | ||||
|         auto gcpad_devices = gcadapter->GetInputDevices(); | ||||
|         devices.insert(devices.end(), gcpad_devices.begin(), gcpad_devices.end()); | ||||
|         return devices; | ||||
|     } | ||||
|  | ||||
|     [[nodiscard]] AnalogMapping GetAnalogMappingForDevice( | ||||
|         const Common::ParamPackage& params) const { | ||||
|         if (!params.Has("engine") || params.Get("engine", "") == "any") { | ||||
|         if (!params.Has("class") || params.Get("class", "") == "any") { | ||||
|             return {}; | ||||
|         } | ||||
|         const std::string engine = params.Get("engine", ""); | ||||
|         if (engine == mouse->GetEngineName()) { | ||||
|             return mouse->GetAnalogMappingForDevice(params); | ||||
|         } | ||||
|         if (engine == gcadapter->GetEngineName()) { | ||||
|         if (params.Get("class", "") == "gcpad") { | ||||
|             return gcadapter->GetAnalogMappingForDevice(params); | ||||
|         } | ||||
|         if (engine == tas_input->GetEngineName()) { | ||||
|             return tas_input->GetAnalogMappingForDevice(params); | ||||
|         if (params.Get("class", "") == "tas") { | ||||
|             return tas->GetAnalogMappingForDevice(params); | ||||
|         } | ||||
| #ifdef HAVE_SDL2 | ||||
|         if (engine == sdl->GetEngineName()) { | ||||
|         if (params.Get("class", "") == "sdl") { | ||||
|             return sdl->GetAnalogMappingForDevice(params); | ||||
|         } | ||||
| #endif | ||||
| @@ -170,18 +152,17 @@ struct InputSubsystem::Impl { | ||||
|  | ||||
|     [[nodiscard]] ButtonMapping GetButtonMappingForDevice( | ||||
|         const Common::ParamPackage& params) const { | ||||
|         if (!params.Has("engine") || params.Get("engine", "") == "any") { | ||||
|         if (!params.Has("class") || params.Get("class", "") == "any") { | ||||
|             return {}; | ||||
|         } | ||||
|         const std::string engine = params.Get("engine", ""); | ||||
|         if (engine == gcadapter->GetEngineName()) { | ||||
|         if (params.Get("class", "") == "gcpad") { | ||||
|             return gcadapter->GetButtonMappingForDevice(params); | ||||
|         } | ||||
|         if (engine == tas_input->GetEngineName()) { | ||||
|             return tas_input->GetButtonMappingForDevice(params); | ||||
|         if (params.Get("class", "") == "tas") { | ||||
|             return tas->GetButtonMappingForDevice(params); | ||||
|         } | ||||
| #ifdef HAVE_SDL2 | ||||
|         if (engine == sdl->GetEngineName()) { | ||||
|         if (params.Get("class", "") == "sdl") { | ||||
|             return sdl->GetButtonMappingForDevice(params); | ||||
|         } | ||||
| #endif | ||||
| @@ -190,115 +171,40 @@ struct InputSubsystem::Impl { | ||||
|  | ||||
|     [[nodiscard]] MotionMapping GetMotionMappingForDevice( | ||||
|         const Common::ParamPackage& params) const { | ||||
|         if (!params.Has("engine") || params.Get("engine", "") == "any") { | ||||
|         if (!params.Has("class") || params.Get("class", "") == "any") { | ||||
|             return {}; | ||||
|         } | ||||
|         const std::string engine = params.Get("engine", ""); | ||||
|         if (engine == gcadapter->GetEngineName()) { | ||||
|             return gcadapter->GetMotionMappingForDevice(params); | ||||
|         if (params.Get("class", "") == "cemuhookudp") { | ||||
|             // TODO return the correct motion device | ||||
|             return {}; | ||||
|         } | ||||
| #ifdef HAVE_SDL2 | ||||
|         if (engine == sdl->GetEngineName()) { | ||||
|         if (params.Get("class", "") == "sdl") { | ||||
|             return sdl->GetMotionMappingForDevice(params); | ||||
|         } | ||||
| #endif | ||||
|         return {}; | ||||
|     } | ||||
|  | ||||
|     std::string GetButtonName(const Common::ParamPackage& params) const { | ||||
|         if (!params.Has("engine") || params.Get("engine", "") == "any") { | ||||
|             return "Unknown"; | ||||
|         } | ||||
|         const std::string engine = params.Get("engine", ""); | ||||
|         if (engine == mouse->GetEngineName()) { | ||||
|             return mouse->GetUIName(params); | ||||
|         } | ||||
|         if (engine == gcadapter->GetEngineName()) { | ||||
|             return gcadapter->GetUIName(params); | ||||
|         } | ||||
|         if (engine == udp_client->GetEngineName()) { | ||||
|             return udp_client->GetUIName(params); | ||||
|         } | ||||
|         if (engine == tas_input->GetEngineName()) { | ||||
|             return tas_input->GetUIName(params); | ||||
|         } | ||||
| #ifdef HAVE_SDL2 | ||||
|         if (engine == sdl->GetEngineName()) { | ||||
|             return sdl->GetUIName(params); | ||||
|         } | ||||
| #endif | ||||
|         return "Bad engine"; | ||||
|     } | ||||
|  | ||||
|     bool IsController(const Common::ParamPackage& params) { | ||||
|         const std::string engine = params.Get("engine", ""); | ||||
|         if (engine == mouse->GetEngineName()) { | ||||
|             return true; | ||||
|         } | ||||
|         if (engine == gcadapter->GetEngineName()) { | ||||
|             return true; | ||||
|         } | ||||
|         if (engine == tas_input->GetEngineName()) { | ||||
|             return true; | ||||
|         } | ||||
| #ifdef HAVE_SDL2 | ||||
|         if (engine == sdl->GetEngineName()) { | ||||
|             return true; | ||||
|         } | ||||
| #endif | ||||
|         return false; | ||||
|     } | ||||
|  | ||||
|     void BeginConfiguration() { | ||||
|         keyboard->BeginConfiguration(); | ||||
|         mouse->BeginConfiguration(); | ||||
|         gcadapter->BeginConfiguration(); | ||||
|         udp_client->BeginConfiguration(); | ||||
| #ifdef HAVE_SDL2 | ||||
|         sdl->BeginConfiguration(); | ||||
| #endif | ||||
|     } | ||||
|  | ||||
|     void EndConfiguration() { | ||||
|         keyboard->EndConfiguration(); | ||||
|         mouse->EndConfiguration(); | ||||
|         gcadapter->EndConfiguration(); | ||||
|         udp_client->EndConfiguration(); | ||||
| #ifdef HAVE_SDL2 | ||||
|         sdl->EndConfiguration(); | ||||
| #endif | ||||
|     } | ||||
|  | ||||
|     void RegisterInput(MappingData data) { | ||||
|         mapping_factory->RegisterInput(data); | ||||
|     } | ||||
|  | ||||
|     std::shared_ptr<MappingFactory> mapping_factory; | ||||
|  | ||||
|     std::shared_ptr<Keyboard> keyboard; | ||||
|     std::shared_ptr<Mouse> mouse; | ||||
|     std::shared_ptr<GCAdapter> gcadapter; | ||||
|     std::shared_ptr<TouchScreen> touch_screen; | ||||
|     std::shared_ptr<TasInput::Tas> tas_input; | ||||
|     std::shared_ptr<CemuhookUDP::UDPClient> udp_client; | ||||
|  | ||||
|     std::shared_ptr<InputFactory> keyboard_factory; | ||||
|     std::shared_ptr<InputFactory> mouse_factory; | ||||
|     std::shared_ptr<InputFactory> gcadapter_input_factory; | ||||
|     std::shared_ptr<InputFactory> touch_screen_factory; | ||||
|     std::shared_ptr<InputFactory> udp_client_factory; | ||||
|     std::shared_ptr<InputFactory> tas_input_factory; | ||||
|  | ||||
|     std::shared_ptr<OutputFactory> keyboard_output_factory; | ||||
|     std::shared_ptr<OutputFactory> mouse_output_factory; | ||||
|     std::shared_ptr<OutputFactory> gcadapter_output_factory; | ||||
|     std::shared_ptr<OutputFactory> tas_output_factory; | ||||
|  | ||||
| #ifdef HAVE_SDL2 | ||||
|     std::shared_ptr<SDLDriver> sdl; | ||||
|     std::shared_ptr<InputFactory> sdl_input_factory; | ||||
|     std::shared_ptr<OutputFactory> sdl_output_factory; | ||||
|     std::unique_ptr<SDL::State> sdl; | ||||
| #endif | ||||
|     std::shared_ptr<GCButtonFactory> gcbuttons; | ||||
|     std::shared_ptr<GCAnalogFactory> gcanalog; | ||||
|     std::shared_ptr<GCVibrationFactory> gcvibration; | ||||
|     std::shared_ptr<UDPMotionFactory> udpmotion; | ||||
|     std::shared_ptr<UDPTouchFactory> udptouch; | ||||
|     std::shared_ptr<MouseButtonFactory> mousebuttons; | ||||
|     std::shared_ptr<MouseAnalogFactory> mouseanalog; | ||||
|     std::shared_ptr<MouseMotionFactory> mousemotion; | ||||
|     std::shared_ptr<MouseTouchFactory> mousetouch; | ||||
|     std::shared_ptr<TasButtonFactory> tasbuttons; | ||||
|     std::shared_ptr<TasAnalogFactory> tasanalog; | ||||
|     std::shared_ptr<CemuhookUDP::Client> udp; | ||||
|     std::shared_ptr<GCAdapter::Adapter> gcadapter; | ||||
|     std::shared_ptr<MouseInput::Mouse> mouse; | ||||
|     std::shared_ptr<TasInput::Tas> tas; | ||||
| }; | ||||
|  | ||||
| InputSubsystem::InputSubsystem() : impl{std::make_unique<Impl>()} {} | ||||
| @@ -321,28 +227,20 @@ const Keyboard* InputSubsystem::GetKeyboard() const { | ||||
|     return impl->keyboard.get(); | ||||
| } | ||||
|  | ||||
| Mouse* InputSubsystem::GetMouse() { | ||||
| MouseInput::Mouse* InputSubsystem::GetMouse() { | ||||
|     return impl->mouse.get(); | ||||
| } | ||||
|  | ||||
| const Mouse* InputSubsystem::GetMouse() const { | ||||
| const MouseInput::Mouse* InputSubsystem::GetMouse() const { | ||||
|     return impl->mouse.get(); | ||||
| } | ||||
|  | ||||
| TouchScreen* InputSubsystem::GetTouchScreen() { | ||||
|     return impl->touch_screen.get(); | ||||
| } | ||||
|  | ||||
| const TouchScreen* InputSubsystem::GetTouchScreen() const { | ||||
|     return impl->touch_screen.get(); | ||||
| } | ||||
|  | ||||
| TasInput::Tas* InputSubsystem::GetTas() { | ||||
|     return impl->tas_input.get(); | ||||
|     return impl->tas.get(); | ||||
| } | ||||
|  | ||||
| const TasInput::Tas* InputSubsystem::GetTas() const { | ||||
|     return impl->tas_input.get(); | ||||
|     return impl->tas.get(); | ||||
| } | ||||
|  | ||||
| std::vector<Common::ParamPackage> InputSubsystem::GetInputDevices() const { | ||||
| @@ -361,37 +259,100 @@ MotionMapping InputSubsystem::GetMotionMappingForDevice(const Common::ParamPacka | ||||
|     return impl->GetMotionMappingForDevice(device); | ||||
| } | ||||
|  | ||||
| std::string InputSubsystem::GetButtonName(const Common::ParamPackage& params) const { | ||||
|     const std::string toggle = params.Get("toggle", false) ? "~" : ""; | ||||
|     const std::string inverted = params.Get("inverted", false) ? "!" : ""; | ||||
|     const std::string button_name = impl->GetButtonName(params); | ||||
|     std::string axis_direction = ""; | ||||
|     if (params.Has("axis")) { | ||||
|         axis_direction = params.Get("invert", "+"); | ||||
|     } | ||||
|     return fmt::format("{}{}{}{}", toggle, inverted, button_name, axis_direction); | ||||
| GCAnalogFactory* InputSubsystem::GetGCAnalogs() { | ||||
|     return impl->gcanalog.get(); | ||||
| } | ||||
|  | ||||
| bool InputSubsystem::IsController(const Common::ParamPackage& params) const { | ||||
|     return impl->IsController(params); | ||||
| const GCAnalogFactory* InputSubsystem::GetGCAnalogs() const { | ||||
|     return impl->gcanalog.get(); | ||||
| } | ||||
|  | ||||
| GCButtonFactory* InputSubsystem::GetGCButtons() { | ||||
|     return impl->gcbuttons.get(); | ||||
| } | ||||
|  | ||||
| const GCButtonFactory* InputSubsystem::GetGCButtons() const { | ||||
|     return impl->gcbuttons.get(); | ||||
| } | ||||
|  | ||||
| UDPMotionFactory* InputSubsystem::GetUDPMotions() { | ||||
|     return impl->udpmotion.get(); | ||||
| } | ||||
|  | ||||
| const UDPMotionFactory* InputSubsystem::GetUDPMotions() const { | ||||
|     return impl->udpmotion.get(); | ||||
| } | ||||
|  | ||||
| UDPTouchFactory* InputSubsystem::GetUDPTouch() { | ||||
|     return impl->udptouch.get(); | ||||
| } | ||||
|  | ||||
| const UDPTouchFactory* InputSubsystem::GetUDPTouch() const { | ||||
|     return impl->udptouch.get(); | ||||
| } | ||||
|  | ||||
| MouseButtonFactory* InputSubsystem::GetMouseButtons() { | ||||
|     return impl->mousebuttons.get(); | ||||
| } | ||||
|  | ||||
| const MouseButtonFactory* InputSubsystem::GetMouseButtons() const { | ||||
|     return impl->mousebuttons.get(); | ||||
| } | ||||
|  | ||||
| MouseAnalogFactory* InputSubsystem::GetMouseAnalogs() { | ||||
|     return impl->mouseanalog.get(); | ||||
| } | ||||
|  | ||||
| const MouseAnalogFactory* InputSubsystem::GetMouseAnalogs() const { | ||||
|     return impl->mouseanalog.get(); | ||||
| } | ||||
|  | ||||
| MouseMotionFactory* InputSubsystem::GetMouseMotions() { | ||||
|     return impl->mousemotion.get(); | ||||
| } | ||||
|  | ||||
| const MouseMotionFactory* InputSubsystem::GetMouseMotions() const { | ||||
|     return impl->mousemotion.get(); | ||||
| } | ||||
|  | ||||
| MouseTouchFactory* InputSubsystem::GetMouseTouch() { | ||||
|     return impl->mousetouch.get(); | ||||
| } | ||||
|  | ||||
| const MouseTouchFactory* InputSubsystem::GetMouseTouch() const { | ||||
|     return impl->mousetouch.get(); | ||||
| } | ||||
|  | ||||
| TasButtonFactory* InputSubsystem::GetTasButtons() { | ||||
|     return impl->tasbuttons.get(); | ||||
| } | ||||
|  | ||||
| const TasButtonFactory* InputSubsystem::GetTasButtons() const { | ||||
|     return impl->tasbuttons.get(); | ||||
| } | ||||
|  | ||||
| TasAnalogFactory* InputSubsystem::GetTasAnalogs() { | ||||
|     return impl->tasanalog.get(); | ||||
| } | ||||
|  | ||||
| const TasAnalogFactory* InputSubsystem::GetTasAnalogs() const { | ||||
|     return impl->tasanalog.get(); | ||||
| } | ||||
|  | ||||
| void InputSubsystem::ReloadInputDevices() { | ||||
|     impl->udp_client.get()->ReloadSockets(); | ||||
|     if (!impl->udp) { | ||||
|         return; | ||||
|     } | ||||
|     impl->udp->ReloadSockets(); | ||||
| } | ||||
|  | ||||
| void InputSubsystem::BeginMapping(Polling::InputType type) { | ||||
|     impl->BeginConfiguration(); | ||||
|     impl->mapping_factory->BeginMapping(type); | ||||
| } | ||||
|  | ||||
| const Common::ParamPackage InputSubsystem::GetNextInput() const { | ||||
|     return impl->mapping_factory->GetNextInput(); | ||||
| } | ||||
|  | ||||
| void InputSubsystem::StopMapping() const { | ||||
|     impl->EndConfiguration(); | ||||
|     impl->mapping_factory->StopMapping(); | ||||
| std::vector<std::unique_ptr<Polling::DevicePoller>> InputSubsystem::GetPollers( | ||||
|     [[maybe_unused]] Polling::DeviceType type) const { | ||||
| #ifdef HAVE_SDL2 | ||||
|     return impl->sdl->GetPollers(type); | ||||
| #else | ||||
|     return {}; | ||||
| #endif | ||||
| } | ||||
|  | ||||
| std::string GenerateKeyboardParam(int key_code) { | ||||
| @@ -402,15 +363,6 @@ std::string GenerateKeyboardParam(int key_code) { | ||||
|     return param.Serialize(); | ||||
| } | ||||
|  | ||||
| std::string GenerateModdifierKeyboardParam(int key_code) { | ||||
|     Common::ParamPackage param; | ||||
|     param.Set("engine", "keyboard"); | ||||
|     param.Set("code", key_code); | ||||
|     param.Set("toggle", false); | ||||
|     param.Set("pad", 1); | ||||
|     return param.Serialize(); | ||||
| } | ||||
|  | ||||
| std::string GenerateAnalogParamFromKeys(int key_up, int key_down, int key_left, int key_right, | ||||
|                                         int key_modifier, float modifier_scale) { | ||||
|     Common::ParamPackage circle_pad_param{ | ||||
|   | ||||
| @@ -25,26 +25,56 @@ namespace Settings::NativeMotion { | ||||
| enum Values : int; | ||||
| } | ||||
|  | ||||
| namespace InputCommon { | ||||
| class Keyboard; | ||||
| namespace MouseInput { | ||||
| class Mouse; | ||||
| class TouchScreen; | ||||
| struct MappingData; | ||||
| } // namespace InputCommon | ||||
| } | ||||
|  | ||||
| namespace InputCommon::TasInput { | ||||
| namespace TasInput { | ||||
| class Tas; | ||||
| } // namespace InputCommon::TasInput | ||||
| } | ||||
|  | ||||
| namespace InputCommon { | ||||
| namespace Polling { | ||||
| /// Type of input desired for mapping purposes | ||||
| enum class InputType { None, Button, Stick, Motion, Touch }; | ||||
|  | ||||
| enum class DeviceType { Button, AnalogPreferred, Motion }; | ||||
|  | ||||
| /** | ||||
|  * A class that can be used to get inputs from an input device like controllers without having to | ||||
|  * poll the device's status yourself | ||||
|  */ | ||||
| class DevicePoller { | ||||
| public: | ||||
|     virtual ~DevicePoller() = default; | ||||
|     /// Setup and start polling for inputs, should be called before GetNextInput | ||||
|     /// If a device_id is provided, events should be filtered to only include events from this | ||||
|     /// device id | ||||
|     virtual void Start(const std::string& device_id = "") = 0; | ||||
|     /// Stop polling | ||||
|     virtual void Stop() = 0; | ||||
|     /** | ||||
|      * Every call to this function returns the next input recorded since calling Start | ||||
|      * @return A ParamPackage of the recorded input, which can be used to create an InputDevice. | ||||
|      *         If there has been no input, the package is empty | ||||
|      */ | ||||
|     virtual Common::ParamPackage GetNextInput() = 0; | ||||
| }; | ||||
| } // namespace Polling | ||||
|  | ||||
| class GCAnalogFactory; | ||||
| class GCButtonFactory; | ||||
| class UDPMotionFactory; | ||||
| class UDPTouchFactory; | ||||
| class MouseButtonFactory; | ||||
| class MouseAnalogFactory; | ||||
| class MouseMotionFactory; | ||||
| class MouseTouchFactory; | ||||
| class TasButtonFactory; | ||||
| class TasAnalogFactory; | ||||
| class Keyboard; | ||||
|  | ||||
| /** | ||||
|  * Given a ParamPackage for a Device returned from `GetInputDevices`, attempt to get the default | ||||
|  * mapping for the device. | ||||
|  * mapping for the device. This is currently only implemented for the SDL backend devices. | ||||
|  */ | ||||
| using AnalogMapping = std::unordered_map<Settings::NativeAnalog::Values, Common::ParamPackage>; | ||||
| using ButtonMapping = std::unordered_map<Settings::NativeButton::Values, Common::ParamPackage>; | ||||
| @@ -74,27 +104,20 @@ public: | ||||
|     [[nodiscard]] const Keyboard* GetKeyboard() const; | ||||
|  | ||||
|     /// Retrieves the underlying mouse device. | ||||
|     [[nodiscard]] Mouse* GetMouse(); | ||||
|     [[nodiscard]] MouseInput::Mouse* GetMouse(); | ||||
|  | ||||
|     /// Retrieves the underlying mouse device. | ||||
|     [[nodiscard]] const Mouse* GetMouse() const; | ||||
|     [[nodiscard]] const MouseInput::Mouse* GetMouse() const; | ||||
|  | ||||
|     /// Retrieves the underlying touch screen device. | ||||
|     [[nodiscard]] TouchScreen* GetTouchScreen(); | ||||
|  | ||||
|     /// Retrieves the underlying touch screen device. | ||||
|     [[nodiscard]] const TouchScreen* GetTouchScreen() const; | ||||
|  | ||||
|     /// Retrieves the underlying tas input device. | ||||
|     /// Retrieves the underlying tas device. | ||||
|     [[nodiscard]] TasInput::Tas* GetTas(); | ||||
|  | ||||
|     /// Retrieves the underlying  tas input  device. | ||||
|     /// Retrieves the underlying tas device. | ||||
|     [[nodiscard]] const TasInput::Tas* GetTas() const; | ||||
|  | ||||
|     /** | ||||
|      * Returns all available input devices that this Factory can create a new device with. | ||||
|      * Each returned ParamPackage should have a `display` field used for display, a `engine` field | ||||
|      * for backends to determine if this backend is meant to service the request and any other | ||||
|      * Each returned ParamPackage should have a `display` field used for display, a class field for | ||||
|      * backends to determine if this backend is meant to service the request and any other | ||||
|      * information needed to identify this in the backend later. | ||||
|      */ | ||||
|     [[nodiscard]] std::vector<Common::ParamPackage> GetInputDevices() const; | ||||
| @@ -108,36 +131,83 @@ public: | ||||
|     /// Retrieves the motion mappings for the given device. | ||||
|     [[nodiscard]] MotionMapping GetMotionMappingForDevice(const Common::ParamPackage& device) const; | ||||
|  | ||||
|     /// Returns a string contaning the name of the button from the input engine. | ||||
|     [[nodiscard]] std::string GetButtonName(const Common::ParamPackage& params) const; | ||||
|     /// Retrieves the underlying GameCube analog handler. | ||||
|     [[nodiscard]] GCAnalogFactory* GetGCAnalogs(); | ||||
|  | ||||
|     /// Returns true if device is a controller. | ||||
|     [[nodiscard]] bool IsController(const Common::ParamPackage& params) const; | ||||
|     /// Retrieves the underlying GameCube analog handler. | ||||
|     [[nodiscard]] const GCAnalogFactory* GetGCAnalogs() const; | ||||
|  | ||||
|     /// Reloads the input devices. | ||||
|     /// Retrieves the underlying GameCube button handler. | ||||
|     [[nodiscard]] GCButtonFactory* GetGCButtons(); | ||||
|  | ||||
|     /// Retrieves the underlying GameCube button handler. | ||||
|     [[nodiscard]] const GCButtonFactory* GetGCButtons() const; | ||||
|  | ||||
|     /// Retrieves the underlying udp motion handler. | ||||
|     [[nodiscard]] UDPMotionFactory* GetUDPMotions(); | ||||
|  | ||||
|     /// Retrieves the underlying udp motion handler. | ||||
|     [[nodiscard]] const UDPMotionFactory* GetUDPMotions() const; | ||||
|  | ||||
|     /// Retrieves the underlying udp touch handler. | ||||
|     [[nodiscard]] UDPTouchFactory* GetUDPTouch(); | ||||
|  | ||||
|     /// Retrieves the underlying udp touch handler. | ||||
|     [[nodiscard]] const UDPTouchFactory* GetUDPTouch() const; | ||||
|  | ||||
|     /// Retrieves the underlying mouse button handler. | ||||
|     [[nodiscard]] MouseButtonFactory* GetMouseButtons(); | ||||
|  | ||||
|     /// Retrieves the underlying mouse button handler. | ||||
|     [[nodiscard]] const MouseButtonFactory* GetMouseButtons() const; | ||||
|  | ||||
|     /// Retrieves the underlying mouse analog handler. | ||||
|     [[nodiscard]] MouseAnalogFactory* GetMouseAnalogs(); | ||||
|  | ||||
|     /// Retrieves the underlying mouse analog handler. | ||||
|     [[nodiscard]] const MouseAnalogFactory* GetMouseAnalogs() const; | ||||
|  | ||||
|     /// Retrieves the underlying mouse motion handler. | ||||
|     [[nodiscard]] MouseMotionFactory* GetMouseMotions(); | ||||
|  | ||||
|     /// Retrieves the underlying mouse motion handler. | ||||
|     [[nodiscard]] const MouseMotionFactory* GetMouseMotions() const; | ||||
|  | ||||
|     /// Retrieves the underlying mouse touch handler. | ||||
|     [[nodiscard]] MouseTouchFactory* GetMouseTouch(); | ||||
|  | ||||
|     /// Retrieves the underlying mouse touch handler. | ||||
|     [[nodiscard]] const MouseTouchFactory* GetMouseTouch() const; | ||||
|  | ||||
|     /// Retrieves the underlying tas button handler. | ||||
|     [[nodiscard]] TasButtonFactory* GetTasButtons(); | ||||
|  | ||||
|     /// Retrieves the underlying tas button handler. | ||||
|     [[nodiscard]] const TasButtonFactory* GetTasButtons() const; | ||||
|  | ||||
|     /// Retrieves the underlying tas analogs handler. | ||||
|     [[nodiscard]] TasAnalogFactory* GetTasAnalogs(); | ||||
|  | ||||
|     /// Retrieves the underlying tas analogs handler. | ||||
|     [[nodiscard]] const TasAnalogFactory* GetTasAnalogs() const; | ||||
|  | ||||
|     /// Reloads the input devices | ||||
|     void ReloadInputDevices(); | ||||
|  | ||||
|     /// Start polling from all backends for a desired input type. | ||||
|     void BeginMapping(Polling::InputType type); | ||||
|  | ||||
|     /// Returns an input event with mapping information. | ||||
|     [[nodiscard]] const Common::ParamPackage GetNextInput() const; | ||||
|  | ||||
|     /// Stop polling from all backends. | ||||
|     void StopMapping() const; | ||||
|     /// Get all DevicePoller from all backends for a specific device type | ||||
|     [[nodiscard]] std::vector<std::unique_ptr<Polling::DevicePoller>> GetPollers( | ||||
|         Polling::DeviceType type) const; | ||||
|  | ||||
| private: | ||||
|     struct Impl; | ||||
|     std::unique_ptr<Impl> impl; | ||||
| }; | ||||
|  | ||||
| /// Generates a serialized param package for creating a keyboard button device. | ||||
| /// Generates a serialized param package for creating a keyboard button device | ||||
| std::string GenerateKeyboardParam(int key_code); | ||||
|  | ||||
| /// Generates a serialized param package for creating a moddifier keyboard button device. | ||||
| std::string GenerateModdifierKeyboardParam(int key_code); | ||||
|  | ||||
| /// Generates a serialized param package for creating an analog device taking input from keyboard. | ||||
| /// Generates a serialized param package for creating an analog device taking input from keyboard | ||||
| std::string GenerateAnalogParamFromKeys(int key_up, int key_down, int key_left, int key_right, | ||||
|                                         int key_modifier, float modifier_scale); | ||||
|  | ||||
| } // namespace InputCommon | ||||
|   | ||||
| @@ -6,12 +6,8 @@ | ||||
| #include <thread> | ||||
|  | ||||
| #include "common/assert.h" | ||||
| #include "common/param_package.h" | ||||
| #include "common/string_util.h" | ||||
| #include "core/core.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hle/lock.h" | ||||
| #include "core/hle/service/hid/controllers/npad.h" | ||||
| #include "core/hle/service/hid/hid.h" | ||||
| @@ -27,32 +23,49 @@ | ||||
|  | ||||
| namespace { | ||||
|  | ||||
| void UpdateController(Core::HID::EmulatedController* controller, | ||||
|                       Core::HID::NpadStyleIndex controller_type, bool connected) { | ||||
|     if (controller->IsConnected()) { | ||||
|         controller->Disconnect(); | ||||
|     } | ||||
|     controller->SetNpadStyleIndex(controller_type); | ||||
|     if (connected) { | ||||
|         controller->Connect(); | ||||
| constexpr std::size_t HANDHELD_INDEX = 8; | ||||
|  | ||||
| constexpr std::array<std::array<bool, 4>, 8> led_patterns{{ | ||||
|     {true, false, false, false}, | ||||
|     {true, true, false, false}, | ||||
|     {true, true, true, false}, | ||||
|     {true, true, true, true}, | ||||
|     {true, false, false, true}, | ||||
|     {true, false, true, false}, | ||||
|     {true, false, true, true}, | ||||
|     {false, true, true, false}, | ||||
| }}; | ||||
|  | ||||
| void UpdateController(Settings::ControllerType controller_type, std::size_t npad_index, | ||||
|                       bool connected, Core::System& system) { | ||||
|     if (!system.IsPoweredOn()) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     auto& npad = | ||||
|         system.ServiceManager() | ||||
|             .GetService<Service::HID::Hid>("hid") | ||||
|             ->GetAppletResource() | ||||
|             ->GetController<Service::HID::Controller_NPad>(Service::HID::HidController::NPad); | ||||
|  | ||||
|     npad.UpdateControllerAt(npad.MapSettingsTypeToNPad(controller_type), npad_index, connected); | ||||
| } | ||||
|  | ||||
| // Returns true if the given controller type is compatible with the given parameters. | ||||
| bool IsControllerCompatible(Core::HID::NpadStyleIndex controller_type, | ||||
| bool IsControllerCompatible(Settings::ControllerType controller_type, | ||||
|                             Core::Frontend::ControllerParameters parameters) { | ||||
|     switch (controller_type) { | ||||
|     case Core::HID::NpadStyleIndex::ProController: | ||||
|     case Settings::ControllerType::ProController: | ||||
|         return parameters.allow_pro_controller; | ||||
|     case Core::HID::NpadStyleIndex::JoyconDual: | ||||
|     case Settings::ControllerType::DualJoyconDetached: | ||||
|         return parameters.allow_dual_joycons; | ||||
|     case Core::HID::NpadStyleIndex::JoyconLeft: | ||||
|     case Settings::ControllerType::LeftJoycon: | ||||
|         return parameters.allow_left_joycon; | ||||
|     case Core::HID::NpadStyleIndex::JoyconRight: | ||||
|     case Settings::ControllerType::RightJoycon: | ||||
|         return parameters.allow_right_joycon; | ||||
|     case Core::HID::NpadStyleIndex::Handheld: | ||||
|     case Settings::ControllerType::Handheld: | ||||
|         return parameters.enable_single_mode && parameters.allow_handheld; | ||||
|     case Core::HID::NpadStyleIndex::GameCube: | ||||
|     case Settings::ControllerType::GameCube: | ||||
|         return parameters.allow_gamecube_controller; | ||||
|     default: | ||||
|         return false; | ||||
| @@ -183,7 +196,7 @@ QtControllerSelectorDialog::QtControllerSelectorDialog( | ||||
|             connect(emulated_controllers[i], qOverload<int>(&QComboBox::currentIndexChanged), | ||||
|                     [this, i](int index) { | ||||
|                         UpdateDockedState(GetControllerTypeFromIndex(index, i) == | ||||
|                                           Core::HID::NpadStyleIndex::Handheld); | ||||
|                                           Settings::ControllerType::Handheld); | ||||
|                     }); | ||||
|         } | ||||
|     } | ||||
| @@ -236,17 +249,17 @@ void QtControllerSelectorDialog::ApplyConfiguration() { | ||||
| } | ||||
|  | ||||
| void QtControllerSelectorDialog::LoadConfiguration() { | ||||
|     const auto* handheld = system.HIDCore().GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|     for (std::size_t index = 0; index < NUM_PLAYERS; ++index) { | ||||
|         const auto* controller = system.HIDCore().GetEmulatedControllerByIndex(index); | ||||
|         const auto connected = controller->IsConnected() || (index == 0 && handheld->IsConnected()); | ||||
|         const auto connected = | ||||
|             Settings::values.players.GetValue()[index].connected || | ||||
|             (index == 0 && Settings::values.players.GetValue()[HANDHELD_INDEX].connected); | ||||
|         player_groupboxes[index]->setChecked(connected); | ||||
|         connected_controller_checkboxes[index]->setChecked(connected); | ||||
|         emulated_controllers[index]->setCurrentIndex( | ||||
|             GetIndexFromControllerType(controller->GetNpadStyleIndex(), index)); | ||||
|         emulated_controllers[index]->setCurrentIndex(GetIndexFromControllerType( | ||||
|             Settings::values.players.GetValue()[index].controller_type, index)); | ||||
|     } | ||||
|  | ||||
|     UpdateDockedState(handheld->IsConnected()); | ||||
|     UpdateDockedState(Settings::values.players.GetValue()[HANDHELD_INDEX].connected); | ||||
|  | ||||
|     ui->vibrationGroup->setChecked(Settings::values.vibration_enabled.GetValue()); | ||||
|     ui->motionGroup->setChecked(Settings::values.motion_enabled.GetValue()); | ||||
| @@ -402,33 +415,33 @@ void QtControllerSelectorDialog::SetEmulatedControllers(std::size_t player_index | ||||
|     emulated_controllers[player_index]->clear(); | ||||
|  | ||||
|     pairs.emplace_back(emulated_controllers[player_index]->count(), | ||||
|                        Core::HID::NpadStyleIndex::ProController); | ||||
|                        Settings::ControllerType::ProController); | ||||
|     emulated_controllers[player_index]->addItem(tr("Pro Controller")); | ||||
|  | ||||
|     pairs.emplace_back(emulated_controllers[player_index]->count(), | ||||
|                        Core::HID::NpadStyleIndex::JoyconDual); | ||||
|                        Settings::ControllerType::DualJoyconDetached); | ||||
|     emulated_controllers[player_index]->addItem(tr("Dual Joycons")); | ||||
|  | ||||
|     pairs.emplace_back(emulated_controllers[player_index]->count(), | ||||
|                        Core::HID::NpadStyleIndex::JoyconLeft); | ||||
|                        Settings::ControllerType::LeftJoycon); | ||||
|     emulated_controllers[player_index]->addItem(tr("Left Joycon")); | ||||
|  | ||||
|     pairs.emplace_back(emulated_controllers[player_index]->count(), | ||||
|                        Core::HID::NpadStyleIndex::JoyconRight); | ||||
|                        Settings::ControllerType::RightJoycon); | ||||
|     emulated_controllers[player_index]->addItem(tr("Right Joycon")); | ||||
|  | ||||
|     if (player_index == 0) { | ||||
|         pairs.emplace_back(emulated_controllers[player_index]->count(), | ||||
|                            Core::HID::NpadStyleIndex::Handheld); | ||||
|                            Settings::ControllerType::Handheld); | ||||
|         emulated_controllers[player_index]->addItem(tr("Handheld")); | ||||
|     } | ||||
|  | ||||
|     pairs.emplace_back(emulated_controllers[player_index]->count(), | ||||
|                        Core::HID::NpadStyleIndex::GameCube); | ||||
|                        Settings::ControllerType::GameCube); | ||||
|     emulated_controllers[player_index]->addItem(tr("GameCube Controller")); | ||||
| } | ||||
|  | ||||
| Core::HID::NpadStyleIndex QtControllerSelectorDialog::GetControllerTypeFromIndex( | ||||
| Settings::ControllerType QtControllerSelectorDialog::GetControllerTypeFromIndex( | ||||
|     int index, std::size_t player_index) const { | ||||
|     const auto& pairs = index_controller_type_pairs[player_index]; | ||||
|  | ||||
| @@ -436,13 +449,13 @@ Core::HID::NpadStyleIndex QtControllerSelectorDialog::GetControllerTypeFromIndex | ||||
|                                  [index](const auto& pair) { return pair.first == index; }); | ||||
|  | ||||
|     if (it == pairs.end()) { | ||||
|         return Core::HID::NpadStyleIndex::ProController; | ||||
|         return Settings::ControllerType::ProController; | ||||
|     } | ||||
|  | ||||
|     return it->second; | ||||
| } | ||||
|  | ||||
| int QtControllerSelectorDialog::GetIndexFromControllerType(Core::HID::NpadStyleIndex type, | ||||
| int QtControllerSelectorDialog::GetIndexFromControllerType(Settings::ControllerType type, | ||||
|                                                            std::size_t player_index) const { | ||||
|     const auto& pairs = index_controller_type_pairs[player_index]; | ||||
|  | ||||
| @@ -466,16 +479,16 @@ void QtControllerSelectorDialog::UpdateControllerIcon(std::size_t player_index) | ||||
|     const QString stylesheet = [this, player_index] { | ||||
|         switch (GetControllerTypeFromIndex(emulated_controllers[player_index]->currentIndex(), | ||||
|                                            player_index)) { | ||||
|         case Core::HID::NpadStyleIndex::ProController: | ||||
|         case Core::HID::NpadStyleIndex::GameCube: | ||||
|         case Settings::ControllerType::ProController: | ||||
|         case Settings::ControllerType::GameCube: | ||||
|             return QStringLiteral("image: url(:/controller/applet_pro_controller%0); "); | ||||
|         case Core::HID::NpadStyleIndex::JoyconDual: | ||||
|         case Settings::ControllerType::DualJoyconDetached: | ||||
|             return QStringLiteral("image: url(:/controller/applet_dual_joycon%0); "); | ||||
|         case Core::HID::NpadStyleIndex::JoyconLeft: | ||||
|         case Settings::ControllerType::LeftJoycon: | ||||
|             return QStringLiteral("image: url(:/controller/applet_joycon_left%0); "); | ||||
|         case Core::HID::NpadStyleIndex::JoyconRight: | ||||
|         case Settings::ControllerType::RightJoycon: | ||||
|             return QStringLiteral("image: url(:/controller/applet_joycon_right%0); "); | ||||
|         case Core::HID::NpadStyleIndex::Handheld: | ||||
|         case Settings::ControllerType::Handheld: | ||||
|             return QStringLiteral("image: url(:/controller/applet_handheld%0); "); | ||||
|         default: | ||||
|             return QString{}; | ||||
| @@ -503,42 +516,54 @@ void QtControllerSelectorDialog::UpdateControllerIcon(std::size_t player_index) | ||||
| } | ||||
|  | ||||
| void QtControllerSelectorDialog::UpdateControllerState(std::size_t player_index) { | ||||
|     auto* controller = system.HIDCore().GetEmulatedControllerByIndex(player_index); | ||||
|     auto& player = Settings::values.players.GetValue()[player_index]; | ||||
|  | ||||
|     const auto controller_type = GetControllerTypeFromIndex( | ||||
|         emulated_controllers[player_index]->currentIndex(), player_index); | ||||
|     const auto player_connected = player_groupboxes[player_index]->isChecked() && | ||||
|                                   controller_type != Core::HID::NpadStyleIndex::Handheld; | ||||
|                                   controller_type != Settings::ControllerType::Handheld; | ||||
|  | ||||
|     if (controller->GetNpadStyleIndex() == controller_type && | ||||
|         controller->IsConnected() == player_connected) { | ||||
|     if (player.controller_type == controller_type && player.connected == player_connected) { | ||||
|         // Set vibration devices in the event that the input device has changed. | ||||
|         ConfigureVibration::SetVibrationDevices(player_index); | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // Disconnect the controller first. | ||||
|     UpdateController(controller, controller_type, false); | ||||
|     UpdateController(controller_type, player_index, false, system); | ||||
|  | ||||
|     player.controller_type = controller_type; | ||||
|     player.connected = player_connected; | ||||
|  | ||||
|     ConfigureVibration::SetVibrationDevices(player_index); | ||||
|  | ||||
|     // Handheld | ||||
|     if (player_index == 0) { | ||||
|         if (controller_type == Core::HID::NpadStyleIndex::Handheld) { | ||||
|             auto* handheld = | ||||
|                 system.HIDCore().GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|             UpdateController(handheld, Core::HID::NpadStyleIndex::Handheld, | ||||
|                              player_groupboxes[player_index]->isChecked()); | ||||
|         auto& handheld = Settings::values.players.GetValue()[HANDHELD_INDEX]; | ||||
|         if (controller_type == Settings::ControllerType::Handheld) { | ||||
|             handheld = player; | ||||
|         } | ||||
|         handheld.connected = player_groupboxes[player_index]->isChecked() && | ||||
|                              controller_type == Settings::ControllerType::Handheld; | ||||
|         UpdateController(Settings::ControllerType::Handheld, 8, handheld.connected, system); | ||||
|     } | ||||
|  | ||||
|     UpdateController(controller, controller_type, player_connected); | ||||
|     if (!player.connected) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     // This emulates a delay between disconnecting and reconnecting controllers as some games | ||||
|     // do not respond to a change in controller type if it was instantaneous. | ||||
|     using namespace std::chrono_literals; | ||||
|     std::this_thread::sleep_for(60ms); | ||||
|  | ||||
|     UpdateController(controller_type, player_index, player_connected, system); | ||||
| } | ||||
|  | ||||
| void QtControllerSelectorDialog::UpdateLEDPattern(std::size_t player_index) { | ||||
|     if (!player_groupboxes[player_index]->isChecked() || | ||||
|         GetControllerTypeFromIndex(emulated_controllers[player_index]->currentIndex(), | ||||
|                                    player_index) == Core::HID::NpadStyleIndex::Handheld) { | ||||
|                                    player_index) == Settings::ControllerType::Handheld) { | ||||
|         led_patterns_boxes[player_index][0]->setChecked(false); | ||||
|         led_patterns_boxes[player_index][1]->setChecked(false); | ||||
|         led_patterns_boxes[player_index][2]->setChecked(false); | ||||
| @@ -546,12 +571,10 @@ void QtControllerSelectorDialog::UpdateLEDPattern(std::size_t player_index) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const auto* controller = system.HIDCore().GetEmulatedControllerByIndex(player_index); | ||||
|     const auto led_pattern = controller->GetLedPattern(); | ||||
|     led_patterns_boxes[player_index][0]->setChecked(led_pattern.position1); | ||||
|     led_patterns_boxes[player_index][1]->setChecked(led_pattern.position2); | ||||
|     led_patterns_boxes[player_index][2]->setChecked(led_pattern.position3); | ||||
|     led_patterns_boxes[player_index][3]->setChecked(led_pattern.position4); | ||||
|     led_patterns_boxes[player_index][0]->setChecked(led_patterns[player_index][0]); | ||||
|     led_patterns_boxes[player_index][1]->setChecked(led_patterns[player_index][1]); | ||||
|     led_patterns_boxes[player_index][2]->setChecked(led_patterns[player_index][2]); | ||||
|     led_patterns_boxes[player_index][3]->setChecked(led_patterns[player_index][3]); | ||||
| } | ||||
|  | ||||
| void QtControllerSelectorDialog::UpdateBorderColor(std::size_t player_index) { | ||||
| @@ -631,9 +654,10 @@ void QtControllerSelectorDialog::DisableUnsupportedPlayers() { | ||||
|     } | ||||
|  | ||||
|     for (std::size_t index = max_supported_players; index < NUM_PLAYERS; ++index) { | ||||
|         auto* controller = system.HIDCore().GetEmulatedControllerByIndex(index); | ||||
|         // Disconnect any unsupported players here and disable or hide them if applicable. | ||||
|         UpdateController(controller, controller->GetNpadStyleIndex(), false); | ||||
|         Settings::values.players.GetValue()[index].connected = false; | ||||
|         UpdateController(Settings::values.players.GetValue()[index].controller_type, index, false, | ||||
|                          system); | ||||
|         // Hide the player widgets when max_supported_controllers is less than or equal to 4. | ||||
|         if (max_supported_players <= 4) { | ||||
|             player_widgets[index]->hide(); | ||||
|   | ||||
| @@ -23,18 +23,14 @@ namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
|  | ||||
| namespace Settings { | ||||
| enum class ControllerType; | ||||
| } | ||||
|  | ||||
| namespace Ui { | ||||
| class QtControllerSelectorDialog; | ||||
| } | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| enum class NpadStyleIndex : u8; | ||||
| } | ||||
|  | ||||
| class QtControllerSelectorDialog final : public QDialog { | ||||
|     Q_OBJECT | ||||
|  | ||||
| @@ -74,10 +70,10 @@ private: | ||||
|     void SetEmulatedControllers(std::size_t player_index); | ||||
|  | ||||
|     // Gets the Controller Type for a given controller combobox index per player. | ||||
|     Core::HID::NpadStyleIndex GetControllerTypeFromIndex(int index, std::size_t player_index) const; | ||||
|     Settings::ControllerType GetControllerTypeFromIndex(int index, std::size_t player_index) const; | ||||
|  | ||||
|     // Gets the controller combobox index for a given Controller Type per player. | ||||
|     int GetIndexFromControllerType(Core::HID::NpadStyleIndex type, std::size_t player_index) const; | ||||
|     int GetIndexFromControllerType(Settings::ControllerType type, std::size_t player_index) const; | ||||
|  | ||||
|     // Updates the controller icons per player. | ||||
|     void UpdateControllerIcon(std::size_t player_index); | ||||
| @@ -139,7 +135,7 @@ private: | ||||
|     std::array<QComboBox*, NUM_PLAYERS> emulated_controllers; | ||||
|  | ||||
|     /// Pairs of emulated controller index and Controller Type enum per player. | ||||
|     std::array<std::vector<std::pair<int, Core::HID::NpadStyleIndex>>, NUM_PLAYERS> | ||||
|     std::array<std::vector<std::pair<int, Settings::ControllerType>>, NUM_PLAYERS> | ||||
|         index_controller_type_pairs; | ||||
|  | ||||
|     // Labels representing the number of connected controllers | ||||
|   | ||||
| @@ -10,10 +10,7 @@ | ||||
| #include "common/settings.h" | ||||
| #include "common/string_util.h" | ||||
| #include "core/core.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hid/input_interpreter.h" | ||||
| #include "core/frontend/input_interpreter.h" | ||||
| #include "ui_qt_software_keyboard.h" | ||||
| #include "yuzu/applets/qt_software_keyboard.h" | ||||
| #include "yuzu/main.h" | ||||
| @@ -487,7 +484,7 @@ void QtSoftwareKeyboardDialog::open() { | ||||
| void QtSoftwareKeyboardDialog::reject() { | ||||
|     // Pressing the ESC key in a dialog calls QDialog::reject(). | ||||
|     // We will override this behavior to the "Cancel" action on the software keyboard. | ||||
|     TranslateButtonPress(Core::HID::NpadButton::X); | ||||
|     TranslateButtonPress(HIDButton::X); | ||||
| } | ||||
|  | ||||
| void QtSoftwareKeyboardDialog::keyPressEvent(QKeyEvent* event) { | ||||
| @@ -725,7 +722,7 @@ void QtSoftwareKeyboardDialog::SetTextDrawType() { | ||||
|  | ||||
|         connect( | ||||
|             ui->line_edit_osk, &QLineEdit::returnPressed, this, | ||||
|             [this] { TranslateButtonPress(Core::HID::NpadButton::Plus); }, Qt::QueuedConnection); | ||||
|             [this] { TranslateButtonPress(HIDButton::Plus); }, Qt::QueuedConnection); | ||||
|  | ||||
|         ui->line_edit_osk->setPlaceholderText( | ||||
|             QString::fromStdU16String(initialize_parameters.guide_text)); | ||||
| @@ -798,10 +795,9 @@ void QtSoftwareKeyboardDialog::SetTextDrawType() { | ||||
| } | ||||
|  | ||||
| void QtSoftwareKeyboardDialog::SetControllerImage() { | ||||
|     const auto* handheld = system.HIDCore().GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|     const auto* player_1 = system.HIDCore().GetEmulatedController(Core::HID::NpadIdType::Player1); | ||||
|     const auto controller_type = | ||||
|         handheld->IsConnected() ? handheld->GetNpadStyleIndex() : player_1->GetNpadStyleIndex(); | ||||
|     const auto controller_type = Settings::values.players.GetValue()[8].connected | ||||
|                                      ? Settings::values.players.GetValue()[8].controller_type | ||||
|                                      : Settings::values.players.GetValue()[0].controller_type; | ||||
|  | ||||
|     const QString theme = [] { | ||||
|         if (QIcon::themeName().contains(QStringLiteral("dark")) || | ||||
| @@ -813,8 +809,8 @@ void QtSoftwareKeyboardDialog::SetControllerImage() { | ||||
|     }(); | ||||
|  | ||||
|     switch (controller_type) { | ||||
|     case Core::HID::NpadStyleIndex::ProController: | ||||
|     case Core::HID::NpadStyleIndex::GameCube: | ||||
|     case Settings::ControllerType::ProController: | ||||
|     case Settings::ControllerType::GameCube: | ||||
|         ui->icon_controller->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_pro%1.png);").arg(theme)); | ||||
|         ui->icon_controller_shift->setStyleSheet( | ||||
| @@ -822,7 +818,7 @@ void QtSoftwareKeyboardDialog::SetControllerImage() { | ||||
|         ui->icon_controller_num->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_pro%1.png);").arg(theme)); | ||||
|         break; | ||||
|     case Core::HID::NpadStyleIndex::JoyconDual: | ||||
|     case Settings::ControllerType::DualJoyconDetached: | ||||
|         ui->icon_controller->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_dual_joycon%1.png);").arg(theme)); | ||||
|         ui->icon_controller_shift->setStyleSheet( | ||||
| @@ -830,7 +826,7 @@ void QtSoftwareKeyboardDialog::SetControllerImage() { | ||||
|         ui->icon_controller_num->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_dual_joycon%1.png);").arg(theme)); | ||||
|         break; | ||||
|     case Core::HID::NpadStyleIndex::JoyconLeft: | ||||
|     case Settings::ControllerType::LeftJoycon: | ||||
|         ui->icon_controller->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_single_joycon_left%1.png);") | ||||
|                 .arg(theme)); | ||||
| @@ -841,7 +837,7 @@ void QtSoftwareKeyboardDialog::SetControllerImage() { | ||||
|             QStringLiteral("image: url(:/overlay/controller_single_joycon_left%1.png);") | ||||
|                 .arg(theme)); | ||||
|         break; | ||||
|     case Core::HID::NpadStyleIndex::JoyconRight: | ||||
|     case Settings::ControllerType::RightJoycon: | ||||
|         ui->icon_controller->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_single_joycon_right%1.png);") | ||||
|                 .arg(theme)); | ||||
| @@ -852,7 +848,7 @@ void QtSoftwareKeyboardDialog::SetControllerImage() { | ||||
|             QStringLiteral("image: url(:/overlay/controller_single_joycon_right%1.png);") | ||||
|                 .arg(theme)); | ||||
|         break; | ||||
|     case Core::HID::NpadStyleIndex::Handheld: | ||||
|     case Settings::ControllerType::Handheld: | ||||
|         ui->icon_controller->setStyleSheet( | ||||
|             QStringLiteral("image: url(:/overlay/controller_handheld%1.png);").arg(theme)); | ||||
|         ui->icon_controller_shift->setStyleSheet( | ||||
| @@ -1212,9 +1208,9 @@ void QtSoftwareKeyboardDialog::SetupMouseHover() { | ||||
|     } | ||||
| } | ||||
|  | ||||
| template <Core::HID::NpadButton... T> | ||||
| template <HIDButton... T> | ||||
| void QtSoftwareKeyboardDialog::HandleButtonPressedOnce() { | ||||
|     const auto f = [this](Core::HID::NpadButton button) { | ||||
|     const auto f = [this](HIDButton button) { | ||||
|         if (input_interpreter->IsButtonPressedOnce(button)) { | ||||
|             TranslateButtonPress(button); | ||||
|         } | ||||
| @@ -1223,9 +1219,9 @@ void QtSoftwareKeyboardDialog::HandleButtonPressedOnce() { | ||||
|     (f(T), ...); | ||||
| } | ||||
|  | ||||
| template <Core::HID::NpadButton... T> | ||||
| template <HIDButton... T> | ||||
| void QtSoftwareKeyboardDialog::HandleButtonHold() { | ||||
|     const auto f = [this](Core::HID::NpadButton button) { | ||||
|     const auto f = [this](HIDButton button) { | ||||
|         if (input_interpreter->IsButtonHeld(button)) { | ||||
|             TranslateButtonPress(button); | ||||
|         } | ||||
| @@ -1234,9 +1230,9 @@ void QtSoftwareKeyboardDialog::HandleButtonHold() { | ||||
|     (f(T), ...); | ||||
| } | ||||
|  | ||||
| void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button) { | ||||
| void QtSoftwareKeyboardDialog::TranslateButtonPress(HIDButton button) { | ||||
|     switch (button) { | ||||
|     case Core::HID::NpadButton::A: | ||||
|     case HIDButton::A: | ||||
|         switch (bottom_osk_index) { | ||||
|         case BottomOSKIndex::LowerCase: | ||||
|         case BottomOSKIndex::UpperCase: | ||||
| @@ -1249,7 +1245,7 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button | ||||
|             break; | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::B: | ||||
|     case HIDButton::B: | ||||
|         switch (bottom_osk_index) { | ||||
|         case BottomOSKIndex::LowerCase: | ||||
|             ui->button_backspace->click(); | ||||
| @@ -1264,7 +1260,7 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button | ||||
|             break; | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::X: | ||||
|     case HIDButton::X: | ||||
|         if (is_inline) { | ||||
|             emit SubmitInlineText(SwkbdReplyType::DecidedCancel, current_text, cursor_position); | ||||
|         } else { | ||||
| @@ -1275,7 +1271,7 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button | ||||
|             emit SubmitNormalText(SwkbdResult::Cancel, std::move(text)); | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Y: | ||||
|     case HIDButton::Y: | ||||
|         switch (bottom_osk_index) { | ||||
|         case BottomOSKIndex::LowerCase: | ||||
|             ui->button_space->click(); | ||||
| @@ -1288,8 +1284,8 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button | ||||
|             break; | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::StickL: | ||||
|     case Core::HID::NpadButton::StickR: | ||||
|     case HIDButton::LStick: | ||||
|     case HIDButton::RStick: | ||||
|         switch (bottom_osk_index) { | ||||
|         case BottomOSKIndex::LowerCase: | ||||
|             ui->button_shift->click(); | ||||
| @@ -1302,13 +1298,13 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button | ||||
|             break; | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::L: | ||||
|     case HIDButton::L: | ||||
|         MoveTextCursorDirection(Direction::Left); | ||||
|         break; | ||||
|     case Core::HID::NpadButton::R: | ||||
|     case HIDButton::R: | ||||
|         MoveTextCursorDirection(Direction::Right); | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Plus: | ||||
|     case HIDButton::Plus: | ||||
|         switch (bottom_osk_index) { | ||||
|         case BottomOSKIndex::LowerCase: | ||||
|             ui->button_ok->click(); | ||||
| @@ -1323,24 +1319,24 @@ void QtSoftwareKeyboardDialog::TranslateButtonPress(Core::HID::NpadButton button | ||||
|             break; | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Left: | ||||
|     case Core::HID::NpadButton::StickLLeft: | ||||
|     case Core::HID::NpadButton::StickRLeft: | ||||
|     case HIDButton::DLeft: | ||||
|     case HIDButton::LStickLeft: | ||||
|     case HIDButton::RStickLeft: | ||||
|         MoveButtonDirection(Direction::Left); | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Up: | ||||
|     case Core::HID::NpadButton::StickLUp: | ||||
|     case Core::HID::NpadButton::StickRUp: | ||||
|     case HIDButton::DUp: | ||||
|     case HIDButton::LStickUp: | ||||
|     case HIDButton::RStickUp: | ||||
|         MoveButtonDirection(Direction::Up); | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Right: | ||||
|     case Core::HID::NpadButton::StickLRight: | ||||
|     case Core::HID::NpadButton::StickRRight: | ||||
|     case HIDButton::DRight: | ||||
|     case HIDButton::LStickRight: | ||||
|     case HIDButton::RStickRight: | ||||
|         MoveButtonDirection(Direction::Right); | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Down: | ||||
|     case Core::HID::NpadButton::StickLDown: | ||||
|     case Core::HID::NpadButton::StickRDown: | ||||
|     case HIDButton::DDown: | ||||
|     case HIDButton::LStickDown: | ||||
|     case HIDButton::RStickDown: | ||||
|         MoveButtonDirection(Direction::Down); | ||||
|         break; | ||||
|     default: | ||||
| @@ -1471,25 +1467,19 @@ void QtSoftwareKeyboardDialog::InputThread() { | ||||
|     while (input_thread_running) { | ||||
|         input_interpreter->PollInput(); | ||||
|  | ||||
|         HandleButtonPressedOnce< | ||||
|             Core::HID::NpadButton::A, Core::HID::NpadButton::B, Core::HID::NpadButton::X, | ||||
|             Core::HID::NpadButton::Y, Core::HID::NpadButton::StickL, Core::HID::NpadButton::StickR, | ||||
|             Core::HID::NpadButton::L, Core::HID::NpadButton::R, Core::HID::NpadButton::Plus, | ||||
|             Core::HID::NpadButton::Left, Core::HID::NpadButton::Up, Core::HID::NpadButton::Right, | ||||
|             Core::HID::NpadButton::Down, Core::HID::NpadButton::StickLLeft, | ||||
|             Core::HID::NpadButton::StickLUp, Core::HID::NpadButton::StickLRight, | ||||
|             Core::HID::NpadButton::StickLDown, Core::HID::NpadButton::StickRLeft, | ||||
|             Core::HID::NpadButton::StickRUp, Core::HID::NpadButton::StickRRight, | ||||
|             Core::HID::NpadButton::StickRDown>(); | ||||
|         HandleButtonPressedOnce<HIDButton::A, HIDButton::B, HIDButton::X, HIDButton::Y, | ||||
|                                 HIDButton::LStick, HIDButton::RStick, HIDButton::L, HIDButton::R, | ||||
|                                 HIDButton::Plus, HIDButton::DLeft, HIDButton::DUp, | ||||
|                                 HIDButton::DRight, HIDButton::DDown, HIDButton::LStickLeft, | ||||
|                                 HIDButton::LStickUp, HIDButton::LStickRight, HIDButton::LStickDown, | ||||
|                                 HIDButton::RStickLeft, HIDButton::RStickUp, HIDButton::RStickRight, | ||||
|                                 HIDButton::RStickDown>(); | ||||
|  | ||||
|         HandleButtonHold<Core::HID::NpadButton::B, Core::HID::NpadButton::L, | ||||
|                          Core::HID::NpadButton::R, Core::HID::NpadButton::Left, | ||||
|                          Core::HID::NpadButton::Up, Core::HID::NpadButton::Right, | ||||
|                          Core::HID::NpadButton::Down, Core::HID::NpadButton::StickLLeft, | ||||
|                          Core::HID::NpadButton::StickLUp, Core::HID::NpadButton::StickLRight, | ||||
|                          Core::HID::NpadButton::StickLDown, Core::HID::NpadButton::StickRLeft, | ||||
|                          Core::HID::NpadButton::StickRUp, Core::HID::NpadButton::StickRRight, | ||||
|                          Core::HID::NpadButton::StickRDown>(); | ||||
|         HandleButtonHold<HIDButton::B, HIDButton::L, HIDButton::R, HIDButton::DLeft, HIDButton::DUp, | ||||
|                          HIDButton::DRight, HIDButton::DDown, HIDButton::LStickLeft, | ||||
|                          HIDButton::LStickUp, HIDButton::LStickRight, HIDButton::LStickDown, | ||||
|                          HIDButton::RStickLeft, HIDButton::RStickUp, HIDButton::RStickRight, | ||||
|                          HIDButton::RStickDown>(); | ||||
|  | ||||
|         std::this_thread::sleep_for(std::chrono::milliseconds(50)); | ||||
|     } | ||||
|   | ||||
| @@ -14,16 +14,14 @@ | ||||
|  | ||||
| #include "core/frontend/applets/software_keyboard.h" | ||||
|  | ||||
| enum class HIDButton : u8; | ||||
|  | ||||
| class InputInterpreter; | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| enum class NpadButton : u64; | ||||
| } | ||||
|  | ||||
| namespace Ui { | ||||
| class QtSoftwareKeyboardDialog; | ||||
| } | ||||
| @@ -148,7 +146,7 @@ private: | ||||
|      * | ||||
|      * @tparam HIDButton The list of buttons that can be converted into keyboard input. | ||||
|      */ | ||||
|     template <Core::HID::NpadButton... T> | ||||
|     template <HIDButton... T> | ||||
|     void HandleButtonPressedOnce(); | ||||
|  | ||||
|     /** | ||||
| @@ -156,7 +154,7 @@ private: | ||||
|      * | ||||
|      * @tparam HIDButton The list of buttons that can be converted into keyboard input. | ||||
|      */ | ||||
|     template <Core::HID::NpadButton... T> | ||||
|     template <HIDButton... T> | ||||
|     void HandleButtonHold(); | ||||
|  | ||||
|     /** | ||||
| @@ -164,7 +162,7 @@ private: | ||||
|      * | ||||
|      * @param button The button press to process. | ||||
|      */ | ||||
|     void TranslateButtonPress(Core::HID::NpadButton button); | ||||
|     void TranslateButtonPress(HIDButton button); | ||||
|  | ||||
|     /** | ||||
|      * Moves the focus of a button in a certain direction. | ||||
|   | ||||
| @@ -14,11 +14,9 @@ | ||||
| #endif | ||||
|  | ||||
| #include "common/fs/path_util.h" | ||||
| #include "common/param_package.h" | ||||
| #include "core/core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hid/input_interpreter.h" | ||||
| #include "input_common/drivers/keyboard.h" | ||||
| #include "core/frontend/input_interpreter.h" | ||||
| #include "input_common/keyboard.h" | ||||
| #include "input_common/main.h" | ||||
| #include "yuzu/applets/qt_web_browser.h" | ||||
| #include "yuzu/applets/qt_web_browser_scripts.h" | ||||
| @@ -29,19 +27,19 @@ | ||||
|  | ||||
| namespace { | ||||
|  | ||||
| constexpr int HIDButtonToKey(Core::HID::NpadButton button) { | ||||
| constexpr int HIDButtonToKey(HIDButton button) { | ||||
|     switch (button) { | ||||
|     case Core::HID::NpadButton::Left: | ||||
|     case Core::HID::NpadButton::StickLLeft: | ||||
|     case HIDButton::DLeft: | ||||
|     case HIDButton::LStickLeft: | ||||
|         return Qt::Key_Left; | ||||
|     case Core::HID::NpadButton::Up: | ||||
|     case Core::HID::NpadButton::StickLUp: | ||||
|     case HIDButton::DUp: | ||||
|     case HIDButton::LStickUp: | ||||
|         return Qt::Key_Up; | ||||
|     case Core::HID::NpadButton::Right: | ||||
|     case Core::HID::NpadButton::StickLRight: | ||||
|     case HIDButton::DRight: | ||||
|     case HIDButton::LStickRight: | ||||
|         return Qt::Key_Right; | ||||
|     case Core::HID::NpadButton::Down: | ||||
|     case Core::HID::NpadButton::StickLDown: | ||||
|     case HIDButton::DDown: | ||||
|     case HIDButton::LStickDown: | ||||
|         return Qt::Key_Down; | ||||
|     default: | ||||
|         return 0; | ||||
| @@ -210,25 +208,25 @@ void QtNXWebEngineView::keyReleaseEvent(QKeyEvent* event) { | ||||
|     } | ||||
| } | ||||
|  | ||||
| template <Core::HID::NpadButton... T> | ||||
| template <HIDButton... T> | ||||
| void QtNXWebEngineView::HandleWindowFooterButtonPressedOnce() { | ||||
|     const auto f = [this](Core::HID::NpadButton button) { | ||||
|     const auto f = [this](HIDButton button) { | ||||
|         if (input_interpreter->IsButtonPressedOnce(button)) { | ||||
|             page()->runJavaScript( | ||||
|                 QStringLiteral("yuzu_key_callbacks[%1] == null;").arg(static_cast<u8>(button)), | ||||
|                 [this, button](const QVariant& variant) { | ||||
|                     if (variant.toBool()) { | ||||
|                         switch (button) { | ||||
|                         case Core::HID::NpadButton::A: | ||||
|                         case HIDButton::A: | ||||
|                             SendMultipleKeyPressEvents<Qt::Key_A, Qt::Key_Space, Qt::Key_Return>(); | ||||
|                             break; | ||||
|                         case Core::HID::NpadButton::B: | ||||
|                         case HIDButton::B: | ||||
|                             SendKeyPressEvent(Qt::Key_B); | ||||
|                             break; | ||||
|                         case Core::HID::NpadButton::X: | ||||
|                         case HIDButton::X: | ||||
|                             SendKeyPressEvent(Qt::Key_X); | ||||
|                             break; | ||||
|                         case Core::HID::NpadButton::Y: | ||||
|                         case HIDButton::Y: | ||||
|                             SendKeyPressEvent(Qt::Key_Y); | ||||
|                             break; | ||||
|                         default: | ||||
| @@ -246,9 +244,9 @@ void QtNXWebEngineView::HandleWindowFooterButtonPressedOnce() { | ||||
|     (f(T), ...); | ||||
| } | ||||
|  | ||||
| template <Core::HID::NpadButton... T> | ||||
| template <HIDButton... T> | ||||
| void QtNXWebEngineView::HandleWindowKeyButtonPressedOnce() { | ||||
|     const auto f = [this](Core::HID::NpadButton button) { | ||||
|     const auto f = [this](HIDButton button) { | ||||
|         if (input_interpreter->IsButtonPressedOnce(button)) { | ||||
|             SendKeyPressEvent(HIDButtonToKey(button)); | ||||
|         } | ||||
| @@ -257,9 +255,9 @@ void QtNXWebEngineView::HandleWindowKeyButtonPressedOnce() { | ||||
|     (f(T), ...); | ||||
| } | ||||
|  | ||||
| template <Core::HID::NpadButton... T> | ||||
| template <HIDButton... T> | ||||
| void QtNXWebEngineView::HandleWindowKeyButtonHold() { | ||||
|     const auto f = [this](Core::HID::NpadButton button) { | ||||
|     const auto f = [this](HIDButton button) { | ||||
|         if (input_interpreter->IsButtonHeld(button)) { | ||||
|             SendKeyPressEvent(HIDButtonToKey(button)); | ||||
|         } | ||||
| @@ -310,21 +308,17 @@ void QtNXWebEngineView::InputThread() { | ||||
|     while (input_thread_running) { | ||||
|         input_interpreter->PollInput(); | ||||
|  | ||||
|         HandleWindowFooterButtonPressedOnce<Core::HID::NpadButton::A, Core::HID::NpadButton::B, | ||||
|                                             Core::HID::NpadButton::X, Core::HID::NpadButton::Y, | ||||
|                                             Core::HID::NpadButton::L, Core::HID::NpadButton::R>(); | ||||
|         HandleWindowFooterButtonPressedOnce<HIDButton::A, HIDButton::B, HIDButton::X, HIDButton::Y, | ||||
|                                             HIDButton::L, HIDButton::R>(); | ||||
|  | ||||
|         HandleWindowKeyButtonPressedOnce< | ||||
|             Core::HID::NpadButton::Left, Core::HID::NpadButton::Up, Core::HID::NpadButton::Right, | ||||
|             Core::HID::NpadButton::Down, Core::HID::NpadButton::StickLLeft, | ||||
|             Core::HID::NpadButton::StickLUp, Core::HID::NpadButton::StickLRight, | ||||
|             Core::HID::NpadButton::StickLDown>(); | ||||
|         HandleWindowKeyButtonPressedOnce<HIDButton::DLeft, HIDButton::DUp, HIDButton::DRight, | ||||
|                                          HIDButton::DDown, HIDButton::LStickLeft, | ||||
|                                          HIDButton::LStickUp, HIDButton::LStickRight, | ||||
|                                          HIDButton::LStickDown>(); | ||||
|  | ||||
|         HandleWindowKeyButtonHold< | ||||
|             Core::HID::NpadButton::Left, Core::HID::NpadButton::Up, Core::HID::NpadButton::Right, | ||||
|             Core::HID::NpadButton::Down, Core::HID::NpadButton::StickLLeft, | ||||
|             Core::HID::NpadButton::StickLUp, Core::HID::NpadButton::StickLRight, | ||||
|             Core::HID::NpadButton::StickLDown>(); | ||||
|         HandleWindowKeyButtonHold<HIDButton::DLeft, HIDButton::DUp, HIDButton::DRight, | ||||
|                                   HIDButton::DDown, HIDButton::LStickLeft, HIDButton::LStickUp, | ||||
|                                   HIDButton::LStickRight, HIDButton::LStickDown>(); | ||||
|  | ||||
|         std::this_thread::sleep_for(std::chrono::milliseconds(50)); | ||||
|     } | ||||
|   | ||||
| @@ -16,6 +16,8 @@ | ||||
|  | ||||
| #include "core/frontend/applets/web_browser.h" | ||||
|  | ||||
| enum class HIDButton : u8; | ||||
|  | ||||
| class GMainWindow; | ||||
| class InputInterpreter; | ||||
| class UrlRequestInterceptor; | ||||
| @@ -24,10 +26,6 @@ namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| enum class NpadButton : u64; | ||||
| } | ||||
|  | ||||
| namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
| @@ -116,7 +114,7 @@ private: | ||||
|      * | ||||
|      * @tparam HIDButton The list of buttons contained in yuzu_key_callbacks | ||||
|      */ | ||||
|     template <Core::HID::NpadButton... T> | ||||
|     template <HIDButton... T> | ||||
|     void HandleWindowFooterButtonPressedOnce(); | ||||
|  | ||||
|     /** | ||||
| @@ -125,7 +123,7 @@ private: | ||||
|      * | ||||
|      * @tparam HIDButton The list of buttons that can be converted into keyboard input. | ||||
|      */ | ||||
|     template <Core::HID::NpadButton... T> | ||||
|     template <HIDButton... T> | ||||
|     void HandleWindowKeyButtonPressedOnce(); | ||||
|  | ||||
|     /** | ||||
| @@ -134,7 +132,7 @@ private: | ||||
|      * | ||||
|      * @tparam HIDButton The list of buttons that can be converted into keyboard input. | ||||
|      */ | ||||
|     template <Core::HID::NpadButton... T> | ||||
|     template <HIDButton... T> | ||||
|     void HandleWindowKeyButtonHold(); | ||||
|  | ||||
|     /** | ||||
|   | ||||
| @@ -32,11 +32,10 @@ | ||||
| #include "common/settings.h" | ||||
| #include "core/core.h" | ||||
| #include "core/frontend/framebuffer_layout.h" | ||||
| #include "input_common/drivers/keyboard.h" | ||||
| #include "input_common/drivers/mouse.h" | ||||
| #include "input_common/drivers/tas_input.h" | ||||
| #include "input_common/drivers/touch_screen.h" | ||||
| #include "input_common/keyboard.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/mouse/mouse_input.h" | ||||
| #include "input_common/tas/tas_input.h" | ||||
| #include "video_core/renderer_base.h" | ||||
| #include "video_core/video_core.h" | ||||
| #include "yuzu/bootmanager.h" | ||||
| @@ -297,6 +296,7 @@ GRenderWindow::GRenderWindow(GMainWindow* parent, EmuThread* emu_thread_, | ||||
|     layout->setContentsMargins(0, 0, 0, 0); | ||||
|     setLayout(layout); | ||||
|     input_subsystem->Initialize(); | ||||
|  | ||||
|     this->setMouseTracking(true); | ||||
|  | ||||
|     connect(this, &GRenderWindow::FirstFrameDisplayed, parent, &GMainWindow::OnLoadComplete); | ||||
| @@ -383,306 +383,34 @@ void GRenderWindow::closeEvent(QCloseEvent* event) { | ||||
|     QWidget::closeEvent(event); | ||||
| } | ||||
|  | ||||
| int GRenderWindow::QtKeyToSwitchKey(Qt::Key qt_key) { | ||||
|     switch (qt_key) { | ||||
|     case Qt::Key_A: | ||||
|         return Settings::NativeKeyboard::A; | ||||
|     case Qt::Key_B: | ||||
|         return Settings::NativeKeyboard::B; | ||||
|     case Qt::Key_C: | ||||
|         return Settings::NativeKeyboard::C; | ||||
|     case Qt::Key_D: | ||||
|         return Settings::NativeKeyboard::D; | ||||
|     case Qt::Key_E: | ||||
|         return Settings::NativeKeyboard::E; | ||||
|     case Qt::Key_F: | ||||
|         return Settings::NativeKeyboard::F; | ||||
|     case Qt::Key_G: | ||||
|         return Settings::NativeKeyboard::G; | ||||
|     case Qt::Key_H: | ||||
|         return Settings::NativeKeyboard::H; | ||||
|     case Qt::Key_I: | ||||
|         return Settings::NativeKeyboard::I; | ||||
|     case Qt::Key_J: | ||||
|         return Settings::NativeKeyboard::J; | ||||
|     case Qt::Key_K: | ||||
|         return Settings::NativeKeyboard::K; | ||||
|     case Qt::Key_L: | ||||
|         return Settings::NativeKeyboard::L; | ||||
|     case Qt::Key_M: | ||||
|         return Settings::NativeKeyboard::M; | ||||
|     case Qt::Key_N: | ||||
|         return Settings::NativeKeyboard::N; | ||||
|     case Qt::Key_O: | ||||
|         return Settings::NativeKeyboard::O; | ||||
|     case Qt::Key_P: | ||||
|         return Settings::NativeKeyboard::P; | ||||
|     case Qt::Key_Q: | ||||
|         return Settings::NativeKeyboard::Q; | ||||
|     case Qt::Key_R: | ||||
|         return Settings::NativeKeyboard::R; | ||||
|     case Qt::Key_S: | ||||
|         return Settings::NativeKeyboard::S; | ||||
|     case Qt::Key_T: | ||||
|         return Settings::NativeKeyboard::T; | ||||
|     case Qt::Key_U: | ||||
|         return Settings::NativeKeyboard::U; | ||||
|     case Qt::Key_V: | ||||
|         return Settings::NativeKeyboard::V; | ||||
|     case Qt::Key_W: | ||||
|         return Settings::NativeKeyboard::W; | ||||
|     case Qt::Key_X: | ||||
|         return Settings::NativeKeyboard::X; | ||||
|     case Qt::Key_Y: | ||||
|         return Settings::NativeKeyboard::Y; | ||||
|     case Qt::Key_Z: | ||||
|         return Settings::NativeKeyboard::Z; | ||||
|     case Qt::Key_1: | ||||
|         return Settings::NativeKeyboard::N1; | ||||
|     case Qt::Key_2: | ||||
|         return Settings::NativeKeyboard::N2; | ||||
|     case Qt::Key_3: | ||||
|         return Settings::NativeKeyboard::N3; | ||||
|     case Qt::Key_4: | ||||
|         return Settings::NativeKeyboard::N4; | ||||
|     case Qt::Key_5: | ||||
|         return Settings::NativeKeyboard::N5; | ||||
|     case Qt::Key_6: | ||||
|         return Settings::NativeKeyboard::N6; | ||||
|     case Qt::Key_7: | ||||
|         return Settings::NativeKeyboard::N7; | ||||
|     case Qt::Key_8: | ||||
|         return Settings::NativeKeyboard::N8; | ||||
|     case Qt::Key_9: | ||||
|         return Settings::NativeKeyboard::N9; | ||||
|     case Qt::Key_0: | ||||
|         return Settings::NativeKeyboard::N0; | ||||
|     case Qt::Key_Return: | ||||
|         return Settings::NativeKeyboard::Return; | ||||
|     case Qt::Key_Escape: | ||||
|         return Settings::NativeKeyboard::Escape; | ||||
|     case Qt::Key_Backspace: | ||||
|         return Settings::NativeKeyboard::Backspace; | ||||
|     case Qt::Key_Tab: | ||||
|         return Settings::NativeKeyboard::Tab; | ||||
|     case Qt::Key_Space: | ||||
|         return Settings::NativeKeyboard::Space; | ||||
|     case Qt::Key_Minus: | ||||
|         return Settings::NativeKeyboard::Minus; | ||||
|     case Qt::Key_Plus: | ||||
|     case Qt::Key_questiondown: | ||||
|         return Settings::NativeKeyboard::Plus; | ||||
|     case Qt::Key_BracketLeft: | ||||
|     case Qt::Key_BraceLeft: | ||||
|         return Settings::NativeKeyboard::OpenBracket; | ||||
|     case Qt::Key_BracketRight: | ||||
|     case Qt::Key_BraceRight: | ||||
|         return Settings::NativeKeyboard::CloseBracket; | ||||
|     case Qt::Key_Bar: | ||||
|         return Settings::NativeKeyboard::Pipe; | ||||
|     case Qt::Key_Dead_Tilde: | ||||
|         return Settings::NativeKeyboard::Tilde; | ||||
|     case Qt::Key_Ntilde: | ||||
|     case Qt::Key_Semicolon: | ||||
|         return Settings::NativeKeyboard::Semicolon; | ||||
|     case Qt::Key_Apostrophe: | ||||
|         return Settings::NativeKeyboard::Quote; | ||||
|     case Qt::Key_Dead_Grave: | ||||
|         return Settings::NativeKeyboard::Backquote; | ||||
|     case Qt::Key_Comma: | ||||
|         return Settings::NativeKeyboard::Comma; | ||||
|     case Qt::Key_Period: | ||||
|         return Settings::NativeKeyboard::Period; | ||||
|     case Qt::Key_Slash: | ||||
|         return Settings::NativeKeyboard::Slash; | ||||
|     case Qt::Key_CapsLock: | ||||
|         return Settings::NativeKeyboard::CapsLock; | ||||
|     case Qt::Key_F1: | ||||
|         return Settings::NativeKeyboard::F1; | ||||
|     case Qt::Key_F2: | ||||
|         return Settings::NativeKeyboard::F2; | ||||
|     case Qt::Key_F3: | ||||
|         return Settings::NativeKeyboard::F3; | ||||
|     case Qt::Key_F4: | ||||
|         return Settings::NativeKeyboard::F4; | ||||
|     case Qt::Key_F5: | ||||
|         return Settings::NativeKeyboard::F5; | ||||
|     case Qt::Key_F6: | ||||
|         return Settings::NativeKeyboard::F6; | ||||
|     case Qt::Key_F7: | ||||
|         return Settings::NativeKeyboard::F7; | ||||
|     case Qt::Key_F8: | ||||
|         return Settings::NativeKeyboard::F8; | ||||
|     case Qt::Key_F9: | ||||
|         return Settings::NativeKeyboard::F9; | ||||
|     case Qt::Key_F10: | ||||
|         return Settings::NativeKeyboard::F10; | ||||
|     case Qt::Key_F11: | ||||
|         return Settings::NativeKeyboard::F11; | ||||
|     case Qt::Key_F12: | ||||
|         return Settings::NativeKeyboard::F12; | ||||
|     case Qt::Key_Print: | ||||
|         return Settings::NativeKeyboard::PrintScreen; | ||||
|     case Qt::Key_ScrollLock: | ||||
|         return Settings::NativeKeyboard::ScrollLock; | ||||
|     case Qt::Key_Pause: | ||||
|         return Settings::NativeKeyboard::Pause; | ||||
|     case Qt::Key_Insert: | ||||
|         return Settings::NativeKeyboard::Insert; | ||||
|     case Qt::Key_Home: | ||||
|         return Settings::NativeKeyboard::Home; | ||||
|     case Qt::Key_PageUp: | ||||
|         return Settings::NativeKeyboard::PageUp; | ||||
|     case Qt::Key_Delete: | ||||
|         return Settings::NativeKeyboard::Delete; | ||||
|     case Qt::Key_End: | ||||
|         return Settings::NativeKeyboard::End; | ||||
|     case Qt::Key_PageDown: | ||||
|         return Settings::NativeKeyboard::PageDown; | ||||
|     case Qt::Key_Right: | ||||
|         return Settings::NativeKeyboard::Right; | ||||
|     case Qt::Key_Left: | ||||
|         return Settings::NativeKeyboard::Left; | ||||
|     case Qt::Key_Down: | ||||
|         return Settings::NativeKeyboard::Down; | ||||
|     case Qt::Key_Up: | ||||
|         return Settings::NativeKeyboard::Up; | ||||
|     case Qt::Key_NumLock: | ||||
|         return Settings::NativeKeyboard::NumLock; | ||||
|     // Numpad keys are missing here | ||||
|     case Qt::Key_F13: | ||||
|         return Settings::NativeKeyboard::F13; | ||||
|     case Qt::Key_F14: | ||||
|         return Settings::NativeKeyboard::F14; | ||||
|     case Qt::Key_F15: | ||||
|         return Settings::NativeKeyboard::F15; | ||||
|     case Qt::Key_F16: | ||||
|         return Settings::NativeKeyboard::F16; | ||||
|     case Qt::Key_F17: | ||||
|         return Settings::NativeKeyboard::F17; | ||||
|     case Qt::Key_F18: | ||||
|         return Settings::NativeKeyboard::F18; | ||||
|     case Qt::Key_F19: | ||||
|         return Settings::NativeKeyboard::F19; | ||||
|     case Qt::Key_F20: | ||||
|         return Settings::NativeKeyboard::F20; | ||||
|     case Qt::Key_F21: | ||||
|         return Settings::NativeKeyboard::F21; | ||||
|     case Qt::Key_F22: | ||||
|         return Settings::NativeKeyboard::F22; | ||||
|     case Qt::Key_F23: | ||||
|         return Settings::NativeKeyboard::F23; | ||||
|     case Qt::Key_F24: | ||||
|         return Settings::NativeKeyboard::F24; | ||||
|     // case Qt::: | ||||
|     //    return Settings::NativeKeyboard::KPComma; | ||||
|     // case Qt::: | ||||
|     //    return Settings::NativeKeyboard::Ro; | ||||
|     case Qt::Key_Hiragana_Katakana: | ||||
|         return Settings::NativeKeyboard::KatakanaHiragana; | ||||
|     case Qt::Key_yen: | ||||
|         return Settings::NativeKeyboard::Yen; | ||||
|     case Qt::Key_Henkan: | ||||
|         return Settings::NativeKeyboard::Henkan; | ||||
|     case Qt::Key_Muhenkan: | ||||
|         return Settings::NativeKeyboard::Muhenkan; | ||||
|     // case Qt::: | ||||
|     //    return Settings::NativeKeyboard::NumPadCommaPc98; | ||||
|     case Qt::Key_Hangul: | ||||
|         return Settings::NativeKeyboard::HangulEnglish; | ||||
|     case Qt::Key_Hangul_Hanja: | ||||
|         return Settings::NativeKeyboard::Hanja; | ||||
|     case Qt::Key_Katakana: | ||||
|         return Settings::NativeKeyboard::KatakanaKey; | ||||
|     case Qt::Key_Hiragana: | ||||
|         return Settings::NativeKeyboard::HiraganaKey; | ||||
|     case Qt::Key_Zenkaku_Hankaku: | ||||
|         return Settings::NativeKeyboard::ZenkakuHankaku; | ||||
|     // Modifier keys are handled by the modifier property | ||||
|     default: | ||||
|         return 0; | ||||
|     } | ||||
| } | ||||
|  | ||||
| int GRenderWindow::QtModifierToSwitchModdifier(quint32 qt_moddifiers) { | ||||
|     int moddifier = 0; | ||||
|     // The values are obtained through testing, Qt doesn't seem to provide a proper enum | ||||
|     if ((qt_moddifiers & 0x1) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::LeftShift; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x2) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::LeftControl; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x4) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::LeftAlt; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x08) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::LeftMeta; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x10) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::RightShift; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x20) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::RightControl; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x40) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::RightAlt; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x80) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::RightMeta; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x100) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::CapsLock; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x200) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::NumLock; | ||||
|     } | ||||
|     // Verify the last two keys | ||||
|     if ((qt_moddifiers & 0x400) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::Katakana; | ||||
|     } | ||||
|     if ((qt_moddifiers & 0x800) != 0) { | ||||
|         moddifier |= 1 << Settings::NativeKeyboard::Hiragana; | ||||
|     } | ||||
|     return moddifier; | ||||
| } | ||||
|  | ||||
| void GRenderWindow::keyPressEvent(QKeyEvent* event) { | ||||
|     if (!event->isAutoRepeat()) { | ||||
|         const auto moddifier = QtModifierToSwitchModdifier(event->nativeModifiers()); | ||||
|         // Replace event->key() with event->nativeVirtualKey() since the second one provides raw key | ||||
|         // buttons | ||||
|         const auto key = QtKeyToSwitchKey(Qt::Key(event->key())); | ||||
|         input_subsystem->GetKeyboard()->SetModifiers(moddifier); | ||||
|         input_subsystem->GetKeyboard()->PressKey(key); | ||||
|         input_subsystem->GetKeyboard()->PressKey(event->key()); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void GRenderWindow::keyReleaseEvent(QKeyEvent* event) { | ||||
|     if (!event->isAutoRepeat()) { | ||||
|         const auto moddifier = QtModifierToSwitchModdifier(event->nativeModifiers()); | ||||
|         const auto key = QtKeyToSwitchKey(Qt::Key(event->key())); | ||||
|         input_subsystem->GetKeyboard()->SetModifiers(moddifier); | ||||
|         input_subsystem->GetKeyboard()->ReleaseKey(key); | ||||
|         input_subsystem->GetKeyboard()->ReleaseKey(event->key()); | ||||
|     } | ||||
| } | ||||
|  | ||||
| InputCommon::MouseButton GRenderWindow::QtButtonToMouseButton(Qt::MouseButton button) { | ||||
| MouseInput::MouseButton GRenderWindow::QtButtonToMouseButton(Qt::MouseButton button) { | ||||
|     switch (button) { | ||||
|     case Qt::LeftButton: | ||||
|         return InputCommon::MouseButton::Left; | ||||
|         return MouseInput::MouseButton::Left; | ||||
|     case Qt::RightButton: | ||||
|         return InputCommon::MouseButton::Right; | ||||
|         return MouseInput::MouseButton::Right; | ||||
|     case Qt::MiddleButton: | ||||
|         return InputCommon::MouseButton::Wheel; | ||||
|         return MouseInput::MouseButton::Wheel; | ||||
|     case Qt::BackButton: | ||||
|         return InputCommon::MouseButton::Backward; | ||||
|         return MouseInput::MouseButton::Backward; | ||||
|     case Qt::ForwardButton: | ||||
|         return InputCommon::MouseButton::Forward; | ||||
|         return MouseInput::MouseButton::Forward; | ||||
|     case Qt::TaskButton: | ||||
|         return InputCommon::MouseButton::Task; | ||||
|         return MouseInput::MouseButton::Task; | ||||
|     default: | ||||
|         return InputCommon::MouseButton::Extra; | ||||
|         return MouseInput::MouseButton::Extra; | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -695,9 +423,12 @@ void GRenderWindow::mousePressEvent(QMouseEvent* event) { | ||||
|     // coordinates and map them to the current render area | ||||
|     const auto pos = mapFromGlobal(QCursor::pos()); | ||||
|     const auto [x, y] = ScaleTouch(pos); | ||||
|     const auto [touch_x, touch_y] = MapToTouchScreen(x, y); | ||||
|     const auto button = QtButtonToMouseButton(event->button()); | ||||
|     input_subsystem->GetMouse()->PressButton(x, y, touch_x, touch_y, button); | ||||
|     input_subsystem->GetMouse()->PressButton(x, y, button); | ||||
|  | ||||
|     if (event->button() == Qt::LeftButton) { | ||||
|         this->TouchPressed(x, y, 0); | ||||
|     } | ||||
|  | ||||
|     emit MouseActivity(); | ||||
| } | ||||
| @@ -711,10 +442,10 @@ void GRenderWindow::mouseMoveEvent(QMouseEvent* event) { | ||||
|     // coordinates and map them to the current render area | ||||
|     const auto pos = mapFromGlobal(QCursor::pos()); | ||||
|     const auto [x, y] = ScaleTouch(pos); | ||||
|     const auto [touch_x, touch_y] = MapToTouchScreen(x, y); | ||||
|     const int center_x = width() / 2; | ||||
|     const int center_y = height() / 2; | ||||
|     input_subsystem->GetMouse()->MouseMove(x, y, touch_x, touch_y, center_x, center_y); | ||||
|     input_subsystem->GetMouse()->MouseMove(x, y, center_x, center_y); | ||||
|     this->TouchMoved(x, y, 0); | ||||
|  | ||||
|     if (Settings::values.mouse_panning) { | ||||
|         QCursor::setPos(mapToGlobal({center_x, center_y})); | ||||
| @@ -731,6 +462,10 @@ void GRenderWindow::mouseReleaseEvent(QMouseEvent* event) { | ||||
|  | ||||
|     const auto button = QtButtonToMouseButton(event->button()); | ||||
|     input_subsystem->GetMouse()->ReleaseButton(button); | ||||
|  | ||||
|     if (event->button() == Qt::LeftButton) { | ||||
|         this->TouchReleased(0); | ||||
|     } | ||||
| } | ||||
|  | ||||
| void GRenderWindow::TouchBeginEvent(const QTouchEvent* event) { | ||||
| @@ -753,7 +488,7 @@ void GRenderWindow::TouchUpdateEvent(const QTouchEvent* event) { | ||||
|     for (std::size_t id = 0; id < touch_ids.size(); ++id) { | ||||
|         if (!TouchExist(touch_ids[id], touch_points)) { | ||||
|             touch_ids[id] = 0; | ||||
|             input_subsystem->GetTouchScreen()->TouchReleased(id); | ||||
|             this->TouchReleased(id + 1); | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @@ -762,28 +497,28 @@ void GRenderWindow::TouchEndEvent() { | ||||
|     for (std::size_t id = 0; id < touch_ids.size(); ++id) { | ||||
|         if (touch_ids[id] != 0) { | ||||
|             touch_ids[id] = 0; | ||||
|             input_subsystem->GetTouchScreen()->TouchReleased(id); | ||||
|             this->TouchReleased(id + 1); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| void GRenderWindow::TouchStart(const QTouchEvent::TouchPoint& touch_point) { | ||||
| bool GRenderWindow::TouchStart(const QTouchEvent::TouchPoint& touch_point) { | ||||
|     for (std::size_t id = 0; id < touch_ids.size(); ++id) { | ||||
|         if (touch_ids[id] == 0) { | ||||
|             touch_ids[id] = touch_point.id() + 1; | ||||
|             const auto [x, y] = ScaleTouch(touch_point.pos()); | ||||
|             const auto [touch_x, touch_y] = MapToTouchScreen(x, y); | ||||
|             input_subsystem->GetTouchScreen()->TouchPressed(touch_x, touch_y, id); | ||||
|             this->TouchPressed(x, y, id + 1); | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
|     return false; | ||||
| } | ||||
|  | ||||
| bool GRenderWindow::TouchUpdate(const QTouchEvent::TouchPoint& touch_point) { | ||||
|     for (std::size_t id = 0; id < touch_ids.size(); ++id) { | ||||
|         if (touch_ids[id] == static_cast<std::size_t>(touch_point.id() + 1)) { | ||||
|             const auto [x, y] = ScaleTouch(touch_point.pos()); | ||||
|             const auto [touch_x, touch_y] = MapToTouchScreen(x, y); | ||||
|             input_subsystem->GetTouchScreen()->TouchMoved(touch_x, touch_y, id); | ||||
|             this->TouchMoved(x, y, id + 1); | ||||
|             return true; | ||||
|         } | ||||
|     } | ||||
| @@ -816,7 +551,7 @@ void GRenderWindow::focusOutEvent(QFocusEvent* event) { | ||||
|     QWidget::focusOutEvent(event); | ||||
|     input_subsystem->GetKeyboard()->ReleaseAllKeys(); | ||||
|     input_subsystem->GetMouse()->ReleaseAllButtons(); | ||||
|     input_subsystem->GetTouchScreen()->ReleaseAllTouch(); | ||||
|     this->TouchReleased(0); | ||||
| } | ||||
|  | ||||
| void GRenderWindow::resizeEvent(QResizeEvent* event) { | ||||
|   | ||||
| @@ -30,8 +30,11 @@ class System; | ||||
|  | ||||
| namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
|  | ||||
| namespace MouseInput { | ||||
| enum class MouseButton; | ||||
| } // namespace InputCommon | ||||
| } | ||||
|  | ||||
| namespace VideoCore { | ||||
| enum class LoadCallbackStage; | ||||
| @@ -154,17 +157,11 @@ public: | ||||
|  | ||||
|     void resizeEvent(QResizeEvent* event) override; | ||||
|  | ||||
|     /// Converts a Qt keybard key into NativeKeyboard key | ||||
|     static int QtKeyToSwitchKey(Qt::Key qt_keys); | ||||
|  | ||||
|     /// Converts a Qt modifier keys into NativeKeyboard modifier keys | ||||
|     static int QtModifierToSwitchModdifier(quint32 qt_moddifiers); | ||||
|  | ||||
|     void keyPressEvent(QKeyEvent* event) override; | ||||
|     void keyReleaseEvent(QKeyEvent* event) override; | ||||
|  | ||||
|     /// Converts a Qt mouse button into MouseInput mouse button | ||||
|     static InputCommon::MouseButton QtButtonToMouseButton(Qt::MouseButton button); | ||||
|     static MouseInput::MouseButton QtButtonToMouseButton(Qt::MouseButton button); | ||||
|  | ||||
|     void mousePressEvent(QMouseEvent* event) override; | ||||
|     void mouseMoveEvent(QMouseEvent* event) override; | ||||
| @@ -212,7 +209,7 @@ private: | ||||
|     void TouchUpdateEvent(const QTouchEvent* event); | ||||
|     void TouchEndEvent(); | ||||
|  | ||||
|     void TouchStart(const QTouchEvent::TouchPoint& touch_point); | ||||
|     bool TouchStart(const QTouchEvent::TouchPoint& touch_point); | ||||
|     bool TouchUpdate(const QTouchEvent::TouchPoint& touch_point); | ||||
|     bool TouchExist(std::size_t id, const QList<QTouchEvent::TouchPoint>& touch_points) const; | ||||
|  | ||||
|   | ||||
| @@ -11,6 +11,7 @@ | ||||
| #include "core/hle/service/acc/profile_manager.h" | ||||
| #include "core/hle/service/hid/controllers/npad.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/udp/client.h" | ||||
| #include "yuzu/configuration/config.h" | ||||
|  | ||||
| namespace FS = Common::FS; | ||||
| @@ -65,6 +66,157 @@ const std::array<int, Settings::NativeMouseButton::NumMouseButtons> Config::defa | ||||
|         Qt::Key_BracketLeft, Qt::Key_BracketRight, Qt::Key_Apostrophe, Qt::Key_Minus, Qt::Key_Equal, | ||||
| }; | ||||
|  | ||||
| const std::array<int, Settings::NativeKeyboard::NumKeyboardKeys> Config::default_keyboard_keys = { | ||||
|     0, | ||||
|     0, | ||||
|     0, | ||||
|     0, | ||||
|     Qt::Key_A, | ||||
|     Qt::Key_B, | ||||
|     Qt::Key_C, | ||||
|     Qt::Key_D, | ||||
|     Qt::Key_E, | ||||
|     Qt::Key_F, | ||||
|     Qt::Key_G, | ||||
|     Qt::Key_H, | ||||
|     Qt::Key_I, | ||||
|     Qt::Key_J, | ||||
|     Qt::Key_K, | ||||
|     Qt::Key_L, | ||||
|     Qt::Key_M, | ||||
|     Qt::Key_N, | ||||
|     Qt::Key_O, | ||||
|     Qt::Key_P, | ||||
|     Qt::Key_Q, | ||||
|     Qt::Key_R, | ||||
|     Qt::Key_S, | ||||
|     Qt::Key_T, | ||||
|     Qt::Key_U, | ||||
|     Qt::Key_V, | ||||
|     Qt::Key_W, | ||||
|     Qt::Key_X, | ||||
|     Qt::Key_Y, | ||||
|     Qt::Key_Z, | ||||
|     Qt::Key_1, | ||||
|     Qt::Key_2, | ||||
|     Qt::Key_3, | ||||
|     Qt::Key_4, | ||||
|     Qt::Key_5, | ||||
|     Qt::Key_6, | ||||
|     Qt::Key_7, | ||||
|     Qt::Key_8, | ||||
|     Qt::Key_9, | ||||
|     Qt::Key_0, | ||||
|     Qt::Key_Enter, | ||||
|     Qt::Key_Escape, | ||||
|     Qt::Key_Backspace, | ||||
|     Qt::Key_Tab, | ||||
|     Qt::Key_Space, | ||||
|     Qt::Key_Minus, | ||||
|     Qt::Key_Equal, | ||||
|     Qt::Key_BracketLeft, | ||||
|     Qt::Key_BracketRight, | ||||
|     Qt::Key_Backslash, | ||||
|     Qt::Key_Dead_Tilde, | ||||
|     Qt::Key_Semicolon, | ||||
|     Qt::Key_Apostrophe, | ||||
|     Qt::Key_Dead_Grave, | ||||
|     Qt::Key_Comma, | ||||
|     Qt::Key_Period, | ||||
|     Qt::Key_Slash, | ||||
|     Qt::Key_CapsLock, | ||||
|  | ||||
|     Qt::Key_F1, | ||||
|     Qt::Key_F2, | ||||
|     Qt::Key_F3, | ||||
|     Qt::Key_F4, | ||||
|     Qt::Key_F5, | ||||
|     Qt::Key_F6, | ||||
|     Qt::Key_F7, | ||||
|     Qt::Key_F8, | ||||
|     Qt::Key_F9, | ||||
|     Qt::Key_F10, | ||||
|     Qt::Key_F11, | ||||
|     Qt::Key_F12, | ||||
|  | ||||
|     Qt::Key_SysReq, | ||||
|     Qt::Key_ScrollLock, | ||||
|     Qt::Key_Pause, | ||||
|     Qt::Key_Insert, | ||||
|     Qt::Key_Home, | ||||
|     Qt::Key_PageUp, | ||||
|     Qt::Key_Delete, | ||||
|     Qt::Key_End, | ||||
|     Qt::Key_PageDown, | ||||
|     Qt::Key_Right, | ||||
|     Qt::Key_Left, | ||||
|     Qt::Key_Down, | ||||
|     Qt::Key_Up, | ||||
|  | ||||
|     Qt::Key_NumLock, | ||||
|     Qt::Key_Slash, | ||||
|     Qt::Key_Asterisk, | ||||
|     Qt::Key_Minus, | ||||
|     Qt::Key_Plus, | ||||
|     Qt::Key_Enter, | ||||
|     Qt::Key_1, | ||||
|     Qt::Key_2, | ||||
|     Qt::Key_3, | ||||
|     Qt::Key_4, | ||||
|     Qt::Key_5, | ||||
|     Qt::Key_6, | ||||
|     Qt::Key_7, | ||||
|     Qt::Key_8, | ||||
|     Qt::Key_9, | ||||
|     Qt::Key_0, | ||||
|     Qt::Key_Period, | ||||
|  | ||||
|     0, | ||||
|     0, | ||||
|     Qt::Key_PowerOff, | ||||
|     Qt::Key_Equal, | ||||
|  | ||||
|     Qt::Key_F13, | ||||
|     Qt::Key_F14, | ||||
|     Qt::Key_F15, | ||||
|     Qt::Key_F16, | ||||
|     Qt::Key_F17, | ||||
|     Qt::Key_F18, | ||||
|     Qt::Key_F19, | ||||
|     Qt::Key_F20, | ||||
|     Qt::Key_F21, | ||||
|     Qt::Key_F22, | ||||
|     Qt::Key_F23, | ||||
|     Qt::Key_F24, | ||||
|  | ||||
|     Qt::Key_Open, | ||||
|     Qt::Key_Help, | ||||
|     Qt::Key_Menu, | ||||
|     0, | ||||
|     Qt::Key_Stop, | ||||
|     Qt::Key_AudioRepeat, | ||||
|     Qt::Key_Undo, | ||||
|     Qt::Key_Cut, | ||||
|     Qt::Key_Copy, | ||||
|     Qt::Key_Paste, | ||||
|     Qt::Key_Find, | ||||
|     Qt::Key_VolumeMute, | ||||
|     Qt::Key_VolumeUp, | ||||
|     Qt::Key_VolumeDown, | ||||
|     Qt::Key_CapsLock, | ||||
|     Qt::Key_NumLock, | ||||
|     Qt::Key_ScrollLock, | ||||
|     Qt::Key_Comma, | ||||
|  | ||||
|     Qt::Key_ParenLeft, | ||||
|     Qt::Key_ParenRight, | ||||
| }; | ||||
|  | ||||
| const std::array<int, Settings::NativeKeyboard::NumKeyboardMods> Config::default_keyboard_mods = { | ||||
|     Qt::Key_Control, Qt::Key_Shift, Qt::Key_Alt,   Qt::Key_ApplicationLeft, | ||||
|     Qt::Key_Control, Qt::Key_Shift, Qt::Key_AltGr, Qt::Key_ApplicationRight, | ||||
| }; | ||||
|  | ||||
| // This shouldn't have anything except static initializers (no functions). So | ||||
| // QKeySequence(...).toString() is NOT ALLOWED HERE. | ||||
| // This must be in alphabetical order according to action name as it must have the same order as | ||||
| @@ -345,14 +497,14 @@ void Config::ReadDebugValues() { | ||||
| void Config::ReadKeyboardValues() { | ||||
|     ReadBasicSetting(Settings::values.keyboard_enabled); | ||||
|  | ||||
|     for (std::size_t i = 0; i < Settings::values.keyboard_keys.size(); ++i) { | ||||
|         Settings::values.keyboard_keys[i] = InputCommon::GenerateKeyboardParam(static_cast<int>(i)); | ||||
|     } | ||||
|  | ||||
|     for (std::size_t i = 0; i < Settings::values.keyboard_mods.size(); ++i) { | ||||
|         Settings::values.keyboard_mods[i] = | ||||
|             InputCommon::GenerateModdifierKeyboardParam(static_cast<int>(i)); | ||||
|     } | ||||
|     std::transform(default_keyboard_keys.begin(), default_keyboard_keys.end(), | ||||
|                    Settings::values.keyboard_keys.begin(), InputCommon::GenerateKeyboardParam); | ||||
|     std::transform(default_keyboard_mods.begin(), default_keyboard_mods.end(), | ||||
|                    Settings::values.keyboard_keys.begin() + | ||||
|                        Settings::NativeKeyboard::LeftControlKey, | ||||
|                    InputCommon::GenerateKeyboardParam); | ||||
|     std::transform(default_keyboard_mods.begin(), default_keyboard_mods.end(), | ||||
|                    Settings::values.keyboard_mods.begin(), InputCommon::GenerateKeyboardParam); | ||||
| } | ||||
|  | ||||
| void Config::ReadMouseValues() { | ||||
| @@ -422,6 +574,7 @@ void Config::ReadControlValues() { | ||||
|  | ||||
|     ReadBasicSetting(Settings::values.tas_enable); | ||||
|     ReadBasicSetting(Settings::values.tas_loop); | ||||
|     ReadBasicSetting(Settings::values.tas_swap_controllers); | ||||
|     ReadBasicSetting(Settings::values.pause_tas_on_load); | ||||
|  | ||||
|     ReadGlobalSetting(Settings::values.use_docked_mode); | ||||
| @@ -472,7 +625,9 @@ void Config::ReadMotionTouchValues() { | ||||
|     } | ||||
|     qt_config->endArray(); | ||||
|  | ||||
|     ReadBasicSetting(Settings::values.motion_device); | ||||
|     ReadBasicSetting(Settings::values.touch_device); | ||||
|     ReadBasicSetting(Settings::values.use_touch_from_button); | ||||
|     ReadBasicSetting(Settings::values.touch_from_button_map_index); | ||||
|     Settings::values.touch_from_button_map_index = std::clamp( | ||||
|         Settings::values.touch_from_button_map_index.GetValue(), 0, num_touch_from_button_maps - 1); | ||||
| @@ -978,7 +1133,9 @@ void Config::SaveTouchscreenValues() { | ||||
| } | ||||
|  | ||||
| void Config::SaveMotionTouchValues() { | ||||
|     WriteBasicSetting(Settings::values.motion_device); | ||||
|     WriteBasicSetting(Settings::values.touch_device); | ||||
|     WriteBasicSetting(Settings::values.use_touch_from_button); | ||||
|     WriteBasicSetting(Settings::values.touch_from_button_map_index); | ||||
|     WriteBasicSetting(Settings::values.udp_input_servers); | ||||
|  | ||||
| @@ -1053,6 +1210,7 @@ void Config::SaveControlValues() { | ||||
|  | ||||
|     WriteBasicSetting(Settings::values.tas_enable); | ||||
|     WriteBasicSetting(Settings::values.tas_loop); | ||||
|     WriteBasicSetting(Settings::values.tas_swap_controllers); | ||||
|     WriteBasicSetting(Settings::values.pause_tas_on_load); | ||||
|  | ||||
|     qt_config->endGroup(); | ||||
|   | ||||
| @@ -2,18 +2,17 @@ | ||||
| // Licensed under GPLv2 or any later version | ||||
| // Refer to the license.txt file included. | ||||
|  | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/core.h" | ||||
| #include "ui_configure_debug_controller.h" | ||||
| #include "yuzu/configuration/configure_debug_controller.h" | ||||
| #include "yuzu/configuration/configure_input_player.h" | ||||
|  | ||||
| ConfigureDebugController::ConfigureDebugController(QWidget* parent, | ||||
|                                                    InputCommon::InputSubsystem* input_subsystem, | ||||
|                                                    InputProfiles* profiles, | ||||
|                                                    Core::HID::HIDCore& hid_core, bool is_powered_on) | ||||
|                                                    InputProfiles* profiles, Core::System& system) | ||||
|     : QDialog(parent), ui(std::make_unique<Ui::ConfigureDebugController>()), | ||||
|       debug_controller(new ConfigureInputPlayer(this, 9, nullptr, input_subsystem, profiles, | ||||
|                                                 hid_core, is_powered_on, true)) { | ||||
|       debug_controller( | ||||
|           new ConfigureInputPlayer(this, 9, nullptr, input_subsystem, profiles, system, true)) { | ||||
|     ui->setupUi(this); | ||||
|  | ||||
|     ui->controllerLayout->addWidget(debug_controller); | ||||
|   | ||||
| @@ -13,8 +13,8 @@ class ConfigureInputPlayer; | ||||
|  | ||||
| class InputProfiles; | ||||
|  | ||||
| namespace Core::HID { | ||||
| class HIDCore; | ||||
| namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace InputCommon { | ||||
| @@ -30,8 +30,7 @@ class ConfigureDebugController : public QDialog { | ||||
|  | ||||
| public: | ||||
|     explicit ConfigureDebugController(QWidget* parent, InputCommon::InputSubsystem* input_subsystem, | ||||
|                                       InputProfiles* profiles, Core::HID::HIDCore& hid_core, | ||||
|                                       bool is_powered_on); | ||||
|                                       InputProfiles* profiles, Core::System& system); | ||||
|     ~ConfigureDebugController() override; | ||||
|  | ||||
|     void ApplyConfiguration(); | ||||
|   | ||||
| @@ -10,8 +10,6 @@ | ||||
| #include <QTimer> | ||||
|  | ||||
| #include "core/core.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/am/am.h" | ||||
| #include "core/hle/service/am/applet_ae.h" | ||||
| #include "core/hle/service/am/applet_oe.h" | ||||
| @@ -77,25 +75,23 @@ ConfigureInput::~ConfigureInput() = default; | ||||
|  | ||||
| void ConfigureInput::Initialize(InputCommon::InputSubsystem* input_subsystem, | ||||
|                                 std::size_t max_players) { | ||||
|     const bool is_powered_on = system.IsPoweredOn(); | ||||
|     auto& hid_core = system.HIDCore(); | ||||
|     player_controllers = { | ||||
|         new ConfigureInputPlayer(this, 0, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 1, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 2, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 3, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 4, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 5, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 6, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|         new ConfigureInputPlayer(this, 7, ui->consoleInputSettings, input_subsystem, profiles.get(), | ||||
|                                  hid_core, is_powered_on), | ||||
|                                  system), | ||||
|     }; | ||||
|  | ||||
|     player_tabs = { | ||||
| @@ -118,7 +114,6 @@ void ConfigureInput::Initialize(InputCommon::InputSubsystem* input_subsystem, | ||||
|         player_tabs[i]->setLayout(new QHBoxLayout(player_tabs[i])); | ||||
|         player_tabs[i]->layout()->addWidget(player_controllers[i]); | ||||
|         connect(player_controllers[i], &ConfigureInputPlayer::Connected, [&, i](bool is_connected) { | ||||
|             // Ensures that the controllers are always connected in sequential order | ||||
|             if (is_connected) { | ||||
|                 for (std::size_t index = 0; index <= i; ++index) { | ||||
|                     player_connected[index]->setChecked(is_connected); | ||||
| @@ -151,12 +146,10 @@ void ConfigureInput::Initialize(InputCommon::InputSubsystem* input_subsystem, | ||||
|     advanced = new ConfigureInputAdvanced(this); | ||||
|     ui->tabAdvanced->setLayout(new QHBoxLayout(ui->tabAdvanced)); | ||||
|     ui->tabAdvanced->layout()->addWidget(advanced); | ||||
|  | ||||
|     connect(advanced, &ConfigureInputAdvanced::CallDebugControllerDialog, | ||||
|             [this, input_subsystem, &hid_core, is_powered_on] { | ||||
|                 CallConfigureDialog<ConfigureDebugController>( | ||||
|                     *this, input_subsystem, profiles.get(), hid_core, is_powered_on); | ||||
|             }); | ||||
|     connect(advanced, &ConfigureInputAdvanced::CallDebugControllerDialog, [this, input_subsystem] { | ||||
|         CallConfigureDialog<ConfigureDebugController>(*this, input_subsystem, profiles.get(), | ||||
|                                                       system); | ||||
|     }); | ||||
|     connect(advanced, &ConfigureInputAdvanced::CallMouseConfigDialog, [this, input_subsystem] { | ||||
|         CallConfigureDialog<ConfigureMouseAdvanced>(*this, input_subsystem); | ||||
|     }); | ||||
| @@ -191,8 +184,22 @@ QList<QWidget*> ConfigureInput::GetSubTabs() const { | ||||
| void ConfigureInput::ApplyConfiguration() { | ||||
|     for (auto* controller : player_controllers) { | ||||
|         controller->ApplyConfiguration(); | ||||
|         controller->TryDisconnectSelectedController(); | ||||
|     } | ||||
|  | ||||
|     // This emulates a delay between disconnecting and reconnecting controllers as some games | ||||
|     // do not respond to a change in controller type if it was instantaneous. | ||||
|     using namespace std::chrono_literals; | ||||
|     std::this_thread::sleep_for(150ms); | ||||
|  | ||||
|     for (auto* controller : player_controllers) { | ||||
|         controller->TryConnectSelectedController(); | ||||
|     } | ||||
|  | ||||
|     // This emulates a delay between disconnecting and reconnecting controllers as some games | ||||
|     // do not respond to a change in controller type if it was instantaneous. | ||||
|     std::this_thread::sleep_for(150ms); | ||||
|  | ||||
|     advanced->ApplyConfiguration(); | ||||
|  | ||||
|     const bool pre_docked_mode = Settings::values.use_docked_mode.GetValue(); | ||||
| @@ -216,10 +223,8 @@ void ConfigureInput::RetranslateUI() { | ||||
| } | ||||
|  | ||||
| void ConfigureInput::LoadConfiguration() { | ||||
|     const auto* handheld = system.HIDCore().GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|  | ||||
|     LoadPlayerControllerIndices(); | ||||
|     UpdateDockedState(handheld->IsConnected()); | ||||
|     UpdateDockedState(Settings::values.players.GetValue()[8].connected); | ||||
|  | ||||
|     ui->vibrationGroup->setChecked(Settings::values.vibration_enabled.GetValue()); | ||||
|     ui->motionGroup->setChecked(Settings::values.motion_enabled.GetValue()); | ||||
| @@ -227,16 +232,9 @@ void ConfigureInput::LoadConfiguration() { | ||||
|  | ||||
| void ConfigureInput::LoadPlayerControllerIndices() { | ||||
|     for (std::size_t i = 0; i < player_connected.size(); ++i) { | ||||
|         if (i == 0) { | ||||
|             auto* handheld = | ||||
|                 system.HIDCore().GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|             if (handheld->IsConnected()) { | ||||
|                 player_connected[i]->setChecked(true); | ||||
|                 continue; | ||||
|             } | ||||
|         } | ||||
|         const auto* controller = system.HIDCore().GetEmulatedControllerByIndex(i); | ||||
|         player_connected[i]->setChecked(controller->IsConnected()); | ||||
|         const auto connected = Settings::values.players.GetValue()[i].connected || | ||||
|                                (i == 0 && Settings::values.players.GetValue()[8].connected); | ||||
|         player_connected[i]->setChecked(connected); | ||||
|     } | ||||
| } | ||||
|  | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -29,37 +29,48 @@ class QWidget; | ||||
|  | ||||
| class InputProfiles; | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
|  | ||||
| namespace InputCommon::Polling { | ||||
| enum class InputType; | ||||
| class DevicePoller; | ||||
| enum class DeviceType; | ||||
| } // namespace InputCommon::Polling | ||||
|  | ||||
| namespace Ui { | ||||
| class ConfigureInputPlayer; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| class HIDCore; | ||||
| class EmulatedController; | ||||
| enum class NpadStyleIndex : u8; | ||||
| } // namespace Core::HID | ||||
|  | ||||
| class ConfigureInputPlayer : public QWidget { | ||||
|     Q_OBJECT | ||||
|  | ||||
| public: | ||||
|     explicit ConfigureInputPlayer(QWidget* parent, std::size_t player_index, QWidget* bottom_row, | ||||
|                                   InputCommon::InputSubsystem* input_subsystem_, | ||||
|                                   InputProfiles* profiles_, Core::HID::HIDCore& hid_core_, | ||||
|                                   bool is_powered_on_, bool debug = false); | ||||
|                                   InputProfiles* profiles_, Core::System& system_, | ||||
|                                   bool debug = false); | ||||
|     ~ConfigureInputPlayer() override; | ||||
|  | ||||
|     /// Save all button configurations to settings file. | ||||
|     void ApplyConfiguration(); | ||||
|  | ||||
|     /** | ||||
|      * Attempts to connect the currently selected controller in the HID backend. | ||||
|      * This function will not do anything if it is not connected in the frontend. | ||||
|      */ | ||||
|     void TryConnectSelectedController(); | ||||
|  | ||||
|     /** | ||||
|      * Attempts to disconnect the currently selected controller in the HID backend. | ||||
|      * This function will not do anything if the configuration has not changed. | ||||
|      */ | ||||
|     void TryDisconnectSelectedController(); | ||||
|  | ||||
|     /// Set the connection state checkbox (used to sync state). | ||||
|     void ConnectPlayer(bool connected); | ||||
|  | ||||
| @@ -93,10 +104,6 @@ protected: | ||||
|     void showEvent(QShowEvent* event) override; | ||||
|  | ||||
| private: | ||||
|     QString ButtonToText(const Common::ParamPackage& param); | ||||
|  | ||||
|     QString AnalogToText(const Common::ParamPackage& param, const std::string& dir); | ||||
|  | ||||
|     void changeEvent(QEvent* event) override; | ||||
|     void RetranslateUI(); | ||||
|  | ||||
| @@ -106,7 +113,7 @@ private: | ||||
|     /// Called when the button was pressed. | ||||
|     void HandleClick(QPushButton* button, std::size_t button_id, | ||||
|                      std::function<void(const Common::ParamPackage&)> new_input_setter, | ||||
|                      InputCommon::Polling::InputType type); | ||||
|                      InputCommon::Polling::DeviceType type); | ||||
|  | ||||
|     /// Finish polling and configure input using the input_setter. | ||||
|     void SetPollingResult(const Common::ParamPackage& params, bool abort); | ||||
| @@ -127,14 +134,17 @@ private: | ||||
|     void SetConnectableControllers(); | ||||
|  | ||||
|     /// Gets the Controller Type for a given controller combobox index. | ||||
|     Core::HID::NpadStyleIndex GetControllerTypeFromIndex(int index) const; | ||||
|     Settings::ControllerType GetControllerTypeFromIndex(int index) const; | ||||
|  | ||||
|     /// Gets the controller combobox index for a given Controller Type. | ||||
|     int GetIndexFromControllerType(Core::HID::NpadStyleIndex type) const; | ||||
|     int GetIndexFromControllerType(Settings::ControllerType type) const; | ||||
|  | ||||
|     /// Update the available input devices. | ||||
|     void UpdateInputDevices(); | ||||
|  | ||||
|     /// Update the current controller icon. | ||||
|     void UpdateControllerIcon(); | ||||
|  | ||||
|     /// Hides and disables controller settings based on the current controller type. | ||||
|     void UpdateControllerAvailableButtons(); | ||||
|  | ||||
| @@ -166,7 +176,6 @@ private: | ||||
|  | ||||
|     std::size_t player_index; | ||||
|     bool debug; | ||||
|     bool is_powered_on; | ||||
|  | ||||
|     InputCommon::InputSubsystem* input_subsystem; | ||||
|  | ||||
| @@ -176,7 +185,7 @@ private: | ||||
|     std::unique_ptr<QTimer> poll_timer; | ||||
|  | ||||
|     /// Stores a pair of "Connected Controllers" combobox index and Controller Type enum. | ||||
|     std::vector<std::pair<int, Core::HID::NpadStyleIndex>> index_controller_type_pairs; | ||||
|     std::vector<std::pair<int, Settings::ControllerType>> index_controller_type_pairs; | ||||
|  | ||||
|     static constexpr int PLAYER_COUNT = 8; | ||||
|     std::array<QCheckBox*, PLAYER_COUNT> player_connected_checkbox; | ||||
| @@ -184,7 +193,9 @@ private: | ||||
|     /// This will be the the setting function when an input is awaiting configuration. | ||||
|     std::optional<std::function<void(const Common::ParamPackage&)>> input_setter; | ||||
|  | ||||
|     Core::HID::EmulatedController* emulated_controller; | ||||
|     std::array<Common::ParamPackage, Settings::NativeButton::NumButtons> buttons_param; | ||||
|     std::array<Common::ParamPackage, Settings::NativeAnalog::NumAnalogs> analogs_param; | ||||
|     std::array<Common::ParamPackage, Settings::NativeMotion::NumMotions> motions_param; | ||||
|  | ||||
|     static constexpr int ANALOG_SUB_BUTTONS_NUM = 4; | ||||
|  | ||||
| @@ -210,9 +221,15 @@ private: | ||||
|  | ||||
|     static const std::array<std::string, ANALOG_SUB_BUTTONS_NUM> analog_sub_buttons; | ||||
|  | ||||
|     std::vector<std::unique_ptr<InputCommon::Polling::DevicePoller>> device_pollers; | ||||
|  | ||||
|     /// A flag to indicate that the "Map Analog Stick" pop-up has been shown and accepted once. | ||||
|     bool map_analog_stick_accepted{}; | ||||
|  | ||||
|     /// A flag to indicate if keyboard keys are okay when configuring an input. If this is false, | ||||
|     /// keyboard events are ignored. | ||||
|     bool want_keyboard_mouse{}; | ||||
|  | ||||
|     /// List of physical devices users can map with. If a SDL backed device is selected, then you | ||||
|     /// can use this device to get a default mapping. | ||||
|     std::vector<Common::ParamPackage> input_devices; | ||||
| @@ -222,5 +239,5 @@ private: | ||||
|     /// parent of the widget to this widget (but thats fine). | ||||
|     QWidget* bottom_row; | ||||
|  | ||||
|     Core::HID::HIDCore& hid_core; | ||||
|     Core::System& system; | ||||
| }; | ||||
|   | ||||
| @@ -89,6 +89,31 @@ | ||||
|               <height>21</height> | ||||
|              </size> | ||||
|             </property> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Pro Controller</string> | ||||
|              </property> | ||||
|             </item> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Dual Joycons</string> | ||||
|              </property> | ||||
|             </item> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Left Joycon</string> | ||||
|              </property> | ||||
|             </item> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Right Joycon</string> | ||||
|              </property> | ||||
|             </item> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Handheld</string> | ||||
|              </property> | ||||
|             </item> | ||||
|            </widget> | ||||
|           </item> | ||||
|          </layout> | ||||
| @@ -117,9 +142,22 @@ | ||||
|           </property> | ||||
|           <item> | ||||
|            <widget class="QComboBox" name="comboDevices"> | ||||
|             <property name="minimumContentsLength"> | ||||
|              <number>60</number> | ||||
|             <property name="minimumSize"> | ||||
|              <size> | ||||
|               <width>0</width> | ||||
|               <height>21</height> | ||||
|              </size> | ||||
|             </property> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Any</string> | ||||
|              </property> | ||||
|             </item> | ||||
|             <item> | ||||
|              <property name="text"> | ||||
|               <string>Keyboard/Mouse</string> | ||||
|              </property> | ||||
|             </item> | ||||
|            </widget> | ||||
|           </item> | ||||
|           <item> | ||||
| @@ -304,7 +342,7 @@ | ||||
|                <number>3</number> | ||||
|               </property> | ||||
|               <property name="topMargin"> | ||||
|                <number>6</number> | ||||
|                <number>0</number> | ||||
|               </property> | ||||
|               <property name="rightMargin"> | ||||
|                <number>3</number> | ||||
| @@ -880,7 +918,7 @@ | ||||
|                <number>3</number> | ||||
|               </property> | ||||
|               <property name="topMargin"> | ||||
|                <number>6</number> | ||||
|                <number>0</number> | ||||
|               </property> | ||||
|               <property name="rightMargin"> | ||||
|                <number>3</number> | ||||
| @@ -2183,7 +2221,7 @@ | ||||
|                <number>3</number> | ||||
|               </property> | ||||
|               <property name="topMargin"> | ||||
|                <number>6</number> | ||||
|                <number>0</number> | ||||
|               </property> | ||||
|               <property name="rightMargin"> | ||||
|                <number>3</number> | ||||
| @@ -2532,7 +2570,7 @@ | ||||
|                <number>3</number> | ||||
|               </property> | ||||
|               <property name="topMargin"> | ||||
|                <number>6</number> | ||||
|                <number>0</number> | ||||
|               </property> | ||||
|               <property name="rightMargin"> | ||||
|                <number>3</number> | ||||
|   | ||||
										
											
												File diff suppressed because it is too large
												Load Diff
											
										
									
								
							| @@ -7,11 +7,9 @@ | ||||
| #include <array> | ||||
| #include <QFrame> | ||||
| #include <QPointer> | ||||
|  | ||||
| #include "common/input.h" | ||||
| #include "common/settings_input.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "common/settings.h" | ||||
| #include "core/frontend/input.h" | ||||
| #include "yuzu/debugger/controller.h" | ||||
|  | ||||
| class QLabel; | ||||
|  | ||||
| @@ -26,26 +24,17 @@ public: | ||||
|     explicit PlayerControlPreview(QWidget* parent); | ||||
|     ~PlayerControlPreview() override; | ||||
|  | ||||
|     // Sets the emulated controller to be displayed | ||||
|     void SetController(Core::HID::EmulatedController* controller); | ||||
|  | ||||
|     // Disables events from the emulated controller | ||||
|     void UnloadController(); | ||||
|  | ||||
|     // Starts blinking animation at the button specified | ||||
|     void SetPlayerInput(std::size_t index, const ButtonParam& buttons_param, | ||||
|                         const AnalogParam& analogs_param); | ||||
|     void SetPlayerInputRaw(std::size_t index, const Settings::ButtonsRaw& buttons_, | ||||
|                            Settings::AnalogsRaw analogs_); | ||||
|     void SetConnectedStatus(bool checked); | ||||
|     void SetControllerType(Settings::ControllerType type); | ||||
|     void BeginMappingButton(std::size_t button_id); | ||||
|  | ||||
|     // Starts moving animation at the stick specified | ||||
|     void BeginMappingAnalog(std::size_t stick_id); | ||||
|  | ||||
|     // Stops any ongoing animation | ||||
|     void BeginMappingAnalog(std::size_t button_id); | ||||
|     void EndMapping(); | ||||
|  | ||||
|     // Handles emulated controller events | ||||
|     void ControllerUpdate(Core::HID::ControllerTriggerType type); | ||||
|  | ||||
|     // Updates input on sheduled interval | ||||
|     void UpdateInput(); | ||||
|     void SetCallBack(ControllerCallback callback_); | ||||
|  | ||||
| protected: | ||||
|     void paintEvent(QPaintEvent* event) override; | ||||
| @@ -74,6 +63,22 @@ private: | ||||
|         SR, | ||||
|     }; | ||||
|  | ||||
|     struct AxisValue { | ||||
|         QPointF value{}; | ||||
|         QPointF raw_value{}; | ||||
|         Input::AnalogProperties properties{}; | ||||
|         int size{}; | ||||
|         QPoint offset{}; | ||||
|         bool active{}; | ||||
|     }; | ||||
|  | ||||
|     struct LedPattern { | ||||
|         bool position1; | ||||
|         bool position2; | ||||
|         bool position3; | ||||
|         bool position4; | ||||
|     }; | ||||
|  | ||||
|     struct ColorMapping { | ||||
|         QColor outline{}; | ||||
|         QColor primary{}; | ||||
| @@ -96,6 +101,7 @@ private: | ||||
|         QColor deadzone{}; | ||||
|     }; | ||||
|  | ||||
|     static LedPattern GetColorPattern(std::size_t index, bool player_on); | ||||
|     void UpdateColors(); | ||||
|     void ResetInputs(); | ||||
|  | ||||
| @@ -116,75 +122,47 @@ private: | ||||
|     void DrawGCBody(QPainter& p, QPointF center); | ||||
|  | ||||
|     // Draw triggers functions | ||||
|     void DrawProTriggers(QPainter& p, QPointF center, | ||||
|                          const Common::Input::ButtonStatus& left_pressed, | ||||
|                          const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawGCTriggers(QPainter& p, QPointF center, Common::Input::TriggerStatus left_trigger, | ||||
|                         Common::Input::TriggerStatus right_trigger); | ||||
|     void DrawHandheldTriggers(QPainter& p, QPointF center, | ||||
|                               const Common::Input::ButtonStatus& left_pressed, | ||||
|                               const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawDualTriggers(QPainter& p, QPointF center, | ||||
|                           const Common::Input::ButtonStatus& left_pressed, | ||||
|                           const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawDualTriggersTopView(QPainter& p, QPointF center, | ||||
|                                  const Common::Input::ButtonStatus& left_pressed, | ||||
|                                  const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawDualZTriggersTopView(QPainter& p, QPointF center, | ||||
|                                   const Common::Input::ButtonStatus& left_pressed, | ||||
|                                   const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawLeftTriggers(QPainter& p, QPointF center, | ||||
|                           const Common::Input::ButtonStatus& left_pressed); | ||||
|     void DrawLeftZTriggers(QPainter& p, QPointF center, | ||||
|                            const Common::Input::ButtonStatus& left_pressed); | ||||
|     void DrawLeftTriggersTopView(QPainter& p, QPointF center, | ||||
|                                  const Common::Input::ButtonStatus& left_pressed); | ||||
|     void DrawLeftZTriggersTopView(QPainter& p, QPointF center, | ||||
|                                   const Common::Input::ButtonStatus& left_pressed); | ||||
|     void DrawRightTriggers(QPainter& p, QPointF center, | ||||
|                            const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawRightZTriggers(QPainter& p, QPointF center, | ||||
|                             const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawRightTriggersTopView(QPainter& p, QPointF center, | ||||
|                                   const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawRightZTriggersTopView(QPainter& p, QPointF center, | ||||
|                                    const Common::Input::ButtonStatus& right_pressed); | ||||
|     void DrawProTriggers(QPainter& p, QPointF center, bool left_pressed, bool right_pressed); | ||||
|     void DrawGCTriggers(QPainter& p, QPointF center, bool left_pressed, bool right_pressed); | ||||
|     void DrawHandheldTriggers(QPainter& p, QPointF center, bool left_pressed, bool right_pressed); | ||||
|     void DrawDualTriggers(QPainter& p, QPointF center, bool left_pressed, bool right_pressed); | ||||
|     void DrawDualTriggersTopView(QPainter& p, QPointF center, bool left_pressed, | ||||
|                                  bool right_pressed); | ||||
|     void DrawDualZTriggersTopView(QPainter& p, QPointF center, bool left_pressed, | ||||
|                                   bool right_pressed); | ||||
|     void DrawLeftTriggers(QPainter& p, QPointF center, bool left_pressed); | ||||
|     void DrawLeftZTriggers(QPainter& p, QPointF center, bool left_pressed); | ||||
|     void DrawLeftTriggersTopView(QPainter& p, QPointF center, bool left_pressed); | ||||
|     void DrawLeftZTriggersTopView(QPainter& p, QPointF center, bool left_pressed); | ||||
|     void DrawRightTriggers(QPainter& p, QPointF center, bool right_pressed); | ||||
|     void DrawRightZTriggers(QPainter& p, QPointF center, bool right_pressed); | ||||
|     void DrawRightTriggersTopView(QPainter& p, QPointF center, bool right_pressed); | ||||
|     void DrawRightZTriggersTopView(QPainter& p, QPointF center, bool right_pressed); | ||||
|  | ||||
|     // Draw joystick functions | ||||
|     void DrawJoystick(QPainter& p, QPointF center, float size, | ||||
|                       const Common::Input::ButtonStatus& pressed); | ||||
|     void DrawJoystickSideview(QPainter& p, QPointF center, float angle, float size, | ||||
|                               const Common::Input::ButtonStatus& pressed); | ||||
|     void DrawJoystick(QPainter& p, QPointF center, float size, bool pressed); | ||||
|     void DrawJoystickSideview(QPainter& p, QPointF center, float angle, float size, bool pressed); | ||||
|     void DrawRawJoystick(QPainter& p, QPointF center_left, QPointF center_right); | ||||
|     void DrawJoystickProperties(QPainter& p, QPointF center, | ||||
|                                 const Common::Input::AnalogProperties& properties); | ||||
|     void DrawJoystickDot(QPainter& p, QPointF center, const Common::Input::StickStatus& stick, | ||||
|                          bool raw); | ||||
|     void DrawProJoystick(QPainter& p, QPointF center, QPointF offset, float scalar, | ||||
|                          const Common::Input::ButtonStatus& pressed); | ||||
|     void DrawGCJoystick(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed); | ||||
|                                 const Input::AnalogProperties& properties); | ||||
|     void DrawJoystickDot(QPainter& p, QPointF center, QPointF value, | ||||
|                          const Input::AnalogProperties& properties); | ||||
|     void DrawProJoystick(QPainter& p, QPointF center, QPointF offset, float scalar, bool pressed); | ||||
|     void DrawGCJoystick(QPainter& p, QPointF center, bool pressed); | ||||
|  | ||||
|     // Draw button functions | ||||
|     void DrawCircleButton(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed, | ||||
|                           float button_size); | ||||
|     void DrawRoundButton(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed, | ||||
|                          float width, float height, Direction direction = Direction::None, | ||||
|                          float radius = 2); | ||||
|     void DrawMinusButton(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed, | ||||
|                          int button_size); | ||||
|     void DrawPlusButton(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed, | ||||
|                         int button_size); | ||||
|     void DrawGCButtonX(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed); | ||||
|     void DrawGCButtonY(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed); | ||||
|     void DrawGCButtonZ(QPainter& p, QPointF center, const Common::Input::ButtonStatus& pressed); | ||||
|     void DrawCircleButton(QPainter& p, QPointF center, bool pressed, float button_size); | ||||
|     void DrawRoundButton(QPainter& p, QPointF center, bool pressed, float width, float height, | ||||
|                          Direction direction = Direction::None, float radius = 2); | ||||
|     void DrawMinusButton(QPainter& p, QPointF center, bool pressed, int button_size); | ||||
|     void DrawPlusButton(QPainter& p, QPointF center, bool pressed, int button_size); | ||||
|     void DrawGCButtonX(QPainter& p, QPointF center, bool pressed); | ||||
|     void DrawGCButtonY(QPainter& p, QPointF center, bool pressed); | ||||
|     void DrawGCButtonZ(QPainter& p, QPointF center, bool pressed); | ||||
|     void DrawArrowButtonOutline(QPainter& p, const QPointF center, float size = 1.0f); | ||||
|     void DrawArrowButton(QPainter& p, QPointF center, Direction direction, | ||||
|                          const Common::Input::ButtonStatus& pressed, float size = 1.0f); | ||||
|     void DrawTriggerButton(QPainter& p, QPointF center, Direction direction, | ||||
|                            const Common::Input::ButtonStatus& pressed); | ||||
|  | ||||
|     // Draw battery functions | ||||
|     void DrawBattery(QPainter& p, QPointF center, Common::Input::BatteryLevel battery); | ||||
|     void DrawArrowButton(QPainter& p, QPointF center, Direction direction, bool pressed, | ||||
|                          float size = 1.0f); | ||||
|     void DrawTriggerButton(QPainter& p, QPointF center, Direction direction, bool pressed); | ||||
|  | ||||
|     // Draw icon functions | ||||
|     void DrawSymbol(QPainter& p, QPointF center, Symbol symbol, float icon_size); | ||||
| @@ -200,23 +178,24 @@ private: | ||||
|     void SetTextFont(QPainter& p, float text_size, | ||||
|                      const QString& font_family = QStringLiteral("sans-serif")); | ||||
|  | ||||
|     bool is_controller_set{}; | ||||
|     bool is_connected{}; | ||||
|     bool needs_redraw{}; | ||||
|     Core::HID::NpadStyleIndex controller_type; | ||||
|     using ButtonArray = | ||||
|         std::array<std::unique_ptr<Input::ButtonDevice>, Settings::NativeButton::BUTTON_NS_END>; | ||||
|     using StickArray = | ||||
|         std::array<std::unique_ptr<Input::AnalogDevice>, Settings::NativeAnalog::NUM_STICKS_HID>; | ||||
|  | ||||
|     ControllerCallback controller_callback; | ||||
|     bool is_enabled{}; | ||||
|     bool mapping_active{}; | ||||
|     int blink_counter{}; | ||||
|     int callback_key; | ||||
|     QColor button_color{}; | ||||
|     ColorMapping colors{}; | ||||
|     Core::HID::LedPattern led_pattern{0, 0, 0, 0}; | ||||
|     std::array<QColor, 4> led_color{}; | ||||
|     ButtonArray buttons{}; | ||||
|     StickArray sticks{}; | ||||
|     std::size_t player_index{}; | ||||
|     Core::HID::EmulatedController* controller; | ||||
|     std::size_t button_mapping_index{Settings::NativeButton::NumButtons}; | ||||
|     std::size_t analog_mapping_index{Settings::NativeAnalog::NumAnalogs}; | ||||
|     Core::HID::ButtonValues button_values{}; | ||||
|     Core::HID::SticksValues stick_values{}; | ||||
|     Core::HID::TriggerValues trigger_values{}; | ||||
|     Core::HID::BatteryValues battery_values{}; | ||||
|     std::size_t button_mapping_index{Settings::NativeButton::BUTTON_NS_END}; | ||||
|     std::size_t analog_mapping_index{Settings::NativeAnalog::NUM_STICKS_HID}; | ||||
|     std::array<AxisValue, Settings::NativeAnalog::NUM_STICKS_HID> axis_values{}; | ||||
|     std::array<bool, Settings::NativeButton::NumButtons> button_values{}; | ||||
|     Settings::ControllerType controller_type{Settings::ControllerType::ProController}; | ||||
| }; | ||||
|   | ||||
| @@ -11,8 +11,8 @@ ConfigureInputProfileDialog::ConfigureInputProfileDialog( | ||||
|     QWidget* parent, InputCommon::InputSubsystem* input_subsystem, InputProfiles* profiles, | ||||
|     Core::System& system) | ||||
|     : QDialog(parent), ui(std::make_unique<Ui::ConfigureInputProfileDialog>()), | ||||
|       profile_widget(new ConfigureInputPlayer(this, 9, nullptr, input_subsystem, profiles, | ||||
|                                               system.HIDCore(), system.IsPoweredOn(), false)) { | ||||
|       profile_widget( | ||||
|           new ConfigureInputPlayer(this, 9, nullptr, input_subsystem, profiles, system, false)) { | ||||
|     ui->setupUi(this); | ||||
|  | ||||
|     ui->controllerLayout->addWidget(profile_widget); | ||||
|   | ||||
| @@ -15,9 +15,9 @@ | ||||
|  | ||||
| #include "common/logging/log.h" | ||||
| #include "common/settings.h" | ||||
| #include "input_common/drivers/udp_client.h" | ||||
| #include "input_common/helpers/udp_protocol.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/udp/client.h" | ||||
| #include "input_common/udp/udp.h" | ||||
| #include "ui_configure_motion_touch.h" | ||||
| #include "yuzu/configuration/configure_motion_touch.h" | ||||
| #include "yuzu/configuration/configure_touch_from_button.h" | ||||
| @@ -93,7 +93,6 @@ ConfigureMotionTouch::ConfigureMotionTouch(QWidget* parent, | ||||
|            "using-a-controller-or-android-phone-for-motion-or-touch-input'><span " | ||||
|            "style=\"text-decoration: underline; color:#039be5;\">Learn More</span></a>")); | ||||
|  | ||||
|     setWindowFlags(windowFlags() & ~Qt::WindowContextHelpButtonHint); | ||||
|     SetConfiguration(); | ||||
|     UpdateUiDisplay(); | ||||
|     ConnectEvents(); | ||||
| @@ -102,14 +101,17 @@ ConfigureMotionTouch::ConfigureMotionTouch(QWidget* parent, | ||||
| ConfigureMotionTouch::~ConfigureMotionTouch() = default; | ||||
|  | ||||
| void ConfigureMotionTouch::SetConfiguration() { | ||||
|     const Common::ParamPackage motion_param(Settings::values.motion_device.GetValue()); | ||||
|     const Common::ParamPackage touch_param(Settings::values.touch_device.GetValue()); | ||||
|  | ||||
|     ui->touch_from_button_checkbox->setChecked(Settings::values.use_touch_from_button.GetValue()); | ||||
|     touch_from_button_maps = Settings::values.touch_from_button_maps; | ||||
|     for (const auto& touch_map : touch_from_button_maps) { | ||||
|         ui->touch_from_button_map->addItem(QString::fromStdString(touch_map.name)); | ||||
|     } | ||||
|     ui->touch_from_button_map->setCurrentIndex( | ||||
|         Settings::values.touch_from_button_map_index.GetValue()); | ||||
|     ui->motion_sensitivity->setValue(motion_param.Get("sensitivity", 0.01f)); | ||||
|  | ||||
|     min_x = touch_param.Get("min_x", 100); | ||||
|     min_y = touch_param.Get("min_y", 50); | ||||
| @@ -137,6 +139,9 @@ void ConfigureMotionTouch::SetConfiguration() { | ||||
| void ConfigureMotionTouch::UpdateUiDisplay() { | ||||
|     const QString cemuhook_udp = QStringLiteral("cemuhookudp"); | ||||
|  | ||||
|     ui->motion_sensitivity_label->setVisible(true); | ||||
|     ui->motion_sensitivity->setVisible(true); | ||||
|  | ||||
|     ui->touch_calibration->setVisible(true); | ||||
|     ui->touch_calibration_config->setVisible(true); | ||||
|     ui->touch_calibration_label->setVisible(true); | ||||
| @@ -307,6 +312,7 @@ void ConfigureMotionTouch::ApplyConfiguration() { | ||||
|     touch_param.Set("max_y", max_y); | ||||
|  | ||||
|     Settings::values.touch_device = touch_param.Serialize(); | ||||
|     Settings::values.use_touch_from_button = ui->touch_from_button_checkbox->isChecked(); | ||||
|     Settings::values.touch_from_button_map_index = ui->touch_from_button_map->currentIndex(); | ||||
|     Settings::values.touch_from_button_maps = touch_from_button_maps; | ||||
|     Settings::values.udp_input_servers = GetUDPServerString(); | ||||
|   | ||||
| @@ -2,6 +2,14 @@ | ||||
| <ui version="4.0"> | ||||
|  <class>ConfigureMotionTouch</class> | ||||
|  <widget class="QDialog" name="ConfigureMotionTouch"> | ||||
|   <property name="geometry"> | ||||
|    <rect> | ||||
|     <x>0</x> | ||||
|     <y>0</y> | ||||
|     <width>500</width> | ||||
|     <height>482</height> | ||||
|    </rect> | ||||
|   </property> | ||||
|   <property name="windowTitle"> | ||||
|    <string>Configure Motion / Touch</string> | ||||
|   </property> | ||||
| @@ -9,6 +17,48 @@ | ||||
|    <string notr="true"/> | ||||
|   </property> | ||||
|   <layout class="QVBoxLayout"> | ||||
|    <item> | ||||
|     <widget class="QGroupBox" name="motion_group_box"> | ||||
|      <property name="title"> | ||||
|       <string>Mouse Motion</string> | ||||
|      </property> | ||||
|      <layout class="QVBoxLayout"> | ||||
|       <item> | ||||
|        <layout class="QHBoxLayout"> | ||||
|         <item> | ||||
|          <widget class="QLabel" name="motion_sensitivity_label"> | ||||
|           <property name="text"> | ||||
|            <string>Sensitivity:</string> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item> | ||||
|          <widget class="QDoubleSpinBox" name="motion_sensitivity"> | ||||
|           <property name="alignment"> | ||||
|            <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set> | ||||
|           </property> | ||||
|           <property name="decimals"> | ||||
|            <number>4</number> | ||||
|           </property> | ||||
|           <property name="minimum"> | ||||
|            <double>0.010000000000000</double> | ||||
|           </property> | ||||
|           <property name="maximum"> | ||||
|            <double>10.000000000000000</double> | ||||
|           </property> | ||||
|           <property name="singleStep"> | ||||
|            <double>0.001000000000000</double> | ||||
|           </property> | ||||
|           <property name="value"> | ||||
|            <double>0.010000000000000</double> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|        </layout> | ||||
|       </item> | ||||
|      </layout> | ||||
|     </widget> | ||||
|    </item> | ||||
|    <item> | ||||
|     <widget class="QGroupBox" name="touch_group_box"> | ||||
|      <property name="title"> | ||||
| @@ -51,13 +101,19 @@ | ||||
|       </item> | ||||
|       <item> | ||||
|        <layout class="QHBoxLayout"> | ||||
|          <item> | ||||
|            <widget class="QLabel" name="touch_from_button_label"> | ||||
|              <property name="text"> | ||||
|                <string>Touch from button profile:</string> | ||||
|              </property> | ||||
|            </widget> | ||||
|          </item> | ||||
|         <item> | ||||
|          <widget class="QCheckBox" name="touch_from_button_checkbox"> | ||||
|           <property name="sizePolicy"> | ||||
|            <sizepolicy hsizetype="Maximum" vsizetype="Fixed"> | ||||
|             <horstretch>0</horstretch> | ||||
|             <verstretch>0</verstretch> | ||||
|            </sizepolicy> | ||||
|           </property> | ||||
|           <property name="text"> | ||||
|            <string>Use button mapping:</string> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item> | ||||
|          <widget class="QComboBox" name="touch_from_button_map"/> | ||||
|         </item> | ||||
|   | ||||
| @@ -11,11 +11,8 @@ | ||||
|  | ||||
| #include "common/assert.h" | ||||
| #include "common/param_package.h" | ||||
| #include "input_common/drivers/keyboard.h" | ||||
| #include "input_common/drivers/mouse.h" | ||||
| #include "input_common/main.h" | ||||
| #include "ui_configure_mouse_advanced.h" | ||||
| #include "yuzu/bootmanager.h" | ||||
| #include "yuzu/configuration/config.h" | ||||
| #include "yuzu/configuration/configure_mouse_advanced.h" | ||||
|  | ||||
| @@ -104,7 +101,7 @@ ConfigureMouseAdvanced::ConfigureMouseAdvanced(QWidget* parent, | ||||
|                 [=, this](const Common::ParamPackage& params) { | ||||
|                     buttons_param[button_id] = params; | ||||
|                 }, | ||||
|                 InputCommon::Polling::InputType::Button); | ||||
|                 InputCommon::Polling::DeviceType::Button); | ||||
|         }); | ||||
|         connect(button, &QPushButton::customContextMenuRequested, | ||||
|                 [=, this](const QPoint& menu_location) { | ||||
| @@ -130,10 +127,13 @@ ConfigureMouseAdvanced::ConfigureMouseAdvanced(QWidget* parent, | ||||
|     connect(timeout_timer.get(), &QTimer::timeout, [this] { SetPollingResult({}, true); }); | ||||
|  | ||||
|     connect(poll_timer.get(), &QTimer::timeout, [this] { | ||||
|         const auto& params = input_subsystem->GetNextInput(); | ||||
|         if (params.Has("engine")) { | ||||
|             SetPollingResult(params, false); | ||||
|             return; | ||||
|         Common::ParamPackage params; | ||||
|         for (auto& poller : device_pollers) { | ||||
|             params = poller->GetNextInput(); | ||||
|             if (params.Has("engine")) { | ||||
|                 SetPollingResult(params, false); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     }); | ||||
|  | ||||
| @@ -196,13 +196,26 @@ void ConfigureMouseAdvanced::UpdateButtonLabels() { | ||||
|  | ||||
| void ConfigureMouseAdvanced::HandleClick( | ||||
|     QPushButton* button, std::function<void(const Common::ParamPackage&)> new_input_setter, | ||||
|     InputCommon::Polling::InputType type) { | ||||
|     InputCommon::Polling::DeviceType type) { | ||||
|     button->setText(tr("[press key]")); | ||||
|     button->setFocus(); | ||||
|  | ||||
|     // Keyboard keys or mouse buttons can only be used as button devices | ||||
|     want_keyboard_mouse = type == InputCommon::Polling::DeviceType::Button; | ||||
|     if (want_keyboard_mouse) { | ||||
|         const auto iter = std::find(button_map.begin(), button_map.end(), button); | ||||
|         ASSERT(iter != button_map.end()); | ||||
|         const auto index = std::distance(button_map.begin(), iter); | ||||
|         ASSERT(index < Settings::NativeButton::NumButtons && index >= 0); | ||||
|     } | ||||
|  | ||||
|     input_setter = new_input_setter; | ||||
|  | ||||
|     input_subsystem->BeginMapping(type); | ||||
|     device_pollers = input_subsystem->GetPollers(type); | ||||
|  | ||||
|     for (auto& poller : device_pollers) { | ||||
|         poller->Start(); | ||||
|     } | ||||
|  | ||||
|     QWidget::grabMouse(); | ||||
|     QWidget::grabKeyboard(); | ||||
| @@ -214,7 +227,9 @@ void ConfigureMouseAdvanced::HandleClick( | ||||
| void ConfigureMouseAdvanced::SetPollingResult(const Common::ParamPackage& params, bool abort) { | ||||
|     timeout_timer->stop(); | ||||
|     poll_timer->stop(); | ||||
|     input_subsystem->StopMapping(); | ||||
|     for (auto& poller : device_pollers) { | ||||
|         poller->Stop(); | ||||
|     } | ||||
|  | ||||
|     QWidget::releaseMouse(); | ||||
|     QWidget::releaseKeyboard(); | ||||
| @@ -232,8 +247,15 @@ void ConfigureMouseAdvanced::mousePressEvent(QMouseEvent* event) { | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     const auto button = GRenderWindow::QtButtonToMouseButton(event->button()); | ||||
|     input_subsystem->GetMouse()->PressButton(0, 0, 0, 0, button); | ||||
|     if (want_keyboard_mouse) { | ||||
|         SetPollingResult(Common::ParamPackage{InputCommon::GenerateKeyboardParam(event->button())}, | ||||
|                          false); | ||||
|     } else { | ||||
|         // We don't want any mouse buttons, so don't stop polling | ||||
|         return; | ||||
|     } | ||||
|  | ||||
|     SetPollingResult({}, true); | ||||
| } | ||||
|  | ||||
| void ConfigureMouseAdvanced::keyPressEvent(QKeyEvent* event) { | ||||
| @@ -242,6 +264,13 @@ void ConfigureMouseAdvanced::keyPressEvent(QKeyEvent* event) { | ||||
|     } | ||||
|  | ||||
|     if (event->key() != Qt::Key_Escape) { | ||||
|         input_subsystem->GetKeyboard()->PressKey(event->key()); | ||||
|         if (want_keyboard_mouse) { | ||||
|             SetPollingResult(Common::ParamPackage{InputCommon::GenerateKeyboardParam(event->key())}, | ||||
|                              false); | ||||
|         } else { | ||||
|             // Escape key wasn't pressed and we don't want any keyboard keys, so don't stop polling | ||||
|             return; | ||||
|         } | ||||
|     } | ||||
|     SetPollingResult({}, true); | ||||
| } | ||||
|   | ||||
| @@ -46,7 +46,7 @@ private: | ||||
|     /// Called when the button was pressed. | ||||
|     void HandleClick(QPushButton* button, | ||||
|                      std::function<void(const Common::ParamPackage&)> new_input_setter, | ||||
|                      InputCommon::Polling::InputType type); | ||||
|                      InputCommon::Polling::DeviceType type); | ||||
|  | ||||
|     /// Finish polling and configure input using the input_setter | ||||
|     void SetPollingResult(const Common::ParamPackage& params, bool abort); | ||||
| @@ -67,6 +67,12 @@ private: | ||||
|     std::array<QPushButton*, Settings::NativeMouseButton::NumMouseButtons> button_map; | ||||
|     std::array<Common::ParamPackage, Settings::NativeMouseButton::NumMouseButtons> buttons_param; | ||||
|  | ||||
|     std::vector<std::unique_ptr<InputCommon::Polling::DevicePoller>> device_pollers; | ||||
|  | ||||
|     std::unique_ptr<QTimer> timeout_timer; | ||||
|     std::unique_ptr<QTimer> poll_timer; | ||||
|  | ||||
|     /// A flag to indicate if keyboard keys are okay when configuring an input. If this is false, | ||||
|     /// keyboard events are ignored. | ||||
|     bool want_keyboard_mouse = false; | ||||
| }; | ||||
|   | ||||
| @@ -32,6 +32,7 @@ void ConfigureTasDialog::LoadConfiguration() { | ||||
|     ui->tas_path_edit->setText( | ||||
|         QString::fromStdString(Common::FS::GetYuzuPathString(Common::FS::YuzuPath::TASDir))); | ||||
|     ui->tas_enable->setChecked(Settings::values.tas_enable.GetValue()); | ||||
|     ui->tas_control_swap->setChecked(Settings::values.tas_swap_controllers.GetValue()); | ||||
|     ui->tas_loop_script->setChecked(Settings::values.tas_loop.GetValue()); | ||||
|     ui->tas_pause_on_load->setChecked(Settings::values.pause_tas_on_load.GetValue()); | ||||
| } | ||||
| @@ -39,6 +40,7 @@ void ConfigureTasDialog::LoadConfiguration() { | ||||
| void ConfigureTasDialog::ApplyConfiguration() { | ||||
|     Common::FS::SetYuzuPath(Common::FS::YuzuPath::TASDir, ui->tas_path_edit->text().toStdString()); | ||||
|     Settings::values.tas_enable.SetValue(ui->tas_enable->isChecked()); | ||||
|     Settings::values.tas_swap_controllers.SetValue(ui->tas_control_swap->isChecked()); | ||||
|     Settings::values.tas_loop.SetValue(ui->tas_loop_script->isChecked()); | ||||
|     Settings::values.pause_tas_on_load.SetValue(ui->tas_pause_on_load->isChecked()); | ||||
| } | ||||
|   | ||||
| @@ -59,13 +59,20 @@ | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item row="1" column="0" colspan="4"> | ||||
|          <widget class="QCheckBox" name="tas_control_swap"> | ||||
|           <property name="text"> | ||||
|            <string>Automatic controller profile swapping</string> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item row="2" column="0" colspan="4"> | ||||
|          <widget class="QCheckBox" name="tas_loop_script"> | ||||
|           <property name="text"> | ||||
|            <string>Loop script</string> | ||||
|           </property> | ||||
|          </widget> | ||||
|         </item> | ||||
|         <item row="2" column="0" colspan="4"> | ||||
|         <item row="3" column="0" colspan="4"> | ||||
|          <widget class="QCheckBox" name="tas_pause_on_load"> | ||||
|           <property name="enabled"> | ||||
|            <bool>false</bool> | ||||
|   | ||||
| @@ -163,10 +163,13 @@ void ConfigureTouchFromButton::ConnectEvents() { | ||||
|     connect(timeout_timer.get(), &QTimer::timeout, [this]() { SetPollingResult({}, true); }); | ||||
|  | ||||
|     connect(poll_timer.get(), &QTimer::timeout, [this]() { | ||||
|         const auto& params = input_subsystem->GetNextInput(); | ||||
|         if (params.Has("engine")) { | ||||
|             SetPollingResult(params, false); | ||||
|             return; | ||||
|         Common::ParamPackage params; | ||||
|         for (auto& poller : device_pollers) { | ||||
|             params = poller->GetNextInput(); | ||||
|             if (params.Has("engine")) { | ||||
|                 SetPollingResult(params, false); | ||||
|                 return; | ||||
|             } | ||||
|         } | ||||
|     }); | ||||
| } | ||||
| @@ -245,7 +248,11 @@ void ConfigureTouchFromButton::GetButtonInput(const int row_index, const bool is | ||||
|         } | ||||
|     }; | ||||
|  | ||||
|     input_subsystem->BeginMapping(InputCommon::Polling::InputType::Button); | ||||
|     device_pollers = input_subsystem->GetPollers(InputCommon::Polling::DeviceType::Button); | ||||
|  | ||||
|     for (auto& poller : device_pollers) { | ||||
|         poller->Start(); | ||||
|     } | ||||
|  | ||||
|     grabKeyboard(); | ||||
|     grabMouse(); | ||||
| @@ -358,14 +365,14 @@ void ConfigureTouchFromButton::SetCoordinates(const int dot_id, const QPoint& po | ||||
|  | ||||
| void ConfigureTouchFromButton::SetPollingResult(const Common::ParamPackage& params, | ||||
|                                                 const bool cancel) { | ||||
|     timeout_timer->stop(); | ||||
|     poll_timer->stop(); | ||||
|     input_subsystem->StopMapping(); | ||||
|  | ||||
|     releaseKeyboard(); | ||||
|     releaseMouse(); | ||||
|     qApp->restoreOverrideCursor(); | ||||
|  | ||||
|     timeout_timer->stop(); | ||||
|     poll_timer->stop(); | ||||
|     for (auto& poller : device_pollers) { | ||||
|         poller->Stop(); | ||||
|     } | ||||
|     if (input_setter) { | ||||
|         (*input_setter)(params, cancel); | ||||
|         input_setter.reset(); | ||||
|   | ||||
| @@ -24,6 +24,10 @@ namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
|  | ||||
| namespace InputCommon::Polling { | ||||
| class DevicePoller; | ||||
| } | ||||
|  | ||||
| namespace Settings { | ||||
| struct TouchFromButtonMap; | ||||
| } | ||||
| @@ -81,6 +85,7 @@ private: | ||||
|  | ||||
|     std::unique_ptr<QTimer> timeout_timer; | ||||
|     std::unique_ptr<QTimer> poll_timer; | ||||
|     std::vector<std::unique_ptr<InputCommon::Polling::DevicePoller>> device_pollers; | ||||
|     std::optional<std::function<void(const Common::ParamPackage&, bool)>> input_setter; | ||||
|  | ||||
|     static constexpr int DataRoleDot = Qt::ItemDataRole::UserRole + 2; | ||||
|   | ||||
| @@ -97,7 +97,7 @@ void ConfigureVibration::SetVibrationDevices(std::size_t player_index) { | ||||
|  | ||||
|         const auto engine = param.Get("engine", ""); | ||||
|         const auto guid = param.Get("guid", ""); | ||||
|         const auto port = param.Get("port", 0); | ||||
|         const auto port = param.Get("port", ""); | ||||
|  | ||||
|         if (engine.empty() || engine == "keyboard" || engine == "mouse" || engine == "tas") { | ||||
|             continue; | ||||
| @@ -105,7 +105,7 @@ void ConfigureVibration::SetVibrationDevices(std::size_t player_index) { | ||||
|  | ||||
|         vibration_param_str += fmt::format("engine:{}", engine); | ||||
|  | ||||
|         if (port != 0) { | ||||
|         if (!port.empty()) { | ||||
|             vibration_param_str += fmt::format(",port:{}", port); | ||||
|         } | ||||
|         if (!guid.empty()) { | ||||
|   | ||||
| @@ -6,17 +6,13 @@ | ||||
| #include <QLayout> | ||||
| #include <QString> | ||||
| #include "common/settings.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "input_common/drivers/tas_input.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/tas/tas_input.h" | ||||
| #include "yuzu/configuration/configure_input_player_widget.h" | ||||
| #include "yuzu/debugger/controller.h" | ||||
|  | ||||
| ControllerDialog::ControllerDialog(Core::HID::HIDCore& hid_core_, | ||||
|                                    std::shared_ptr<InputCommon::InputSubsystem> input_subsystem_, | ||||
|                                    QWidget* parent) | ||||
|     : QWidget(parent, Qt::Dialog), hid_core{hid_core_}, input_subsystem{input_subsystem_} { | ||||
| ControllerDialog::ControllerDialog(QWidget* parent, InputCommon::InputSubsystem* input_subsystem_) | ||||
|     : QWidget(parent, Qt::Dialog), input_subsystem{input_subsystem_} { | ||||
|     setObjectName(QStringLiteral("Controller")); | ||||
|     setWindowTitle(tr("Controller P1")); | ||||
|     resize(500, 350); | ||||
| @@ -35,24 +31,20 @@ ControllerDialog::ControllerDialog(Core::HID::HIDCore& hid_core_, | ||||
|     // Configure focus so that widget is focusable and the dialog automatically forwards focus to | ||||
|     // it. | ||||
|     setFocusProxy(widget); | ||||
|     widget->SetConnectedStatus(false); | ||||
|     widget->setFocusPolicy(Qt::StrongFocus); | ||||
|     widget->setFocus(); | ||||
| } | ||||
|  | ||||
| void ControllerDialog::refreshConfiguration() { | ||||
|     UnloadController(); | ||||
|     auto* player_1 = hid_core.GetEmulatedController(Core::HID::NpadIdType::Player1); | ||||
|     auto* handheld = hid_core.GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|     // Display the correct controller | ||||
|     controller = handheld->IsConnected() ? handheld : player_1; | ||||
|  | ||||
|     Core::HID::ControllerUpdateCallback engine_callback{ | ||||
|         .on_change = [this](Core::HID::ControllerTriggerType type) { ControllerUpdate(type); }, | ||||
|         .is_npad_service = true, | ||||
|     }; | ||||
|     callback_key = controller->SetCallback(engine_callback); | ||||
|     widget->SetController(controller); | ||||
|     is_controller_set = true; | ||||
|     const auto& players = Settings::values.players.GetValue(); | ||||
|     constexpr std::size_t player = 0; | ||||
|     widget->SetPlayerInputRaw(player, players[player].buttons, players[player].analogs); | ||||
|     widget->SetControllerType(players[player].controller_type); | ||||
|     ControllerCallback callback{[this](ControllerInput input) { InputController(input); }}; | ||||
|     widget->SetCallBack(callback); | ||||
|     widget->repaint(); | ||||
|     widget->SetConnectedStatus(players[player].connected); | ||||
| } | ||||
|  | ||||
| QAction* ControllerDialog::toggleViewAction() { | ||||
| @@ -66,18 +58,11 @@ QAction* ControllerDialog::toggleViewAction() { | ||||
|     return toggle_view_action; | ||||
| } | ||||
|  | ||||
| void ControllerDialog::UnloadController() { | ||||
|     widget->UnloadController(); | ||||
|     if (is_controller_set) { | ||||
|         controller->DeleteCallback(callback_key); | ||||
|         is_controller_set = false; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void ControllerDialog::showEvent(QShowEvent* ev) { | ||||
|     if (toggle_view_action) { | ||||
|         toggle_view_action->setChecked(isVisible()); | ||||
|     } | ||||
|     refreshConfiguration(); | ||||
|     QWidget::showEvent(ev); | ||||
| } | ||||
|  | ||||
| @@ -85,34 +70,16 @@ void ControllerDialog::hideEvent(QHideEvent* ev) { | ||||
|     if (toggle_view_action) { | ||||
|         toggle_view_action->setChecked(isVisible()); | ||||
|     } | ||||
|     widget->SetConnectedStatus(false); | ||||
|     QWidget::hideEvent(ev); | ||||
| } | ||||
|  | ||||
| void ControllerDialog::ControllerUpdate(Core::HID::ControllerTriggerType type) { | ||||
|     // TODO(german77): Remove TAS from here | ||||
|     switch (type) { | ||||
|     case Core::HID::ControllerTriggerType::Button: | ||||
|     case Core::HID::ControllerTriggerType::Stick: { | ||||
|         const auto buttons_values = controller->GetButtonsValues(); | ||||
|         const auto stick_values = controller->GetSticksValues(); | ||||
|         u64 buttons = 0; | ||||
|         std::size_t index = 0; | ||||
|         for (const auto& button : buttons_values) { | ||||
|             buttons |= button.value ? 1LLU << index : 0; | ||||
|             index++; | ||||
|         } | ||||
|         const InputCommon::TasInput::TasAnalog left_axis = { | ||||
|             .x = stick_values[Settings::NativeAnalog::LStick].x.value, | ||||
|             .y = stick_values[Settings::NativeAnalog::LStick].y.value, | ||||
|         }; | ||||
|         const InputCommon::TasInput::TasAnalog right_axis = { | ||||
|             .x = stick_values[Settings::NativeAnalog::RStick].x.value, | ||||
|             .y = stick_values[Settings::NativeAnalog::RStick].y.value, | ||||
|         }; | ||||
|         input_subsystem->GetTas()->RecordInput(buttons, left_axis, right_axis); | ||||
|         break; | ||||
|     } | ||||
|     default: | ||||
|         break; | ||||
| void ControllerDialog::InputController(ControllerInput input) { | ||||
|     u32 buttons = 0; | ||||
|     int index = 0; | ||||
|     for (bool btn : input.button_values) { | ||||
|         buttons |= (btn ? 1U : 0U) << index; | ||||
|         index++; | ||||
|     } | ||||
|     input_subsystem->GetTas()->RecordInput(buttons, input.axis_values); | ||||
| } | ||||
|   | ||||
| @@ -4,7 +4,9 @@ | ||||
|  | ||||
| #pragma once | ||||
|  | ||||
| #include <QFileSystemWatcher> | ||||
| #include <QWidget> | ||||
| #include "common/settings.h" | ||||
|  | ||||
| class QAction; | ||||
| class QHideEvent; | ||||
| @@ -15,43 +17,35 @@ namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| class HIDCore; | ||||
| class EmulatedController; | ||||
| enum class ControllerTriggerType; | ||||
| } // namespace Core::HID | ||||
| struct ControllerInput { | ||||
|     std::array<std::pair<float, float>, Settings::NativeAnalog::NUM_STICKS_HID> axis_values{}; | ||||
|     std::array<bool, Settings::NativeButton::NumButtons> button_values{}; | ||||
|     bool changed{}; | ||||
| }; | ||||
|  | ||||
| struct ControllerCallback { | ||||
|     std::function<void(ControllerInput)> input; | ||||
| }; | ||||
|  | ||||
| class ControllerDialog : public QWidget { | ||||
|     Q_OBJECT | ||||
|  | ||||
| public: | ||||
|     explicit ControllerDialog(Core::HID::HIDCore& hid_core_, | ||||
|                               std::shared_ptr<InputCommon::InputSubsystem> input_subsystem_, | ||||
|                               QWidget* parent = nullptr); | ||||
|     explicit ControllerDialog(QWidget* parent = nullptr, | ||||
|                               InputCommon::InputSubsystem* input_subsystem_ = nullptr); | ||||
|  | ||||
|     /// Returns a QAction that can be used to toggle visibility of this dialog. | ||||
|     QAction* toggleViewAction(); | ||||
|  | ||||
|     /// Reloads the widget to apply any changes in the configuration | ||||
|     void refreshConfiguration(); | ||||
|  | ||||
|     /// Disables events from the emulated controller | ||||
|     void UnloadController(); | ||||
|  | ||||
| protected: | ||||
|     void showEvent(QShowEvent* ev) override; | ||||
|     void hideEvent(QHideEvent* ev) override; | ||||
|  | ||||
| private: | ||||
|     /// Redirects input from the widget to the TAS driver | ||||
|     void ControllerUpdate(Core::HID::ControllerTriggerType type); | ||||
|  | ||||
|     int callback_key; | ||||
|     bool is_controller_set{}; | ||||
|     Core::HID::EmulatedController* controller; | ||||
|  | ||||
|     void InputController(ControllerInput input); | ||||
|     QAction* toggle_view_action = nullptr; | ||||
|     QFileSystemWatcher* watcher = nullptr; | ||||
|     PlayerControlPreview* widget; | ||||
|     Core::HID::HIDCore& hid_core; | ||||
|     std::shared_ptr<InputCommon::InputSubsystem> input_subsystem; | ||||
|     InputCommon::InputSubsystem* input_subsystem; | ||||
| }; | ||||
|   | ||||
| @@ -26,8 +26,6 @@ | ||||
| #include "core/frontend/applets/controller.h" | ||||
| #include "core/frontend/applets/general_frontend.h" | ||||
| #include "core/frontend/applets/software_keyboard.h" | ||||
| #include "core/hid/emulated_controller.h" | ||||
| #include "core/hid/hid_core.h" | ||||
| #include "core/hle/service/acc/profile_manager.h" | ||||
| #include "core/hle/service/am/applet_ae.h" | ||||
| #include "core/hle/service/am/applet_oe.h" | ||||
| @@ -108,8 +106,8 @@ static FileSys::VirtualFile VfsDirectoryCreateFileWrapper(const FileSys::Virtual | ||||
| #include "core/loader/loader.h" | ||||
| #include "core/perf_stats.h" | ||||
| #include "core/telemetry_session.h" | ||||
| #include "input_common/drivers/tas_input.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/tas/tas_input.h" | ||||
| #include "ui_main.h" | ||||
| #include "util/overlay_dialog.h" | ||||
| #include "video_core/gpu.h" | ||||
| @@ -229,9 +227,6 @@ GMainWindow::GMainWindow() | ||||
|     ConnectMenuEvents(); | ||||
|     ConnectWidgetEvents(); | ||||
|  | ||||
|     system->HIDCore().ReloadInputDevices(); | ||||
|     controller_dialog->refreshConfiguration(); | ||||
|  | ||||
|     const auto branch_name = std::string(Common::g_scm_branch); | ||||
|     const auto description = std::string(Common::g_scm_desc); | ||||
|     const auto build_id = std::string(Common::g_build_id); | ||||
| @@ -834,16 +829,15 @@ void GMainWindow::InitializeWidgets() { | ||||
|     dock_status_button->setFocusPolicy(Qt::NoFocus); | ||||
|     connect(dock_status_button, &QPushButton::clicked, [&] { | ||||
|         const bool is_docked = Settings::values.use_docked_mode.GetValue(); | ||||
|         auto* player_1 = system->HIDCore().GetEmulatedController(Core::HID::NpadIdType::Player1); | ||||
|         auto* handheld = system->HIDCore().GetEmulatedController(Core::HID::NpadIdType::Handheld); | ||||
|         auto& controller_type = Settings::values.players.GetValue()[0].controller_type; | ||||
|  | ||||
|         if (!is_docked && handheld->IsConnected()) { | ||||
|         if (!is_docked && controller_type == Settings::ControllerType::Handheld) { | ||||
|             QMessageBox::warning(this, tr("Invalid config detected"), | ||||
|                                  tr("Handheld controller can't be used on docked mode. Pro " | ||||
|                                     "controller will be selected.")); | ||||
|             handheld->Disconnect(); | ||||
|             player_1->SetNpadStyleIndex(Core::HID::NpadStyleIndex::ProController); | ||||
|             player_1->Connect(); | ||||
|             controller_type = Settings::ControllerType::ProController; | ||||
|             ConfigureDialog configure_dialog(this, hotkey_registry, input_subsystem.get(), *system); | ||||
|             configure_dialog.ApplyConfiguration(); | ||||
|             controller_dialog->refreshConfiguration(); | ||||
|         } | ||||
|  | ||||
| @@ -928,7 +922,7 @@ void GMainWindow::InitializeDebugWidgets() { | ||||
|     waitTreeWidget->hide(); | ||||
|     debug_menu->addAction(waitTreeWidget->toggleViewAction()); | ||||
|  | ||||
|     controller_dialog = new ControllerDialog(system->HIDCore(), input_subsystem, this); | ||||
|     controller_dialog = new ControllerDialog(this, input_subsystem.get()); | ||||
|     controller_dialog->hide(); | ||||
|     debug_menu->addAction(controller_dialog->toggleViewAction()); | ||||
|  | ||||
| @@ -2780,6 +2774,7 @@ void GMainWindow::OnConfigure() { | ||||
|  | ||||
|         ShowTelemetryCallout(); | ||||
|     } | ||||
|     controller_dialog->refreshConfiguration(); | ||||
|     InitializeHotkeys(); | ||||
|  | ||||
|     if (UISettings::values.theme != old_theme) { | ||||
| @@ -2812,7 +2807,6 @@ void GMainWindow::OnConfigure() { | ||||
|     } | ||||
|  | ||||
|     UpdateStatusButtons(); | ||||
|     controller_dialog->refreshConfiguration(); | ||||
| } | ||||
|  | ||||
| void GMainWindow::OnConfigureTas() { | ||||
| @@ -3009,11 +3003,11 @@ void GMainWindow::UpdateWindowTitle(std::string_view title_name, std::string_vie | ||||
| QString GMainWindow::GetTasStateDescription() const { | ||||
|     auto [tas_status, current_tas_frame, total_tas_frames] = input_subsystem->GetTas()->GetStatus(); | ||||
|     switch (tas_status) { | ||||
|     case InputCommon::TasInput::TasState::Running: | ||||
|     case TasInput::TasState::Running: | ||||
|         return tr("TAS state: Running %1/%2").arg(current_tas_frame).arg(total_tas_frames); | ||||
|     case InputCommon::TasInput::TasState::Recording: | ||||
|     case TasInput::TasState::Recording: | ||||
|         return tr("TAS state: Recording %1").arg(total_tas_frames); | ||||
|     case InputCommon::TasInput::TasState::Stopped: | ||||
|     case TasInput::TasState::Stopped: | ||||
|         return tr("TAS state: Idle %1/%2").arg(current_tas_frame).arg(total_tas_frames); | ||||
|     default: | ||||
|         return tr("TAS State: Invalid"); | ||||
| @@ -3392,8 +3386,6 @@ void GMainWindow::closeEvent(QCloseEvent* event) { | ||||
|     UpdateUISettings(); | ||||
|     game_list->SaveInterfaceLayout(); | ||||
|     hotkey_registry.SaveHotkeys(); | ||||
|     controller_dialog->UnloadController(); | ||||
|     system->HIDCore().UnloadInputDevices(); | ||||
|  | ||||
|     // Shutdown session if the emu thread is active... | ||||
|     if (emu_thread != nullptr) { | ||||
|   | ||||
| @@ -6,8 +6,7 @@ | ||||
| #include <QScreen> | ||||
|  | ||||
| #include "core/core.h" | ||||
| #include "core/hid/hid_types.h" | ||||
| #include "core/hid/input_interpreter.h" | ||||
| #include "core/frontend/input_interpreter.h" | ||||
| #include "ui_overlay_dialog.h" | ||||
| #include "yuzu/util/overlay_dialog.h" | ||||
|  | ||||
| @@ -180,9 +179,9 @@ void OverlayDialog::MoveAndResizeWindow() { | ||||
|     QDialog::resize(width, height); | ||||
| } | ||||
|  | ||||
| template <Core::HID::NpadButton... T> | ||||
| template <HIDButton... T> | ||||
| void OverlayDialog::HandleButtonPressedOnce() { | ||||
|     const auto f = [this](Core::HID::NpadButton button) { | ||||
|     const auto f = [this](HIDButton button) { | ||||
|         if (input_interpreter->IsButtonPressedOnce(button)) { | ||||
|             TranslateButtonPress(button); | ||||
|         } | ||||
| @@ -191,7 +190,7 @@ void OverlayDialog::HandleButtonPressedOnce() { | ||||
|     (f(T), ...); | ||||
| } | ||||
|  | ||||
| void OverlayDialog::TranslateButtonPress(Core::HID::NpadButton button) { | ||||
| void OverlayDialog::TranslateButtonPress(HIDButton button) { | ||||
|     QPushButton* left_button = use_rich_text ? ui->button_cancel_rich : ui->button_cancel; | ||||
|     QPushButton* right_button = use_rich_text ? ui->button_ok_rich : ui->button_ok_label; | ||||
|  | ||||
| @@ -199,20 +198,20 @@ void OverlayDialog::TranslateButtonPress(Core::HID::NpadButton button) { | ||||
|     // TODO (Morph): focusPrevious/NextChild() doesn't work well with the rich text dialog, fix it | ||||
|  | ||||
|     switch (button) { | ||||
|     case Core::HID::NpadButton::A: | ||||
|     case Core::HID::NpadButton::B: | ||||
|     case HIDButton::A: | ||||
|     case HIDButton::B: | ||||
|         if (left_button->hasFocus()) { | ||||
|             left_button->click(); | ||||
|         } else if (right_button->hasFocus()) { | ||||
|             right_button->click(); | ||||
|         } | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Left: | ||||
|     case Core::HID::NpadButton::StickLLeft: | ||||
|     case HIDButton::DLeft: | ||||
|     case HIDButton::LStickLeft: | ||||
|         focusPreviousChild(); | ||||
|         break; | ||||
|     case Core::HID::NpadButton::Right: | ||||
|     case Core::HID::NpadButton::StickLRight: | ||||
|     case HIDButton::DRight: | ||||
|     case HIDButton::LStickRight: | ||||
|         focusNextChild(); | ||||
|         break; | ||||
|     default: | ||||
| @@ -242,10 +241,8 @@ void OverlayDialog::InputThread() { | ||||
|     while (input_thread_running) { | ||||
|         input_interpreter->PollInput(); | ||||
|  | ||||
|         HandleButtonPressedOnce<Core::HID::NpadButton::A, Core::HID::NpadButton::B, | ||||
|                                 Core::HID::NpadButton::Left, Core::HID::NpadButton::Right, | ||||
|                                 Core::HID::NpadButton::StickLLeft, | ||||
|                                 Core::HID::NpadButton::StickLRight>(); | ||||
|         HandleButtonPressedOnce<HIDButton::A, HIDButton::B, HIDButton::DLeft, HIDButton::DRight, | ||||
|                                 HIDButton::LStickLeft, HIDButton::LStickRight>(); | ||||
|  | ||||
|         std::this_thread::sleep_for(std::chrono::milliseconds(50)); | ||||
|     } | ||||
|   | ||||
| @@ -13,16 +13,14 @@ | ||||
|  | ||||
| #include "common/common_types.h" | ||||
|  | ||||
| enum class HIDButton : u8; | ||||
|  | ||||
| class InputInterpreter; | ||||
|  | ||||
| namespace Core { | ||||
| class System; | ||||
| } | ||||
|  | ||||
| namespace Core::HID { | ||||
| enum class NpadButton : u64; | ||||
| } | ||||
|  | ||||
| namespace Ui { | ||||
| class OverlayDialog; | ||||
| } | ||||
| @@ -81,7 +79,7 @@ private: | ||||
|      * | ||||
|      * @tparam HIDButton The list of buttons that can be converted into keyboard input. | ||||
|      */ | ||||
|     template <Core::HID::NpadButton... T> | ||||
|     template <HIDButton... T> | ||||
|     void HandleButtonPressedOnce(); | ||||
|  | ||||
|     /** | ||||
| @@ -89,7 +87,7 @@ private: | ||||
|      * | ||||
|      * @param button The button press to process. | ||||
|      */ | ||||
|     void TranslateButtonPress(Core::HID::NpadButton button); | ||||
|     void TranslateButtonPress(HIDButton button); | ||||
|  | ||||
|     void StartInputThread(); | ||||
|     void StopInputThread(); | ||||
|   | ||||
| @@ -24,6 +24,7 @@ | ||||
| #include "common/settings.h" | ||||
| #include "core/hle/service/acc/profile_manager.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/udp/client.h" | ||||
| #include "yuzu_cmd/config.h" | ||||
| #include "yuzu_cmd/default_ini.h" | ||||
|  | ||||
| @@ -292,6 +293,8 @@ void Config::ReadValues() { | ||||
|             Settings::values.mouse_buttons[i] = default_param; | ||||
|     } | ||||
|  | ||||
|     ReadSetting("ControlsGeneral", Settings::values.motion_device); | ||||
|  | ||||
|     ReadSetting("ControlsGeneral", Settings::values.touch_device); | ||||
|  | ||||
|     ReadSetting("ControlsGeneral", Settings::values.keyboard_enabled); | ||||
| @@ -360,6 +363,7 @@ void Config::ReadValues() { | ||||
|             Settings::TouchFromButtonMap{"default", {}}); | ||||
|         num_touch_from_button_maps = 1; | ||||
|     } | ||||
|     ReadSetting("ControlsGeneral", Settings::values.use_touch_from_button); | ||||
|     Settings::values.touch_from_button_map_index = std::clamp( | ||||
|         Settings::values.touch_from_button_map_index.GetValue(), 0, num_touch_from_button_maps - 1); | ||||
|  | ||||
|   | ||||
| @@ -84,10 +84,23 @@ enable_accurate_vibrations= | ||||
| # 0: Disabled, 1 (default): Enabled | ||||
| motion_enabled = | ||||
|  | ||||
| # Defines the udp device's touch screen coordinate system for cemuhookudp devices | ||||
| #  - "min_x", "min_y", "max_x", "max_y" | ||||
| # for motion input, the following devices are available: | ||||
| #  - "motion_emu" (default) for emulating motion input from mouse input. Required parameters: | ||||
| #      - "update_period": update period in milliseconds (default to 100) | ||||
| #      - "sensitivity": the coefficient converting mouse movement to tilting angle (default to 0.01) | ||||
| #  - "cemuhookudp" reads motion input from a udp server that uses cemuhook's udp protocol | ||||
| motion_device= | ||||
|  | ||||
| # for touch input, the following devices are available: | ||||
| #  - "emu_window" (default) for emulating touch input from mouse input to the emulation window. No parameters required | ||||
| #  - "cemuhookudp" reads touch input from a udp server that uses cemuhook's udp protocol | ||||
| #      - "min_x", "min_y", "max_x", "max_y": defines the udp device's touch screen coordinate system | ||||
| touch_device= | ||||
|  | ||||
| # Whether to enable or disable touch input from button | ||||
| # 0 (default): Disabled, 1: Enabled | ||||
| use_touch_from_button= | ||||
|  | ||||
| # for mapping buttons to touch inputs. | ||||
| #touch_from_button_map=1 | ||||
| #touch_from_button_maps_0_name=default | ||||
|   | ||||
| @@ -9,10 +9,10 @@ | ||||
| #include "common/settings.h" | ||||
| #include "core/core.h" | ||||
| #include "core/perf_stats.h" | ||||
| #include "input_common/drivers/keyboard.h" | ||||
| #include "input_common/drivers/mouse.h" | ||||
| #include "input_common/drivers/touch_screen.h" | ||||
| #include "input_common/keyboard.h" | ||||
| #include "input_common/main.h" | ||||
| #include "input_common/mouse/mouse_input.h" | ||||
| #include "input_common/sdl/sdl.h" | ||||
| #include "yuzu_cmd/emu_window/emu_window_sdl2.h" | ||||
| #include "yuzu_cmd/yuzu_icon.h" | ||||
|  | ||||
| @@ -32,32 +32,42 @@ EmuWindow_SDL2::~EmuWindow_SDL2() { | ||||
| } | ||||
|  | ||||
| void EmuWindow_SDL2::OnMouseMotion(s32 x, s32 y) { | ||||
|     input_subsystem->GetMouse()->MouseMove(x, y, 0, 0, 0, 0); | ||||
|     TouchMoved((unsigned)std::max(x, 0), (unsigned)std::max(y, 0), 0); | ||||
|  | ||||
|     input_subsystem->GetMouse()->MouseMove(x, y, 0, 0); | ||||
| } | ||||
|  | ||||
| InputCommon::MouseButton EmuWindow_SDL2::SDLButtonToMouseButton(u32 button) const { | ||||
| MouseInput::MouseButton EmuWindow_SDL2::SDLButtonToMouseButton(u32 button) const { | ||||
|     switch (button) { | ||||
|     case SDL_BUTTON_LEFT: | ||||
|         return InputCommon::MouseButton::Left; | ||||
|         return MouseInput::MouseButton::Left; | ||||
|     case SDL_BUTTON_RIGHT: | ||||
|         return InputCommon::MouseButton::Right; | ||||
|         return MouseInput::MouseButton::Right; | ||||
|     case SDL_BUTTON_MIDDLE: | ||||
|         return InputCommon::MouseButton::Wheel; | ||||
|         return MouseInput::MouseButton::Wheel; | ||||
|     case SDL_BUTTON_X1: | ||||
|         return InputCommon::MouseButton::Backward; | ||||
|         return MouseInput::MouseButton::Backward; | ||||
|     case SDL_BUTTON_X2: | ||||
|         return InputCommon::MouseButton::Forward; | ||||
|         return MouseInput::MouseButton::Forward; | ||||
|     default: | ||||
|         return InputCommon::MouseButton::Undefined; | ||||
|         return MouseInput::MouseButton::Undefined; | ||||
|     } | ||||
| } | ||||
|  | ||||
| void EmuWindow_SDL2::OnMouseButton(u32 button, u8 state, s32 x, s32 y) { | ||||
|     const auto mouse_button = SDLButtonToMouseButton(button); | ||||
|     if (state == SDL_PRESSED) { | ||||
|         input_subsystem->GetMouse()->PressButton(x, y, 0, 0, mouse_button); | ||||
|     if (button == SDL_BUTTON_LEFT) { | ||||
|         if (state == SDL_PRESSED) { | ||||
|             TouchPressed((unsigned)std::max(x, 0), (unsigned)std::max(y, 0), 0); | ||||
|         } else { | ||||
|             TouchReleased(0); | ||||
|         } | ||||
|     } else { | ||||
|         input_subsystem->GetMouse()->ReleaseButton(mouse_button); | ||||
|         if (state == SDL_PRESSED) { | ||||
|             input_subsystem->GetMouse()->PressButton(x, y, mouse_button); | ||||
|         } else { | ||||
|             input_subsystem->GetMouse()->ReleaseButton(mouse_button); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -72,35 +82,29 @@ std::pair<unsigned, unsigned> EmuWindow_SDL2::TouchToPixelPos(float touch_x, flo | ||||
|             static_cast<unsigned>(std::max(std::round(touch_y), 0.0f))}; | ||||
| } | ||||
|  | ||||
| void EmuWindow_SDL2::OnFingerDown(float x, float y, std::size_t id) { | ||||
|     int width, height; | ||||
|     SDL_GetWindowSize(render_window, &width, &height); | ||||
|     const auto [px, py] = TouchToPixelPos(x, y); | ||||
|     const float fx = px * 1.0f / width; | ||||
|     const float fy = py * 1.0f / height; | ||||
| void EmuWindow_SDL2::OnFingerDown(float x, float y) { | ||||
|     // TODO(NeatNit): keep track of multitouch using the fingerID and a dictionary of some kind | ||||
|     // This isn't critical because the best we can do when we have that is to average them, like the | ||||
|     // 3DS does | ||||
|  | ||||
|     input_subsystem->GetTouchScreen()->TouchPressed(fx, fy, id); | ||||
|     const auto [px, py] = TouchToPixelPos(x, y); | ||||
|     TouchPressed(px, py, 0); | ||||
| } | ||||
|  | ||||
| void EmuWindow_SDL2::OnFingerMotion(float x, float y, std::size_t id) { | ||||
|     int width, height; | ||||
|     SDL_GetWindowSize(render_window, &width, &height); | ||||
| void EmuWindow_SDL2::OnFingerMotion(float x, float y) { | ||||
|     const auto [px, py] = TouchToPixelPos(x, y); | ||||
|     const float fx = px * 1.0f / width; | ||||
|     const float fy = py * 1.0f / height; | ||||
|  | ||||
|     input_subsystem->GetTouchScreen()->TouchMoved(fx, fy, id); | ||||
|     TouchMoved(px, py, 0); | ||||
| } | ||||
|  | ||||
| void EmuWindow_SDL2::OnFingerUp() { | ||||
|     input_subsystem->GetTouchScreen()->TouchReleased(0); | ||||
|     TouchReleased(0); | ||||
| } | ||||
|  | ||||
| void EmuWindow_SDL2::OnKeyEvent(int key, u8 state) { | ||||
|     if (state == SDL_PRESSED) { | ||||
|         input_subsystem->GetKeyboard()->PressKey(static_cast<std::size_t>(key)); | ||||
|         input_subsystem->GetKeyboard()->PressKey(key); | ||||
|     } else if (state == SDL_RELEASED) { | ||||
|         input_subsystem->GetKeyboard()->ReleaseKey(static_cast<std::size_t>(key)); | ||||
|         input_subsystem->GetKeyboard()->ReleaseKey(key); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @@ -201,12 +205,10 @@ void EmuWindow_SDL2::WaitEvent() { | ||||
|         } | ||||
|         break; | ||||
|     case SDL_FINGERDOWN: | ||||
|         OnFingerDown(event.tfinger.x, event.tfinger.y, | ||||
|                      static_cast<std::size_t>(event.tfinger.touchId)); | ||||
|         OnFingerDown(event.tfinger.x, event.tfinger.y); | ||||
|         break; | ||||
|     case SDL_FINGERMOTION: | ||||
|         OnFingerMotion(event.tfinger.x, event.tfinger.y, | ||||
|                        static_cast<std::size_t>(event.tfinger.touchId)); | ||||
|         OnFingerMotion(event.tfinger.x, event.tfinger.y); | ||||
|         break; | ||||
|     case SDL_FINGERUP: | ||||
|         OnFingerUp(); | ||||
|   | ||||
| @@ -16,8 +16,11 @@ class System; | ||||
|  | ||||
| namespace InputCommon { | ||||
| class InputSubsystem; | ||||
| } | ||||
|  | ||||
| namespace MouseInput { | ||||
| enum class MouseButton; | ||||
| } // namespace InputCommon | ||||
| } | ||||
|  | ||||
| class EmuWindow_SDL2 : public Core::Frontend::EmuWindow { | ||||
| public: | ||||
| @@ -44,7 +47,7 @@ protected: | ||||
|     void OnMouseMotion(s32 x, s32 y); | ||||
|  | ||||
|     /// Converts a SDL mouse button into MouseInput mouse button | ||||
|     InputCommon::MouseButton SDLButtonToMouseButton(u32 button) const; | ||||
|     MouseInput::MouseButton SDLButtonToMouseButton(u32 button) const; | ||||
|  | ||||
|     /// Called by WaitEvent when a mouse button is pressed or released | ||||
|     void OnMouseButton(u32 button, u8 state, s32 x, s32 y); | ||||
| @@ -53,10 +56,10 @@ protected: | ||||
|     std::pair<unsigned, unsigned> TouchToPixelPos(float touch_x, float touch_y) const; | ||||
|  | ||||
|     /// Called by WaitEvent when a finger starts touching the touchscreen | ||||
|     void OnFingerDown(float x, float y, std::size_t id); | ||||
|     void OnFingerDown(float x, float y); | ||||
|  | ||||
|     /// Called by WaitEvent when a finger moves while touching the touchscreen | ||||
|     void OnFingerMotion(float x, float y, std::size_t id); | ||||
|     void OnFingerMotion(float x, float y); | ||||
|  | ||||
|     /// Called by WaitEvent when a finger stops touching the touchscreen | ||||
|     void OnFingerUp(); | ||||
|   | ||||
| @@ -17,6 +17,7 @@ | ||||
| #include "common/settings.h" | ||||
| #include "common/string_util.h" | ||||
| #include "core/core.h" | ||||
| #include "input_common/keyboard.h" | ||||
| #include "input_common/main.h" | ||||
| #include "video_core/renderer_base.h" | ||||
| #include "yuzu_cmd/emu_window/emu_window_sdl2_gl.h" | ||||
|   | ||||
		Reference in New Issue
	
	Block a user