2#include <ossia/detail/hash_map.hpp>
4#include <ossia/detail/timer.hpp>
5#include <ossia/network/context.hpp>
6#include <ossia/protocols/joystick/game_controller_protocol.hpp>
7#include <ossia/protocols/joystick/joystick_protocol.hpp>
9#if __has_include(<SDL2/SDL.h>)
15#include <ossia/detail/fmt.hpp>
20struct sdl_joystick_context
22 sdl_joystick_context()
24 SDL_SetHint(SDL_HINT_NO_SIGNAL_HANDLERS,
"1");
26 SDL_SetHint(SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS,
"1");
28 if(
int ret = SDL_Init(
29 SDL_INIT_JOYSTICK | SDL_INIT_HAPTIC | SDL_INIT_SENSOR
30 | SDL_INIT_GAMECONTROLLER);
32 throw std::runtime_error(fmt::format(
"SDL Init failure: {}", SDL_GetError()));
34 SDL_JoystickEventState(SDL_ENABLE);
35 SDL_GameControllerEventState(SDL_ENABLE);
38 static sdl_joystick_context& instance()
40 static sdl_joystick_context instance{};
44 ~sdl_joystick_context()
48 ev.type = SDL_FIRSTEVENT;
55class joystick_protocol_manager
58 static joystick_protocol_manager& instance()
60 static joystick_protocol_manager instance{};
64 joystick_protocol_manager() { sdl_joystick_context::instance(); }
66 ~joystick_protocol_manager() { SDL_Quit(); }
68 bool joystick_is_registered(
const SDL_JoystickID joystick_id)
70 return m_joystick_protocols.find(joystick_id) != m_joystick_protocols.end();
73 void register_protocol(
auto& protocol)
75 const SDL_JoystickID joystick_id = protocol.m_joystick_id;
77 if(joystick_is_registered(joystick_id))
78 throw std::runtime_error(
"A protocol is already registered for this joystick");
81 std::lock_guard<std::mutex> _{m_joystick_protocols_mutex};
82 m_joystick_protocols[joystick_id] = &protocol;
86 void unregister_protocol(
auto& protocol)
88 const SDL_JoystickID joystick_id = protocol.m_joystick_id;
90 if(!joystick_is_registered(joystick_id))
91 throw std::runtime_error(
92 "Cannot unregister a protocol that haven't been registered");
95 std::lock_guard<std::mutex> _{m_joystick_protocols_mutex};
96 m_joystick_protocols.erase(joystick_id);
100 template <
typename T>
101 T* get_protocol_by_id(
const SDL_JoystickID
id)
103 std::lock_guard<std::mutex> _{m_joystick_protocols_mutex};
104 auto it = m_joystick_protocols.find(
id);
105 if(it != m_joystick_protocols.end())
107 proto& p = it->second;
108 if(
auto res = ossia::get_if<T*>(&p); res && *res)
114 using proto = ossia::variant<joystick_protocol*, game_controller_protocol*>;
115 ossia::hash_map<SDL_JoystickID, proto> m_joystick_protocols;
116 std::mutex m_joystick_protocols_mutex;
121struct joystick_event_processor
125 explicit timer_context(boost::asio::io_context& ctx)
131 timer_context(timer_context&&) =
default;
132 timer_context& operator=(timer_context&&) =
default;
134 boost::asio::io_context* context{};
139 static inline std::atomic_int instance_count = 0;
140 joystick_event_processor(joystick_protocol_manager& manager)
145 ~joystick_event_processor() =
default;
147 static joystick_event_processor& instance(joystick_protocol_manager& manager)
149 static joystick_event_processor instance{manager};
153 void register_context(boost::asio::io_context& ctx)
155 for(
auto& tm : this->m_timers)
157 if(tm.context == &ctx)
164 m_timers.emplace_back(ctx);
166 if(instance_count > 0)
168 auto& tm = m_timers.back();
173 void start(ossia::timer& timer)
175 using namespace std::literals;
176 timer.set_delay(4ms);
177 timer.start([
this] { this->process_events(); });
180 void unregister_context(boost::asio::io_context& ctx)
182 for(
auto it = m_timers.begin(); it != m_timers.end();)
185 if(tm.context == &ctx)
190 it = m_timers.erase(it);
199 void start_event_loop()
201 if(instance_count++ > 0)
204 for(
auto& tm : m_timers)
208 void stop_event_loop()
210 if(--instance_count > 0)
213 using namespace std::literals;
216 ev.type = SDL_FIRSTEVENT;
219 for(
auto& tm : m_timers)
226 proto->m_device->apply_incoming_message({*proto, 0}, *param, std::move(val));
229 void push_axis(
const SDL_JoyAxisEvent& ev)
231 if(
auto p = m_manager.get_protocol_by_id<joystick_protocol>(ev.which))
233 const float res = (ev.value + .5f) / (0x7FFF + .5f);
234 push(p, p->m_axis_parameters[ev.axis], res);
238 void push_axis(
const SDL_ControllerAxisEvent& ev)
240 if(
auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
242 const float res = (ev.value + .5f) / (0x7FFF + .5f);
243 push(p, p->m_axis_parameters[ev.axis], res);
247 void push_button(
const SDL_JoyButtonEvent& ev)
249 if(
auto p = m_manager.get_protocol_by_id<joystick_protocol>(ev.which))
251 push(p, p->m_button_parameters[ev.button],
bool(ev.state == SDL_PRESSED));
255 void push_button(
const SDL_ControllerButtonEvent& ev)
257 if(
auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
259 push(p, p->m_button_parameters[ev.button],
bool(ev.state == SDL_PRESSED));
263 void push_sensor(
const SDL_ControllerSensorEvent& ev)
265 if(
auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
268 p, p->m_sensor_parameters[ev.sensor],
269 ossia::vec3f{ev.data[0], ev.data[1], ev.data[2]});
273 void push_touchpad(
const SDL_ControllerTouchpadEvent& ev)
275 if(
auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
277 auto it = p->m_touchpads.find(ev.touchpad);
278 if(it == p->m_touchpads.end())
281 auto& touchpad = it->second;
282 if(ev.finger >= 0 && ev.finger < touchpad.fingers.size())
284 auto& finger = touchpad.fingers[ev.finger];
285 push(p, finger.x, ev.x);
286 push(p, finger.y, ev.y);
287 push(p, finger.pressure, ev.pressure);
292 void push_hat(
const SDL_JoyHatEvent& ev)
294 if(
auto p = m_manager.get_protocol_by_id<joystick_protocol>(ev.which))
296 const uint8_t v = ev.value;
297 float x = 0.0f, y = 0.0f;
301 else if(v & SDL_HAT_RIGHT)
306 else if(v & SDL_HAT_DOWN)
309 push(p, p->m_hat_parameters[ev.hat], std::array<float, 2>{x, y});
313 void process_event(
const SDL_Event& ev)
317 case SDL_JOYAXISMOTION:
321 case SDL_JOYBUTTONDOWN:
322 case SDL_JOYBUTTONUP:
323 push_button(ev.jbutton);
326 case SDL_JOYHATMOTION:
330 case SDL_CONTROLLERAXISMOTION:
333 case SDL_CONTROLLERBUTTONDOWN:
334 case SDL_CONTROLLERBUTTONUP:
335 push_button(ev.cbutton);
337 case SDL_CONTROLLERTOUCHPADDOWN:
338 case SDL_CONTROLLERTOUCHPADMOTION:
339 case SDL_CONTROLLERTOUCHPADUP:
340 push_touchpad(ev.ctouchpad);
342 case SDL_CONTROLLERSENSORUPDATE:
343 push_sensor(ev.csensor);
353 void process_events()
357 int max_event_count = 20;
358 while(SDL_PollEvent(&ev) && max_event_count-- > 0)
364 joystick_protocol_manager& m_manager;
365 std::vector<timer_context> m_timers;
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
The value class.
Definition value.hpp:173