| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | // Copyright 2017 Citra Emulator Project
 | 
					
						
							|  |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							|  |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  | #include <chrono>
 | 
					
						
							|  |  |  | #include <cmath>
 | 
					
						
							|  |  |  | #include <thread>
 | 
					
						
							|  |  |  | #include "common/math_util.h"
 | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | #include "input_common/analog_from_button.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace InputCommon { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class Analog final : public Input::AnalogDevice { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     using Button = std::unique_ptr<Input::ButtonDevice>; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Analog(Button up_, Button down_, Button left_, Button right_, Button modifier_, | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |            float modifier_scale_, float modifier_angle_) | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  |         : up(std::move(up_)), down(std::move(down_)), left(std::move(left_)), | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |           right(std::move(right_)), modifier(std::move(modifier_)), modifier_scale(modifier_scale_), | 
					
						
							|  |  |  |           modifier_angle(modifier_angle_) { | 
					
						
							|  |  |  |         update_thread = std::thread(&Analog::UpdateStatus, this); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     ~Analog() override { | 
					
						
							|  |  |  |         update_thread_running = false; | 
					
						
							|  |  |  |         if (update_thread.joinable()) { | 
					
						
							|  |  |  |             update_thread.join(); | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void MoveToDirection(bool enable, float to_angle) { | 
					
						
							|  |  |  |         if (!enable) { | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |         constexpr float TAU = Common::PI * 2.0f; | 
					
						
							|  |  |  |         // Use wider angle to ease the transition.
 | 
					
						
							|  |  |  |         constexpr float aperture = TAU * 0.15f; | 
					
						
							|  |  |  |         const float top_limit = to_angle + aperture; | 
					
						
							|  |  |  |         const float bottom_limit = to_angle - aperture; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if ((angle > to_angle && angle <= top_limit) || | 
					
						
							|  |  |  |             (angle + TAU > to_angle && angle + TAU <= top_limit)) { | 
					
						
							|  |  |  |             angle -= modifier_angle; | 
					
						
							|  |  |  |             if (angle < 0) { | 
					
						
							|  |  |  |                 angle += TAU; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else if ((angle >= bottom_limit && angle < to_angle) || | 
					
						
							| 
									
										
										
										
											2020-11-10 10:38:15 -06:00
										 |  |  |                    (angle - TAU >= bottom_limit && angle - TAU < to_angle)) { | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |             angle += modifier_angle; | 
					
						
							|  |  |  |             if (angle >= TAU) { | 
					
						
							|  |  |  |                 angle -= TAU; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             angle = to_angle; | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void UpdateStatus() { | 
					
						
							|  |  |  |         while (update_thread_running) { | 
					
						
							|  |  |  |             const float coef = modifier->GetStatus() ? modifier_scale : 1.0f; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             bool r = right->GetStatus(); | 
					
						
							|  |  |  |             bool l = left->GetStatus(); | 
					
						
							|  |  |  |             bool u = up->GetStatus(); | 
					
						
							|  |  |  |             bool d = down->GetStatus(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Eliminate contradictory movements
 | 
					
						
							|  |  |  |             if (r && l) { | 
					
						
							|  |  |  |                 r = false; | 
					
						
							|  |  |  |                 l = false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             if (u && d) { | 
					
						
							|  |  |  |                 u = false; | 
					
						
							|  |  |  |                 d = false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move to the right
 | 
					
						
							|  |  |  |             MoveToDirection(r && !u && !d, 0.0f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move to the upper right
 | 
					
						
							|  |  |  |             MoveToDirection(r && u && !d, Common::PI * 0.25f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move up
 | 
					
						
							|  |  |  |             MoveToDirection(u && !l && !r, Common::PI * 0.5f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move to the upper left
 | 
					
						
							|  |  |  |             MoveToDirection(l && u && !d, Common::PI * 0.75f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move to the left
 | 
					
						
							|  |  |  |             MoveToDirection(l && !u && !d, Common::PI); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move to the bottom left
 | 
					
						
							|  |  |  |             MoveToDirection(l && !u && d, Common::PI * 1.25f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move down
 | 
					
						
							|  |  |  |             MoveToDirection(d && !l && !r, Common::PI * 1.5f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move to the bottom right
 | 
					
						
							|  |  |  |             MoveToDirection(r && !u && d, Common::PI * 1.75f); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Move if a key is pressed
 | 
					
						
							|  |  |  |             if (r || l || u || d) { | 
					
						
							|  |  |  |                 amplitude = coef; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 amplitude = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             // Delay the update rate to 100hz
 | 
					
						
							|  |  |  |             std::this_thread::sleep_for(std::chrono::milliseconds(10)); | 
					
						
							| 
									
										
										
										
											2020-10-14 02:51:14 -04:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     std::tuple<float, float> GetStatus() const override { | 
					
						
							|  |  |  |         return std::make_tuple(std::cos(angle) * amplitude, std::sin(angle) * amplitude); | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-18 06:45:37 +01:00
										 |  |  |     bool GetAnalogDirectionStatus(Input::AnalogDirection direction) const override { | 
					
						
							|  |  |  |         switch (direction) { | 
					
						
							|  |  |  |         case Input::AnalogDirection::RIGHT: | 
					
						
							|  |  |  |             return right->GetStatus(); | 
					
						
							|  |  |  |         case Input::AnalogDirection::LEFT: | 
					
						
							|  |  |  |             return left->GetStatus(); | 
					
						
							|  |  |  |         case Input::AnalogDirection::UP: | 
					
						
							|  |  |  |             return up->GetStatus(); | 
					
						
							|  |  |  |         case Input::AnalogDirection::DOWN: | 
					
						
							|  |  |  |             return down->GetStatus(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | private: | 
					
						
							|  |  |  |     Button up; | 
					
						
							|  |  |  |     Button down; | 
					
						
							|  |  |  |     Button left; | 
					
						
							|  |  |  |     Button right; | 
					
						
							|  |  |  |     Button modifier; | 
					
						
							|  |  |  |     float modifier_scale; | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     float modifier_angle; | 
					
						
							|  |  |  |     float angle{}; | 
					
						
							|  |  |  |     float amplitude{}; | 
					
						
							|  |  |  |     std::thread update_thread; | 
					
						
							|  |  |  |     bool update_thread_running{true}; | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | std::unique_ptr<Input::AnalogDevice> AnalogFromButton::Create(const Common::ParamPackage& params) { | 
					
						
							|  |  |  |     const std::string null_engine = Common::ParamPackage{{"engine", "null"}}.Serialize(); | 
					
						
							|  |  |  |     auto up = Input::CreateDevice<Input::ButtonDevice>(params.Get("up", null_engine)); | 
					
						
							|  |  |  |     auto down = Input::CreateDevice<Input::ButtonDevice>(params.Get("down", null_engine)); | 
					
						
							|  |  |  |     auto left = Input::CreateDevice<Input::ButtonDevice>(params.Get("left", null_engine)); | 
					
						
							|  |  |  |     auto right = Input::CreateDevice<Input::ButtonDevice>(params.Get("right", null_engine)); | 
					
						
							|  |  |  |     auto modifier = Input::CreateDevice<Input::ButtonDevice>(params.Get("modifier", null_engine)); | 
					
						
							|  |  |  |     auto modifier_scale = params.Get("modifier_scale", 0.5f); | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |     auto modifier_angle = params.Get("modifier_angle", 0.035f); | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  |     return std::make_unique<Analog>(std::move(up), std::move(down), std::move(left), | 
					
						
							| 
									
										
										
										
											2020-11-07 09:48:11 -06:00
										 |  |  |                                     std::move(right), std::move(modifier), modifier_scale, | 
					
						
							|  |  |  |                                     modifier_angle); | 
					
						
							| 
									
										
										
										
											2017-01-21 13:04:00 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } // namespace InputCommon
 |