Mercurial > fife-parpg
comparison ext/guichan-0.8.1/src/sdl/sdlinput.cpp @ 0:4a0efb7baf70
* Datasets becomes the new trunk and retires after that :-)
author | mvbarracuda@33b003aa-7bff-0310-803a-e67f0ece8222 |
---|---|
date | Sun, 29 Jun 2008 18:44:17 +0000 |
parents | |
children |
comparison
equal
deleted
inserted
replaced
-1:000000000000 | 0:4a0efb7baf70 |
---|---|
1 /* _______ __ __ __ ______ __ __ _______ __ __ | |
2 * / _____/\ / /\ / /\ / /\ / ____/\ / /\ / /\ / ___ /\ / |\/ /\ | |
3 * / /\____\// / // / // / // /\___\// /_// / // /\_/ / // , |/ / / | |
4 * / / /__ / / // / // / // / / / ___ / // ___ / // /| ' / / | |
5 * / /_// /\ / /_// / // / // /_/_ / / // / // /\_/ / // / | / / | |
6 * /______/ //______/ //_/ //_____/\ /_/ //_/ //_/ //_/ //_/ /|_/ / | |
7 * \______\/ \______\/ \_\/ \_____\/ \_\/ \_\/ \_\/ \_\/ \_\/ \_\/ | |
8 * | |
9 * Copyright (c) 2004 - 2008 Olof Naessén and Per Larsson | |
10 * | |
11 * | |
12 * Per Larsson a.k.a finalman | |
13 * Olof Naessén a.k.a jansem/yakslem | |
14 * | |
15 * Visit: http://guichan.sourceforge.net | |
16 * | |
17 * License: (BSD) | |
18 * Redistribution and use in source and binary forms, with or without | |
19 * modification, are permitted provided that the following conditions | |
20 * are met: | |
21 * 1. Redistributions of source code must retain the above copyright | |
22 * notice, this list of conditions and the following disclaimer. | |
23 * 2. Redistributions in binary form must reproduce the above copyright | |
24 * notice, this list of conditions and the following disclaimer in | |
25 * the documentation and/or other materials provided with the | |
26 * distribution. | |
27 * 3. Neither the name of Guichan nor the names of its contributors may | |
28 * be used to endorse or promote products derived from this software | |
29 * without specific prior written permission. | |
30 * | |
31 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS | |
32 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT | |
33 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR | |
34 * A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT | |
35 * OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, | |
36 * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED | |
37 * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR | |
38 * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF | |
39 * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING | |
40 * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS | |
41 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | |
42 */ | |
43 | |
44 /* | |
45 * For comments regarding functions please see the header file. | |
46 */ | |
47 | |
48 #include "guichan/sdl/sdlinput.hpp" | |
49 | |
50 #include "guichan/exception.hpp" | |
51 | |
52 namespace gcn | |
53 { | |
54 SDLInput::SDLInput() | |
55 { | |
56 mMouseInWindow = true; | |
57 mMouseDown = false; | |
58 } | |
59 | |
60 bool SDLInput::isKeyQueueEmpty() | |
61 { | |
62 return mKeyInputQueue.empty(); | |
63 } | |
64 | |
65 KeyInput SDLInput::dequeueKeyInput() | |
66 { | |
67 KeyInput keyInput; | |
68 | |
69 if (mKeyInputQueue.empty()) | |
70 { | |
71 throw GCN_EXCEPTION("The queue is empty."); | |
72 } | |
73 | |
74 keyInput = mKeyInputQueue.front(); | |
75 mKeyInputQueue.pop(); | |
76 | |
77 return keyInput; | |
78 } | |
79 | |
80 bool SDLInput::isMouseQueueEmpty() | |
81 { | |
82 return mMouseInputQueue.empty(); | |
83 } | |
84 | |
85 MouseInput SDLInput::dequeueMouseInput() | |
86 { | |
87 MouseInput mouseInput; | |
88 | |
89 if (mMouseInputQueue.empty()) | |
90 { | |
91 throw GCN_EXCEPTION("The queue is empty."); | |
92 } | |
93 | |
94 mouseInput = mMouseInputQueue.front(); | |
95 mMouseInputQueue.pop(); | |
96 | |
97 return mouseInput; | |
98 } | |
99 | |
100 void SDLInput::pushInput(SDL_Event event) | |
101 { | |
102 KeyInput keyInput; | |
103 MouseInput mouseInput; | |
104 | |
105 switch (event.type) | |
106 { | |
107 case SDL_KEYDOWN: | |
108 { | |
109 int value = convertSDLEventToGuichanKeyValue(event); | |
110 | |
111 if (value == -1) | |
112 { | |
113 value = (int)event.key.keysym.unicode; | |
114 } | |
115 | |
116 keyInput.setKey(Key(value)); | |
117 keyInput.setType(KeyInput::PRESSED); | |
118 keyInput.setShiftPressed(event.key.keysym.mod & KMOD_SHIFT); | |
119 keyInput.setControlPressed(event.key.keysym.mod & KMOD_CTRL); | |
120 keyInput.setAltPressed(event.key.keysym.mod & KMOD_ALT); | |
121 keyInput.setMetaPressed(event.key.keysym.mod & KMOD_META); | |
122 keyInput.setNumericPad(event.key.keysym.sym >= SDLK_KP0 | |
123 && event.key.keysym.sym <= SDLK_KP_EQUALS); | |
124 | |
125 mKeyInputQueue.push(keyInput); | |
126 break; | |
127 } | |
128 | |
129 case SDL_KEYUP: | |
130 { | |
131 int value = convertSDLEventToGuichanKeyValue(event); | |
132 | |
133 if (value == -1) | |
134 { | |
135 value = (int)event.key.keysym.sym; | |
136 } | |
137 | |
138 keyInput.setKey(Key(value)); | |
139 keyInput.setType(KeyInput::RELEASED); | |
140 keyInput.setShiftPressed(event.key.keysym.mod & KMOD_SHIFT); | |
141 keyInput.setControlPressed(event.key.keysym.mod & KMOD_CTRL); | |
142 keyInput.setAltPressed(event.key.keysym.mod & KMOD_ALT); | |
143 keyInput.setMetaPressed(event.key.keysym.mod & KMOD_META); | |
144 keyInput.setNumericPad(event.key.keysym.sym >= SDLK_KP0 | |
145 && event.key.keysym.sym <= SDLK_KP_EQUALS); | |
146 | |
147 mKeyInputQueue.push(keyInput); | |
148 break; | |
149 } | |
150 | |
151 case SDL_MOUSEBUTTONDOWN: | |
152 mMouseDown = true; | |
153 mouseInput.setX(event.button.x); | |
154 mouseInput.setY(event.button.y); | |
155 mouseInput.setButton(convertMouseButton(event.button.button)); | |
156 | |
157 if (event.button.button == SDL_BUTTON_WHEELDOWN) | |
158 { | |
159 mouseInput.setType(MouseInput::WHEEL_MOVED_DOWN); | |
160 } | |
161 else if (event.button.button == SDL_BUTTON_WHEELUP) | |
162 { | |
163 mouseInput.setType(MouseInput::WHEEL_MOVED_UP); | |
164 } | |
165 else | |
166 { | |
167 mouseInput.setType(MouseInput::PRESSED); | |
168 } | |
169 mouseInput.setTimeStamp(SDL_GetTicks()); | |
170 mMouseInputQueue.push(mouseInput); | |
171 break; | |
172 | |
173 case SDL_MOUSEBUTTONUP: | |
174 mMouseDown = false; | |
175 mouseInput.setX(event.button.x); | |
176 mouseInput.setY(event.button.y); | |
177 mouseInput.setButton(convertMouseButton(event.button.button)); | |
178 mouseInput.setType(MouseInput::RELEASED); | |
179 mouseInput.setTimeStamp(SDL_GetTicks()); | |
180 mMouseInputQueue.push(mouseInput); | |
181 break; | |
182 | |
183 case SDL_MOUSEMOTION: | |
184 mouseInput.setX(event.button.x); | |
185 mouseInput.setY(event.button.y); | |
186 mouseInput.setButton(MouseInput::EMPTY); | |
187 mouseInput.setType(MouseInput::MOVED); | |
188 mouseInput.setTimeStamp(SDL_GetTicks()); | |
189 mMouseInputQueue.push(mouseInput); | |
190 break; | |
191 | |
192 case SDL_ACTIVEEVENT: | |
193 /* | |
194 * This occurs when the mouse leaves the window and the Gui-chan | |
195 * application loses its mousefocus. | |
196 */ | |
197 if ((event.active.state & SDL_APPMOUSEFOCUS) | |
198 && !event.active.gain) | |
199 { | |
200 mMouseInWindow = false; | |
201 | |
202 if (!mMouseDown) | |
203 { | |
204 mouseInput.setX(-1); | |
205 mouseInput.setY(-1); | |
206 mouseInput.setButton(MouseInput::EMPTY); | |
207 mouseInput.setType(MouseInput::MOVED); | |
208 mMouseInputQueue.push(mouseInput); | |
209 } | |
210 } | |
211 | |
212 if ((event.active.state & SDL_APPMOUSEFOCUS) | |
213 && event.active.gain) | |
214 { | |
215 mMouseInWindow = true; | |
216 } | |
217 break; | |
218 | |
219 } // end switch | |
220 } | |
221 | |
222 int SDLInput::convertMouseButton(int button) | |
223 { | |
224 switch (button) | |
225 { | |
226 case SDL_BUTTON_LEFT: | |
227 return MouseInput::LEFT; | |
228 break; | |
229 case SDL_BUTTON_RIGHT: | |
230 return MouseInput::RIGHT; | |
231 break; | |
232 case SDL_BUTTON_MIDDLE: | |
233 return MouseInput::MIDDLE; | |
234 break; | |
235 default: | |
236 // We have an unknown mouse type which is ignored. | |
237 return button; | |
238 } | |
239 } | |
240 | |
241 int SDLInput::convertSDLEventToGuichanKeyValue(SDL_Event event) | |
242 { | |
243 int value = -1; | |
244 | |
245 switch (event.key.keysym.sym) | |
246 { | |
247 case SDLK_TAB: | |
248 value = Key::TAB; | |
249 break; | |
250 case SDLK_LALT: | |
251 value = Key::LEFT_ALT; | |
252 break; | |
253 case SDLK_RALT: | |
254 value = Key::RIGHT_ALT; | |
255 break; | |
256 case SDLK_LSHIFT: | |
257 value = Key::LEFT_SHIFT; | |
258 break; | |
259 case SDLK_RSHIFT: | |
260 value = Key::RIGHT_SHIFT; | |
261 break; | |
262 case SDLK_LCTRL: | |
263 value = Key::LEFT_CONTROL; | |
264 break; | |
265 case SDLK_RCTRL: | |
266 value = Key::RIGHT_CONTROL; | |
267 break; | |
268 case SDLK_BACKSPACE: | |
269 value = Key::BACKSPACE; | |
270 break; | |
271 case SDLK_PAUSE: | |
272 value = Key::PAUSE; | |
273 break; | |
274 case SDLK_SPACE: | |
275 // Special characters like ~ (tilde) ends up | |
276 // with the keysym.sym SDLK_SPACE which | |
277 // without this check would be lost. The check | |
278 // is only valid on key down events in SDL. | |
279 if (event.type == SDL_KEYUP || event.key.keysym.unicode == ' ') | |
280 { | |
281 value = Key::SPACE; | |
282 } | |
283 break; | |
284 case SDLK_ESCAPE: | |
285 value = Key::ESCAPE; | |
286 break; | |
287 case SDLK_DELETE: | |
288 value = Key::DELETE; | |
289 break; | |
290 case SDLK_INSERT: | |
291 value = Key::INSERT; | |
292 break; | |
293 case SDLK_HOME: | |
294 value = Key::HOME; | |
295 break; | |
296 case SDLK_END: | |
297 value = Key::END; | |
298 break; | |
299 case SDLK_PAGEUP: | |
300 value = Key::PAGE_UP; | |
301 break; | |
302 case SDLK_PRINT: | |
303 value = Key::PRINT_SCREEN; | |
304 break; | |
305 case SDLK_PAGEDOWN: | |
306 value = Key::PAGE_DOWN; | |
307 break; | |
308 case SDLK_F1: | |
309 value = Key::F1; | |
310 break; | |
311 case SDLK_F2: | |
312 value = Key::F2; | |
313 break; | |
314 case SDLK_F3: | |
315 value = Key::F3; | |
316 break; | |
317 case SDLK_F4: | |
318 value = Key::F4; | |
319 break; | |
320 case SDLK_F5: | |
321 value = Key::F5; | |
322 break; | |
323 case SDLK_F6: | |
324 value = Key::F6; | |
325 break; | |
326 case SDLK_F7: | |
327 value = Key::F7; | |
328 break; | |
329 case SDLK_F8: | |
330 value = Key::F8; | |
331 break; | |
332 case SDLK_F9: | |
333 value = Key::F9; | |
334 break; | |
335 case SDLK_F10: | |
336 value = Key::F10; | |
337 break; | |
338 case SDLK_F11: | |
339 value = Key::F11; | |
340 break; | |
341 case SDLK_F12: | |
342 value = Key::F12; | |
343 break; | |
344 case SDLK_F13: | |
345 value = Key::F13; | |
346 break; | |
347 case SDLK_F14: | |
348 value = Key::F14; | |
349 break; | |
350 case SDLK_F15: | |
351 value = Key::F15; | |
352 break; | |
353 case SDLK_NUMLOCK: | |
354 value = Key::NUM_LOCK; | |
355 break; | |
356 case SDLK_CAPSLOCK: | |
357 value = Key::CAPS_LOCK; | |
358 break; | |
359 case SDLK_SCROLLOCK: | |
360 value = Key::SCROLL_LOCK; | |
361 break; | |
362 case SDLK_RMETA: | |
363 value = Key::RIGHT_META; | |
364 break; | |
365 case SDLK_LMETA: | |
366 value = Key::LEFT_META; | |
367 break; | |
368 case SDLK_LSUPER: | |
369 value = Key::LEFT_SUPER; | |
370 break; | |
371 case SDLK_RSUPER: | |
372 value = Key::RIGHT_SUPER; | |
373 break; | |
374 case SDLK_MODE: | |
375 value = Key::ALT_GR; | |
376 break; | |
377 case SDLK_UP: | |
378 value = Key::UP; | |
379 break; | |
380 case SDLK_DOWN: | |
381 value = Key::DOWN; | |
382 break; | |
383 case SDLK_LEFT: | |
384 value = Key::LEFT; | |
385 break; | |
386 case SDLK_RIGHT: | |
387 value = Key::RIGHT; | |
388 break; | |
389 case SDLK_RETURN: | |
390 value = Key::ENTER; | |
391 break; | |
392 case SDLK_KP_ENTER: | |
393 value = Key::ENTER; | |
394 break; | |
395 | |
396 default: | |
397 break; | |
398 } | |
399 | |
400 if (!(event.key.keysym.mod & KMOD_NUM)) | |
401 { | |
402 switch (event.key.keysym.sym) | |
403 { | |
404 case SDLK_KP0: | |
405 value = Key::INSERT; | |
406 break; | |
407 case SDLK_KP1: | |
408 value = Key::END; | |
409 break; | |
410 case SDLK_KP2: | |
411 value = Key::DOWN; | |
412 break; | |
413 case SDLK_KP3: | |
414 value = Key::PAGE_DOWN; | |
415 break; | |
416 case SDLK_KP4: | |
417 value = Key::LEFT; | |
418 break; | |
419 case SDLK_KP5: | |
420 value = 0; | |
421 break; | |
422 case SDLK_KP6: | |
423 value = Key::RIGHT; | |
424 break; | |
425 case SDLK_KP7: | |
426 value = Key::HOME; | |
427 break; | |
428 case SDLK_KP8: | |
429 value = Key::UP; | |
430 break; | |
431 case SDLK_KP9: | |
432 value = Key::PAGE_UP; | |
433 break; | |
434 default: | |
435 break; | |
436 } | |
437 } | |
438 | |
439 return value; | |
440 } | |
441 } |