Allow to invert analog axis with right click
This commit is contained in:
		
							parent
							
								
									0dc4ab42cc
								
							
						
					
					
						commit
						80fece4e08
					
				| @ -139,10 +139,10 @@ void GCButtonFactory::EndConfiguration() { | |||||||
| 
 | 
 | ||||||
| class GCAnalog final : public Input::AnalogDevice { | class GCAnalog final : public Input::AnalogDevice { | ||||||
| public: | public: | ||||||
|     explicit GCAnalog(u32 port_, u32 axis_x_, u32 axis_y_, float deadzone_, |     explicit GCAnalog(u32 port_, u32 axis_x_, u32 axis_y_, bool invert_x_, bool invert_y_, | ||||||
|                       const GCAdapter::Adapter* adapter, float range_) |                       float deadzone_, float range_, const GCAdapter::Adapter* adapter) | ||||||
|         : port(port_), axis_x(axis_x_), axis_y(axis_y_), deadzone(deadzone_), gcadapter(adapter), |         : port(port_), axis_x(axis_x_), axis_y(axis_y_), invert_x(invert_x_), invert_y(invert_y_), | ||||||
|           range(range_) {} |           deadzone(deadzone_), range(range_), gcadapter(adapter) {} | ||||||
| 
 | 
 | ||||||
|     float GetAxis(u32 axis) const { |     float GetAxis(u32 axis) const { | ||||||
|         if (gcadapter->DeviceConnected(port)) { |         if (gcadapter->DeviceConnected(port)) { | ||||||
| @ -157,7 +157,12 @@ public: | |||||||
|     std::pair<float, float> GetAnalog(u32 analog_axis_x, u32 analog_axis_y) const { |     std::pair<float, float> GetAnalog(u32 analog_axis_x, u32 analog_axis_y) const { | ||||||
|         float x = GetAxis(analog_axis_x); |         float x = GetAxis(analog_axis_x); | ||||||
|         float y = GetAxis(analog_axis_y); |         float y = GetAxis(analog_axis_y); | ||||||
| 
 |         if (invert_x) { | ||||||
|  |             x = -x; | ||||||
|  |         } | ||||||
|  |         if (invert_y) { | ||||||
|  |             y = -y; | ||||||
|  |         } | ||||||
|         // Make sure the coordinates are in the unit circle,
 |         // Make sure the coordinates are in the unit circle,
 | ||||||
|         // otherwise normalize it.
 |         // otherwise normalize it.
 | ||||||
|         float r = x * x + y * y; |         float r = x * x + y * y; | ||||||
| @ -200,9 +205,11 @@ private: | |||||||
|     const u32 port; |     const u32 port; | ||||||
|     const u32 axis_x; |     const u32 axis_x; | ||||||
|     const u32 axis_y; |     const u32 axis_y; | ||||||
|  |     const bool invert_x; | ||||||
|  |     const bool invert_y; | ||||||
|     const float deadzone; |     const float deadzone; | ||||||
|     const GCAdapter::Adapter* gcadapter; |  | ||||||
|     const float range; |     const float range; | ||||||
|  |     const GCAdapter::Adapter* gcadapter; | ||||||
|     mutable std::mutex mutex; |     mutable std::mutex mutex; | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| @ -223,8 +230,13 @@ std::unique_ptr<Input::AnalogDevice> GCAnalogFactory::Create(const Common::Param | |||||||
|     const auto axis_y = static_cast<u32>(params.Get("axis_y", 1)); |     const auto axis_y = static_cast<u32>(params.Get("axis_y", 1)); | ||||||
|     const auto deadzone = std::clamp(params.Get("deadzone", 0.0f), 0.0f, 1.0f); |     const auto deadzone = std::clamp(params.Get("deadzone", 0.0f), 0.0f, 1.0f); | ||||||
|     const auto range = std::clamp(params.Get("range", 1.0f), 0.50f, 1.50f); |     const auto range = std::clamp(params.Get("range", 1.0f), 0.50f, 1.50f); | ||||||
|  |     const std::string invert_x_value = params.Get("invert_x", "+"); | ||||||
|  |     const std::string invert_y_value = params.Get("invert_y", "+"); | ||||||
|  |     const bool invert_x = invert_x_value == "-"; | ||||||
|  |     const bool invert_y = invert_y_value == "-"; | ||||||
| 
 | 
 | ||||||
|     return std::make_unique<GCAnalog>(port, axis_x, axis_y, deadzone, adapter.get(), range); |     return std::make_unique<GCAnalog>(port, axis_x, axis_y, invert_x, invert_y, deadzone, range, | ||||||
|  |                                       adapter.get()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void GCAnalogFactory::BeginConfiguration() { | void GCAnalogFactory::BeginConfiguration() { | ||||||
| @ -282,6 +294,8 @@ Common::ParamPackage GCAnalogFactory::GetNextInput() { | |||||||
|         params.Set("port", controller_number); |         params.Set("port", controller_number); | ||||||
|         params.Set("axis_x", analog_x_axis); |         params.Set("axis_x", analog_x_axis); | ||||||
|         params.Set("axis_y", analog_y_axis); |         params.Set("axis_y", analog_y_axis); | ||||||
|  |         params.Set("invert_x", "+"); | ||||||
|  |         params.Set("invert_y", "+"); | ||||||
|         analog_x_axis = -1; |         analog_x_axis = -1; | ||||||
|         analog_y_axis = -1; |         analog_y_axis = -1; | ||||||
|         controller_number = -1; |         controller_number = -1; | ||||||
|  | |||||||
| @ -62,10 +62,10 @@ void MouseButtonFactory::EndConfiguration() { | |||||||
| 
 | 
 | ||||||
| class MouseAnalog final : public Input::AnalogDevice { | class MouseAnalog final : public Input::AnalogDevice { | ||||||
| public: | public: | ||||||
|     explicit MouseAnalog(u32 port_, u32 axis_x_, u32 axis_y_, float deadzone_, float range_, |     explicit MouseAnalog(u32 port_, u32 axis_x_, u32 axis_y_, bool invert_x_, bool invert_y_, | ||||||
|                          const MouseInput::Mouse* mouse_input_) |                          float deadzone_, float range_, const MouseInput::Mouse* mouse_input_) | ||||||
|         : button(port_), axis_x(axis_x_), axis_y(axis_y_), deadzone(deadzone_), range(range_), |         : button(port_), axis_x(axis_x_), axis_y(axis_y_), invert_x(invert_x_), invert_y(invert_y_), | ||||||
|           mouse_input(mouse_input_) {} |           deadzone(deadzone_), range(range_), mouse_input(mouse_input_) {} | ||||||
| 
 | 
 | ||||||
|     float GetAxis(u32 axis) const { |     float GetAxis(u32 axis) const { | ||||||
|         std::lock_guard lock{mutex}; |         std::lock_guard lock{mutex}; | ||||||
| @ -77,6 +77,12 @@ public: | |||||||
|     std::pair<float, float> GetAnalog(u32 analog_axis_x, u32 analog_axis_y) const { |     std::pair<float, float> GetAnalog(u32 analog_axis_x, u32 analog_axis_y) const { | ||||||
|         float x = GetAxis(analog_axis_x); |         float x = GetAxis(analog_axis_x); | ||||||
|         float y = GetAxis(analog_axis_y); |         float y = GetAxis(analog_axis_y); | ||||||
|  |         if (invert_x) { | ||||||
|  |             x = -x; | ||||||
|  |         } | ||||||
|  |         if (invert_y) { | ||||||
|  |             y = -y; | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         // Make sure the coordinates are in the unit circle,
 |         // Make sure the coordinates are in the unit circle,
 | ||||||
|         // otherwise normalize it.
 |         // otherwise normalize it.
 | ||||||
| @ -104,6 +110,8 @@ private: | |||||||
|     const u32 button; |     const u32 button; | ||||||
|     const u32 axis_x; |     const u32 axis_x; | ||||||
|     const u32 axis_y; |     const u32 axis_y; | ||||||
|  |     const bool invert_x; | ||||||
|  |     const bool invert_y; | ||||||
|     const float deadzone; |     const float deadzone; | ||||||
|     const float range; |     const float range; | ||||||
|     const MouseInput::Mouse* mouse_input; |     const MouseInput::Mouse* mouse_input; | ||||||
| @ -128,8 +136,13 @@ std::unique_ptr<Input::AnalogDevice> MouseAnalogFactory::Create( | |||||||
|     const auto axis_y = static_cast<u32>(params.Get("axis_y", 1)); |     const auto axis_y = static_cast<u32>(params.Get("axis_y", 1)); | ||||||
|     const auto deadzone = std::clamp(params.Get("deadzone", 0.0f), 0.0f, 1.0f); |     const auto deadzone = std::clamp(params.Get("deadzone", 0.0f), 0.0f, 1.0f); | ||||||
|     const auto range = std::clamp(params.Get("range", 1.0f), 0.50f, 1.50f); |     const auto range = std::clamp(params.Get("range", 1.0f), 0.50f, 1.50f); | ||||||
|  |     const std::string invert_x_value = params.Get("invert_x", "+"); | ||||||
|  |     const std::string invert_y_value = params.Get("invert_y", "+"); | ||||||
|  |     const bool invert_x = invert_x_value == "-"; | ||||||
|  |     const bool invert_y = invert_y_value == "-"; | ||||||
| 
 | 
 | ||||||
|     return std::make_unique<MouseAnalog>(port, axis_x, axis_y, deadzone, range, mouse_input.get()); |     return std::make_unique<MouseAnalog>(port, axis_x, axis_y, invert_x, invert_y, deadzone, range, | ||||||
|  |                                          mouse_input.get()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| void MouseAnalogFactory::BeginConfiguration() { | void MouseAnalogFactory::BeginConfiguration() { | ||||||
| @ -153,6 +166,8 @@ Common::ParamPackage MouseAnalogFactory::GetNextInput() const { | |||||||
|             params.Set("port", static_cast<u16>(pad.button)); |             params.Set("port", static_cast<u16>(pad.button)); | ||||||
|             params.Set("axis_x", 0); |             params.Set("axis_x", 0); | ||||||
|             params.Set("axis_y", 1); |             params.Set("axis_y", 1); | ||||||
|  |             params.Set("invert_x", "+"); | ||||||
|  |             params.Set("invert_y", "+"); | ||||||
|             return params; |             return params; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -352,13 +352,20 @@ private: | |||||||
| class SDLAnalog final : public Input::AnalogDevice { | class SDLAnalog final : public Input::AnalogDevice { | ||||||
| public: | public: | ||||||
|     explicit SDLAnalog(std::shared_ptr<SDLJoystick> joystick_, int axis_x_, int axis_y_, |     explicit SDLAnalog(std::shared_ptr<SDLJoystick> joystick_, int axis_x_, int axis_y_, | ||||||
|                        float deadzone_, float range_) |                        bool invert_x_, bool invert_y_, float deadzone_, float range_) | ||||||
|         : joystick(std::move(joystick_)), axis_x(axis_x_), axis_y(axis_y_), deadzone(deadzone_), |         : joystick(std::move(joystick_)), axis_x(axis_x_), axis_y(axis_y_), invert_x(invert_x_), | ||||||
|           range(range_) {} |           invert_y(invert_y_), deadzone(deadzone_), range(range_) {} | ||||||
| 
 | 
 | ||||||
|     std::tuple<float, float> GetStatus() const override { |     std::tuple<float, float> GetStatus() const override { | ||||||
|         const auto [x, y] = joystick->GetAnalog(axis_x, axis_y, range); |         auto [x, y] = joystick->GetAnalog(axis_x, axis_y, range); | ||||||
|         const float r = std::sqrt((x * x) + (y * y)); |         const float r = std::sqrt((x * x) + (y * y)); | ||||||
|  |         if (invert_x) { | ||||||
|  |             x = -x; | ||||||
|  |         } | ||||||
|  |         if (invert_y) { | ||||||
|  |             y = -y; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|         if (r > deadzone) { |         if (r > deadzone) { | ||||||
|             return std::make_tuple(x / r * (r - deadzone) / (1 - deadzone), |             return std::make_tuple(x / r * (r - deadzone) / (1 - deadzone), | ||||||
|                                    y / r * (r - deadzone) / (1 - deadzone)); |                                    y / r * (r - deadzone) / (1 - deadzone)); | ||||||
| @ -386,6 +393,8 @@ private: | |||||||
|     std::shared_ptr<SDLJoystick> joystick; |     std::shared_ptr<SDLJoystick> joystick; | ||||||
|     const int axis_x; |     const int axis_x; | ||||||
|     const int axis_y; |     const int axis_y; | ||||||
|  |     const bool invert_x; | ||||||
|  |     const bool invert_y; | ||||||
|     const float deadzone; |     const float deadzone; | ||||||
|     const float range; |     const float range; | ||||||
| }; | }; | ||||||
| @ -572,12 +581,17 @@ public: | |||||||
|         const int axis_y = params.Get("axis_y", 1); |         const int axis_y = params.Get("axis_y", 1); | ||||||
|         const float deadzone = std::clamp(params.Get("deadzone", 0.0f), 0.0f, 1.0f); |         const float deadzone = std::clamp(params.Get("deadzone", 0.0f), 0.0f, 1.0f); | ||||||
|         const float range = std::clamp(params.Get("range", 1.0f), 0.50f, 1.50f); |         const float range = std::clamp(params.Get("range", 1.0f), 0.50f, 1.50f); | ||||||
|  |         const std::string invert_x_value = params.Get("invert_x", "+"); | ||||||
|  |         const std::string invert_y_value = params.Get("invert_y", "+"); | ||||||
|  |         const bool invert_x = invert_x_value == "-"; | ||||||
|  |         const bool invert_y = invert_y_value == "-"; | ||||||
|         auto joystick = state.GetSDLJoystickByGUID(guid, port); |         auto joystick = state.GetSDLJoystickByGUID(guid, port); | ||||||
| 
 | 
 | ||||||
|         // This is necessary so accessing GetAxis with axis_x and axis_y won't crash
 |         // This is necessary so accessing GetAxis with axis_x and axis_y won't crash
 | ||||||
|         joystick->SetAxis(axis_x, 0); |         joystick->SetAxis(axis_x, 0); | ||||||
|         joystick->SetAxis(axis_y, 0); |         joystick->SetAxis(axis_y, 0); | ||||||
|         return std::make_unique<SDLAnalog>(joystick, axis_x, axis_y, deadzone, range); |         return std::make_unique<SDLAnalog>(joystick, axis_x, axis_y, invert_x, invert_y, deadzone, | ||||||
|  |                                            range); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
| private: | private: | ||||||
| @ -886,6 +900,8 @@ Common::ParamPackage BuildParamPackageForAnalog(int port, const std::string& gui | |||||||
|     params.Set("guid", guid); |     params.Set("guid", guid); | ||||||
|     params.Set("axis_x", axis_x); |     params.Set("axis_x", axis_x); | ||||||
|     params.Set("axis_y", axis_y); |     params.Set("axis_y", axis_y); | ||||||
|  |     params.Set("invert_x", "+"); | ||||||
|  |     params.Set("invert_y", "+"); | ||||||
|     return params; |     return params; | ||||||
| } | } | ||||||
| } // Anonymous namespace
 | } // Anonymous namespace
 | ||||||
|  | |||||||
| @ -173,61 +173,31 @@ QString AnalogToText(const Common::ParamPackage& param, const std::string& dir) | |||||||
|         return ButtonToText(Common::ParamPackage{param.Get(dir, "")}); |         return ButtonToText(Common::ParamPackage{param.Get(dir, "")}); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     if (param.Get("engine", "") == "sdl") { |     const auto engine_str = param.Get("engine", ""); | ||||||
|  |     const QString axis_x_str = QString::fromStdString(param.Get("axis_x", "")); | ||||||
|  |     const QString axis_y_str = QString::fromStdString(param.Get("axis_y", "")); | ||||||
|  |     const bool invert_x = param.Get("invert_x", "+") == "-"; | ||||||
|  |     const bool invert_y = param.Get("invert_y", "+") == "-"; | ||||||
|  |     if (engine_str == "sdl" || engine_str == "gcpad" || engine_str == "mouse") { | ||||||
|         if (dir == "modifier") { |         if (dir == "modifier") { | ||||||
|             return QObject::tr("[unused]"); |             return QObject::tr("[unused]"); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         if (dir == "left" || dir == "right") { |         if (dir == "left") { | ||||||
|             const QString axis_x_str = QString::fromStdString(param.Get("axis_x", "")); |             const QString invert_x_str = QString::fromStdString(invert_x ? "+" : "-"); | ||||||
| 
 |             return QObject::tr("Axis %1%2").arg(axis_x_str, invert_x_str); | ||||||
|             return QObject::tr("Axis %1").arg(axis_x_str); |  | ||||||
|         } |         } | ||||||
| 
 |         if (dir == "right") { | ||||||
|         if (dir == "up" || dir == "down") { |             const QString invert_x_str = QString::fromStdString(invert_x ? "-" : "+"); | ||||||
|             const QString axis_y_str = QString::fromStdString(param.Get("axis_y", "")); |             return QObject::tr("Axis %1%2").arg(axis_x_str, invert_x_str); | ||||||
| 
 |  | ||||||
|             return QObject::tr("Axis %1").arg(axis_y_str); |  | ||||||
|         } |         } | ||||||
| 
 |         if (dir == "up") { | ||||||
|         return {}; |             const QString invert_y_str = QString::fromStdString(invert_y ? "-" : "+"); | ||||||
|     } |             return QObject::tr("Axis %1%2").arg(axis_y_str, invert_y_str); | ||||||
| 
 |  | ||||||
|     if (param.Get("engine", "") == "gcpad") { |  | ||||||
|         if (dir == "modifier") { |  | ||||||
|             return QObject::tr("[unused]"); |  | ||||||
|         } |         } | ||||||
| 
 |         if (dir == "down") { | ||||||
|         if (dir == "left" || dir == "right") { |             const QString invert_y_str = QString::fromStdString(invert_y ? "+" : "-"); | ||||||
|             const QString axis_x_str = QString::fromStdString(param.Get("axis_x", "")); |             return QObject::tr("Axis %1%2").arg(axis_y_str, invert_y_str); | ||||||
| 
 |  | ||||||
|             return QObject::tr("GC Axis %1").arg(axis_x_str); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (dir == "up" || dir == "down") { |  | ||||||
|             const QString axis_y_str = QString::fromStdString(param.Get("axis_y", "")); |  | ||||||
| 
 |  | ||||||
|             return QObject::tr("GC Axis %1").arg(axis_y_str); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         return {}; |  | ||||||
|     } |  | ||||||
| 
 |  | ||||||
|     if (param.Get("engine", "") == "mouse") { |  | ||||||
|         if (dir == "modifier") { |  | ||||||
|             return QObject::tr("[unused]"); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (dir == "left" || dir == "right") { |  | ||||||
|             const QString axis_x_str = QString::fromStdString(param.Get("axis_x", "")); |  | ||||||
| 
 |  | ||||||
|             return QObject::tr("Mouse %1").arg(axis_x_str); |  | ||||||
|         } |  | ||||||
| 
 |  | ||||||
|         if (dir == "up" || dir == "down") { |  | ||||||
|             const QString axis_y_str = QString::fromStdString(param.Get("axis_y", "")); |  | ||||||
| 
 |  | ||||||
|             return QObject::tr("Mouse %1").arg(axis_y_str); |  | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         return {}; |         return {}; | ||||||
| @ -396,6 +366,25 @@ ConfigureInputPlayer::ConfigureInputPlayer(QWidget* parent, std::size_t player_i | |||||||
|                             analogs_param[analog_id].Clear(); |                             analogs_param[analog_id].Clear(); | ||||||
|                             analog_map_buttons[analog_id][sub_button_id]->setText(tr("[not set]")); |                             analog_map_buttons[analog_id][sub_button_id]->setText(tr("[not set]")); | ||||||
|                         }); |                         }); | ||||||
|  |                         context_menu.addAction(tr("Invert axis"), [&] { | ||||||
|  |                             if (sub_button_id == 2 || sub_button_id == 3) { | ||||||
|  |                                 const bool invert_value = | ||||||
|  |                                     analogs_param[analog_id].Get("invert_x", "+") == "-"; | ||||||
|  |                                 const std::string invert_str = invert_value ? "+" : "-"; | ||||||
|  |                                 analogs_param[analog_id].Set("invert_x", invert_str); | ||||||
|  |                             } | ||||||
|  |                             if (sub_button_id == 0 || sub_button_id == 1) { | ||||||
|  |                                 const bool invert_value = | ||||||
|  |                                     analogs_param[analog_id].Get("invert_y", "+") == "-"; | ||||||
|  |                                 const std::string invert_str = invert_value ? "+" : "-"; | ||||||
|  |                                 analogs_param[analog_id].Set("invert_y", invert_str); | ||||||
|  |                             } | ||||||
|  |                             for (int sub_button_id = 0; sub_button_id < ANALOG_SUB_BUTTONS_NUM; | ||||||
|  |                                  ++sub_button_id) { | ||||||
|  |                                 analog_map_buttons[analog_id][sub_button_id]->setText(AnalogToText( | ||||||
|  |                                     analogs_param[analog_id], analog_sub_buttons[sub_button_id])); | ||||||
|  |                             } | ||||||
|  |                         }); | ||||||
|                         context_menu.exec(analog_map_buttons[analog_id][sub_button_id]->mapToGlobal( |                         context_menu.exec(analog_map_buttons[analog_id][sub_button_id]->mapToGlobal( | ||||||
|                             menu_location)); |                             menu_location)); | ||||||
|                     }); |                     }); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 german
						german