OSSIA
Open Scenario System for Interactive Application
Loading...
Searching...
No Matches
joystick_manager.hpp
1#pragma once
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>
8
9#if __has_include(<SDL2/SDL.h>)
10#include <SDL2/SDL.h>
11#else
12#include <SDL.h>
13#endif
14
15#include <ossia/detail/fmt.hpp>
16
17namespace ossia::net
18{
19
20struct sdl_joystick_context
21{
22 sdl_joystick_context()
23 {
24 SDL_SetHint(SDL_HINT_NO_SIGNAL_HANDLERS, "1");
25 // Prevent SDL from setting SIGINT handler on Posix Systems
26 SDL_SetHint(SDL_HINT_JOYSTICK_ALLOW_BACKGROUND_EVENTS, "1");
27
28 if(int ret = SDL_Init(
29 SDL_INIT_JOYSTICK | SDL_INIT_HAPTIC | SDL_INIT_SENSOR
30 | SDL_INIT_GAMECONTROLLER);
31 ret < 0)
32 throw std::runtime_error(fmt::format("SDL Init failure: {}", SDL_GetError()));
33
34 SDL_JoystickEventState(SDL_ENABLE);
35 SDL_GameControllerEventState(SDL_ENABLE);
36 }
37
38 static sdl_joystick_context& instance()
39 {
40 static sdl_joystick_context instance{};
41 return instance;
42 }
43
44 ~sdl_joystick_context()
45 {
46 // To be sure to quit the event loop
47 SDL_Event ev;
48 ev.type = SDL_FIRSTEVENT;
49 SDL_PushEvent(&ev);
50
51 SDL_Quit();
52 }
53};
54
55class joystick_protocol_manager
56{
57public:
58 static joystick_protocol_manager& instance()
59 {
60 static joystick_protocol_manager instance{};
61 return instance;
62 }
63
64 joystick_protocol_manager() { sdl_joystick_context::instance(); }
65
66 ~joystick_protocol_manager() { SDL_Quit(); }
67
68 bool joystick_is_registered(const SDL_JoystickID joystick_id)
69 {
70 return m_joystick_protocols.find(joystick_id) != m_joystick_protocols.end();
71 }
72
73 void register_protocol(auto& protocol)
74 {
75 const SDL_JoystickID joystick_id = protocol.m_joystick_id;
76
77 if(joystick_is_registered(joystick_id))
78 throw std::runtime_error("A protocol is already registered for this joystick");
79
80 {
81 std::lock_guard<std::mutex> _{m_joystick_protocols_mutex};
82 m_joystick_protocols[joystick_id] = &protocol;
83 }
84 }
85
86 void unregister_protocol(auto& protocol)
87 {
88 const SDL_JoystickID joystick_id = protocol.m_joystick_id;
89
90 if(!joystick_is_registered(joystick_id))
91 throw std::runtime_error(
92 "Cannot unregister a protocol that haven't been registered");
93
94 {
95 std::lock_guard<std::mutex> _{m_joystick_protocols_mutex};
96 m_joystick_protocols.erase(joystick_id);
97 }
98 }
99
100 template <typename T>
101 T* get_protocol_by_id(const SDL_JoystickID id)
102 {
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())
106 {
107 proto& p = it->second;
108 if(auto res = ossia::get_if<T*>(&p); res && *res)
109 return *res;
110 }
111 return nullptr;
112 }
113
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;
117};
118
119// TODO refactor this so that each protocol gets callback only related to its
120// own joystick instead
121struct joystick_event_processor
122{
123 struct timer_context
124 {
125 explicit timer_context(boost::asio::io_context& ctx)
126 : context{&ctx}
127 , timer{ctx}
128 , count{1}
129 {
130 }
131 timer_context(timer_context&&) = default;
132 timer_context& operator=(timer_context&&) = default;
133
134 boost::asio::io_context* context{};
135 ossia::timer timer;
136 int count = 0;
137 };
138
139 static inline std::atomic_int instance_count = 0;
140 joystick_event_processor(joystick_protocol_manager& manager)
141 : m_manager{manager}
142 {
143 }
144
145 ~joystick_event_processor() = default;
146
147 static joystick_event_processor& instance(joystick_protocol_manager& manager)
148 {
149 static joystick_event_processor instance{manager};
150 return instance;
151 }
152
153 void register_context(boost::asio::io_context& ctx)
154 {
155 for(auto& tm : this->m_timers)
156 {
157 if(tm.context == &ctx)
158 {
159 tm.count++;
160 return;
161 }
162 }
163
164 m_timers.emplace_back(ctx);
165
166 if(instance_count > 0)
167 {
168 auto& tm = m_timers.back();
169 start(tm.timer);
170 }
171 }
172
173 void start(ossia::timer& timer)
174 {
175 using namespace std::literals;
176 timer.set_delay(4ms);
177 timer.start([this] { this->process_events(); });
178 }
179
180 void unregister_context(boost::asio::io_context& ctx)
181 {
182 for(auto it = m_timers.begin(); it != m_timers.end();)
183 {
184 auto& tm = *it;
185 if(tm.context == &ctx)
186 {
187 tm.count--;
188 if(tm.count == 0)
189 {
190 it = m_timers.erase(it);
191 continue;
192 }
193 }
194
195 ++it;
196 }
197 }
198
199 void start_event_loop()
200 {
201 if(instance_count++ > 0)
202 return;
203
204 for(auto& tm : m_timers)
205 start(tm.timer);
206 }
207
208 void stop_event_loop()
209 {
210 if(--instance_count > 0)
211 return;
212
213 using namespace std::literals;
214 // To be sure to quit the event loop
215 SDL_Event ev;
216 ev.type = SDL_FIRSTEVENT;
217 SDL_PushEvent(&ev);
218
219 for(auto& tm : m_timers)
220 tm.timer.stop();
221 }
222
223 void push(auto* proto, ossia::net::parameter_base* param, ossia::value val)
224 {
225 if(proto && param)
226 proto->m_device->apply_incoming_message({*proto, 0}, *param, std::move(val));
227 }
228
229 void push_axis(const SDL_JoyAxisEvent& ev)
230 {
231 if(auto p = m_manager.get_protocol_by_id<joystick_protocol>(ev.which))
232 {
233 const float res = (ev.value + .5f) / (0x7FFF + .5f);
234 push(p, p->m_axis_parameters[ev.axis], res);
235 }
236 }
237
238 void push_axis(const SDL_ControllerAxisEvent& ev)
239 {
240 if(auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
241 {
242 const float res = (ev.value + .5f) / (0x7FFF + .5f);
243 push(p, p->m_axis_parameters[ev.axis], res);
244 }
245 }
246
247 void push_button(const SDL_JoyButtonEvent& ev)
248 {
249 if(auto p = m_manager.get_protocol_by_id<joystick_protocol>(ev.which))
250 {
251 push(p, p->m_button_parameters[ev.button], bool(ev.state == SDL_PRESSED));
252 }
253 }
254
255 void push_button(const SDL_ControllerButtonEvent& ev)
256 {
257 if(auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
258 {
259 push(p, p->m_button_parameters[ev.button], bool(ev.state == SDL_PRESSED));
260 }
261 }
262
263 void push_sensor(const SDL_ControllerSensorEvent& ev)
264 {
265 if(auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
266 {
267 push(
268 p, p->m_sensor_parameters[ev.sensor],
269 ossia::vec3f{ev.data[0], ev.data[1], ev.data[2]});
270 }
271 }
272
273 void push_touchpad(const SDL_ControllerTouchpadEvent& ev)
274 {
275 if(auto p = m_manager.get_protocol_by_id<game_controller_protocol>(ev.which))
276 {
277 auto it = p->m_touchpads.find(ev.touchpad);
278 if(it == p->m_touchpads.end())
279 return;
280
281 auto& touchpad = it->second;
282 if(ev.finger >= 0 && ev.finger < touchpad.fingers.size())
283 {
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);
288 }
289 }
290 }
291
292 void push_hat(const SDL_JoyHatEvent& ev)
293 {
294 if(auto p = m_manager.get_protocol_by_id<joystick_protocol>(ev.which))
295 {
296 const uint8_t v = ev.value;
297 float x = 0.0f, y = 0.0f;
298
299 if(v & SDL_HAT_LEFT)
300 x = -1.0;
301 else if(v & SDL_HAT_RIGHT)
302 x = 1.0;
303
304 if(v & SDL_HAT_UP)
305 y = 1.0;
306 else if(v & SDL_HAT_DOWN)
307 y = -1;
308
309 push(p, p->m_hat_parameters[ev.hat], std::array<float, 2>{x, y});
310 }
311 }
312
313 void process_event(const SDL_Event& ev)
314 {
315 switch(ev.type)
316 {
317 case SDL_JOYAXISMOTION:
318 push_axis(ev.jaxis);
319 break;
320
321 case SDL_JOYBUTTONDOWN:
322 case SDL_JOYBUTTONUP:
323 push_button(ev.jbutton);
324 break;
325
326 case SDL_JOYHATMOTION:
327 push_hat(ev.jhat);
328 break;
329
330 case SDL_CONTROLLERAXISMOTION:
331 push_axis(ev.caxis);
332 break;
333 case SDL_CONTROLLERBUTTONDOWN:
334 case SDL_CONTROLLERBUTTONUP:
335 push_button(ev.cbutton);
336 break;
337 case SDL_CONTROLLERTOUCHPADDOWN:
338 case SDL_CONTROLLERTOUCHPADMOTION:
339 case SDL_CONTROLLERTOUCHPADUP:
340 push_touchpad(ev.ctouchpad);
341 break;
342 case SDL_CONTROLLERSENSORUPDATE:
343 push_sensor(ev.csensor);
344 break;
345 // case SDL_CONTROLLERUPDATECOMPLETE_RESERVED_FOR_SDL3:
346 // case SDL_CONTROLLERSTEAMHANDLEUPDATED:
347
348 default:
349 break;
350 }
351 }
352
353 void process_events()
354 {
355 SDL_Event ev;
356
357 int max_event_count = 20;
358 while(SDL_PollEvent(&ev) && max_event_count-- > 0)
359 {
360 process_event(ev);
361 }
362 }
363
364 joystick_protocol_manager& m_manager;
365 std::vector<timer_context> m_timers;
366};
367
368}
The parameter_base class.
Definition ossia/network/base/parameter.hpp:48
The value class.
Definition value.hpp:173