This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_touch.c
530 lines (455 loc) · 12.7 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
/*
SDL - Simple DirectMedia Layer
Copyright (C) 1997-2010 Sam Lantinga
This library is free software; you can redistribute it and/or
modify it under the terms of the GNU Lesser General Public
License as published by the Free Software Foundation; either
version 2.1 of the License, or (at your option) any later version.
This library is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public
License along with this library; if not, write to the Free Software
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
Sam Lantinga
slouken@libsdl.org
*/
#include "SDL_config.h"
/* General touch handling code for SDL */
#include "SDL_events.h"
#include "SDL_events_c.h"
#include "../video/SDL_sysvideo.h"
30
31
#include <stdio.h>
32
33
static int SDL_num_touch = 0;
34
static SDL_Touch **SDL_touchPads = NULL;
35
36
37
38
39
40
/* Public functions */
int
SDL_TouchInit(void)
{
41
return (0);
42
}
43
44
SDL_Touch *
45
SDL_GetTouch(long id)
46
{
47
int index = SDL_GetTouchIndexId(id);
48
49
50
if (index < 0 || index >= SDL_num_touch) {
return NULL;
}
51
52
53
return SDL_touchPads[index];
}
54
55
SDL_Touch *
SDL_GetTouchIndex(int index)
56
{
57
58
59
60
if (index < 0 || index >= SDL_num_touch) {
return NULL;
}
return SDL_touchPads[index];
61
62
63
}
int
64
SDL_GetFingerIndexId(SDL_Touch* touch,long fingerid)
65
66
67
68
69
70
{
int i;
for(i = 0;i < touch->num_fingers;i++)
if(touch->fingers[i]->id == fingerid)
return i;
return -1;
71
72
}
73
74
SDL_Finger *
75
SDL_GetFinger(SDL_Touch* touch,long id)
76
77
78
79
80
81
82
83
{
int index = SDL_GetFingerIndexId(touch,id);
if(index < 0 || index >= touch->num_fingers)
return NULL;
return touch->fingers[index];
}
84
int
85
SDL_GetTouchIndexId(long id)
86
87
88
89
90
{
int index;
SDL_Touch *touch;
for (index = 0; index < SDL_num_touch; ++index) {
91
touch = SDL_touchPads[index];
92
93
94
95
96
97
98
99
if (touch->id == id) {
return index;
}
}
return -1;
}
int
100
SDL_AddTouch(const SDL_Touch * touch, char *name)
101
{
102
SDL_Touch **touchPads;
103
int index,length;
104
105
106
107
108
109
if (SDL_GetTouchIndexId(touch->id) != -1) {
SDL_SetError("Touch ID already in use");
}
/* Add the touch to the list of touch */
110
touchPads = (SDL_Touch **) SDL_realloc(SDL_touchPads,
111
(SDL_num_touch + 1) * sizeof(*touch));
112
if (!touchPads) {
113
114
115
116
SDL_OutOfMemory();
return -1;
}
117
SDL_touchPads = touchPads;
118
119
index = SDL_num_touch++;
120
121
SDL_touchPads[index] = (SDL_Touch *) SDL_malloc(sizeof(*SDL_touchPads[index]));
if (!SDL_touchPads[index]) {
122
123
124
SDL_OutOfMemory();
return -1;
}
125
*SDL_touchPads[index] = *touch;
126
127
128
129
/* we're setting the touch properties */
length = 0;
length = SDL_strlen(name);
130
131
SDL_touchPads[index]->focus = 0;
SDL_touchPads[index]->name = SDL_malloc((length + 2) * sizeof(char));
132
SDL_strlcpy(SDL_touchPads[index]->name, name, length + 1);
133
134
SDL_touchPads[index]->num_fingers = 0;
135
136
137
SDL_touchPads[index]->max_fingers = 1;
SDL_touchPads[index]->fingers = (SDL_Finger **) SDL_malloc(sizeof(SDL_Finger*));
SDL_touchPads[index]->fingers[0] = NULL;
138
139
140
141
SDL_touchPads[index]->buttonstate = 0;
SDL_touchPads[index]->relative_mode = SDL_FALSE;
SDL_touchPads[index]->flush_motion = SDL_FALSE;
142
143
144
//Do I want this here? Probably
SDL_GestureAddTouch(SDL_touchPads[index]);
145
146
147
148
return index;
}
void
149
SDL_DelTouch(long id)
150
{
151
152
int index = SDL_GetTouchIndexId(id);
SDL_Touch *touch = SDL_GetTouch(id);
153
154
155
156
157
if (!touch) {
return;
}
158
159
160
161
162
163
164
165
SDL_free(touch->name);
if (touch->FreeTouch) {
touch->FreeTouch(touch);
}
SDL_free(touch);
166
167
SDL_num_touch--;
SDL_touchPads[index] = SDL_touchPads[SDL_num_touch];
168
169
170
171
172
173
174
}
void
SDL_TouchQuit(void)
{
int i;
175
for (i = SDL_num_touch-1; i > 0 ; --i) {
176
177
178
179
SDL_DelTouch(i);
}
SDL_num_touch = 0;
180
181
182
if (SDL_touchPads) {
SDL_free(SDL_touchPads);
SDL_touchPads = NULL;
183
184
185
186
187
188
189
190
191
}
}
int
SDL_GetNumTouch(void)
{
return SDL_num_touch;
}
SDL_Window *
192
SDL_GetTouchFocusWindow(long id)
193
{
194
SDL_Touch *touch = SDL_GetTouch(id);
195
196
197
198
199
200
201
202
if (!touch) {
return 0;
}
return touch->focus;
}
void
203
SDL_SetTouchFocus(long id, SDL_Window * window)
204
205
{
int index = SDL_GetTouchIndexId(id);
206
SDL_Touch *touch = SDL_GetTouch(id);
207
208
209
210
211
212
213
214
215
216
217
218
219
int i;
SDL_bool focus;
if (!touch || (touch->focus == window)) {
return;
}
/* See if the current window has lost focus */
if (touch->focus) {
focus = SDL_FALSE;
for (i = 0; i < SDL_num_touch; ++i) {
SDL_Touch *check;
if (i != index) {
220
check = SDL_touchPads[i];
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
if (check && check->focus == touch->focus) {
focus = SDL_TRUE;
break;
}
}
}
if (!focus) {
SDL_SendWindowEvent(touch->focus, SDL_WINDOWEVENT_LEAVE, 0, 0);
}
}
touch->focus = window;
if (touch->focus) {
focus = SDL_FALSE;
for (i = 0; i < SDL_num_touch; ++i) {
SDL_Touch *check;
if (i != index) {
239
check = SDL_touchPads[i];
240
241
242
243
244
245
246
247
248
249
250
251
if (check && check->focus == touch->focus) {
focus = SDL_TRUE;
break;
}
}
}
if (!focus) {
SDL_SendWindowEvent(touch->focus, SDL_WINDOWEVENT_ENTER, 0, 0);
}
}
}
252
int
253
SDL_AddFinger(SDL_Touch* touch,SDL_Finger finger)
254
255
256
{
int index;
SDL_Finger **fingers;
257
//printf("Adding Finger...\n");
258
if (SDL_GetFingerIndexId(touch,finger.id) != -1) {
259
SDL_SetError("Finger ID already in use");
260
}
261
262
/* Add the touch to the list of touch */
263
if(touch->num_fingers >= touch->max_fingers){
264
265
266
267
268
269
270
271
272
273
274
//printf("Making room for it!\n");
fingers = (SDL_Finger **) SDL_realloc(touch->fingers,
(touch->num_fingers + 1) * sizeof(SDL_Finger *));
touch->max_fingers = touch->num_fingers+1;
if (!fingers) {
SDL_OutOfMemory();
return -1;
} else {
touch->max_fingers = touch->num_fingers+1;
touch->fingers = fingers;
}
275
}
276
277
index = touch->num_fingers;
278
279
280
//printf("Max_Fingers: %i Index: %i\n",touch->max_fingers,index);
touch->fingers[index] = (SDL_Finger *) SDL_malloc(sizeof(SDL_Finger));
281
282
283
284
if (!touch->fingers[index]) {
SDL_OutOfMemory();
return -1;
}
285
*(touch->fingers[index]) = finger;
286
touch->num_fingers++;
287
288
289
290
return index;
}
291
int
292
SDL_DelFinger(SDL_Touch* touch,long fingerid)
293
{
294
int index = SDL_GetFingerIndexId(touch,fingerid);
295
SDL_Finger* finger = SDL_GetFinger(touch,fingerid);
296
297
if (!finger) {
298
return -1;
299
}
300
301
302
303
304
SDL_free(finger);
touch->num_fingers--;
touch->fingers[index] = touch->fingers[touch->num_fingers];
305
return 0;
306
307
308
309
}
int
310
SDL_SendFingerDown(long id, long fingerid, SDL_bool down, float x, float y, float pressure)
311
{
312
int posted;
313
SDL_Touch* touch = SDL_GetTouch(id);
314
315
316
317
318
if(!touch) {
return SDL_TouchNotFoundError(id);
}
319
if(down) {
320
321
322
323
324
325
326
327
328
329
330
331
332
SDL_Finger *finger = SDL_GetFinger(touch,fingerid);
if(finger == NULL) {
SDL_Finger nf;
nf.id = fingerid;
nf.x = x;
nf.y = y;
nf.pressure = pressure;
nf.xdelta = 0;
nf.ydelta = 0;
nf.last_x = x;
nf.last_y = y;
nf.last_pressure = pressure;
nf.down = SDL_FALSE;
333
SDL_AddFinger(touch,nf);
334
335
336
337
finger = &nf;
}
else if(finger->down) return 0;
if(x < 0 || y < 0) return 0; //should defer if only a partial input
338
339
340
341
posted = 0;
if (SDL_GetEventState(SDL_FINGERDOWN) == SDL_ENABLE) {
SDL_Event event;
event.tfinger.type = SDL_FINGERDOWN;
342
event.tfinger.touchId = id;
343
344
event.tfinger.x = x;
event.tfinger.y = y;
345
346
event.tfinger.state = touch->buttonstate;
event.tfinger.windowID = touch->focus ? touch->focus->id : 0;
347
event.tfinger.fingerId = fingerid;
348
349
posted = (SDL_PushEvent(&event) > 0);
}
350
if(posted) finger->down = SDL_TRUE;
351
352
353
return posted;
}
else {
354
if(SDL_DelFinger(touch,fingerid) < 0) return 0;
355
356
357
358
posted = 0;
if (SDL_GetEventState(SDL_FINGERUP) == SDL_ENABLE) {
SDL_Event event;
event.tfinger.type = SDL_FINGERUP;
359
event.tfinger.touchId = id;
360
361
event.tfinger.state = touch->buttonstate;
event.tfinger.windowID = touch->focus ? touch->focus->id : 0;
362
event.tfinger.fingerId = fingerid;
363
364
365
posted = (SDL_PushEvent(&event) > 0);
}
return posted;
366
367
368
369
}
}
int
370
371
SDL_SendTouchMotion(long id, long fingerid, int relative,
float x, float y, float pressure)
372
373
{
int index = SDL_GetTouchIndexId(id);
374
375
SDL_Touch *touch = SDL_GetTouch(id);
SDL_Finger *finger = SDL_GetFinger(touch,fingerid);
376
int posted;
377
378
379
float xrel;
float yrel;
float x_max = 0, y_max = 0;
380
381
382
383
384
if (!touch) {
return SDL_TouchNotFoundError(id);
}
if(touch->flush_motion) {
385
return 0;
386
387
}
388
389
if(finger == NULL || !finger->down) {
return SDL_SendFingerDown(id,fingerid,SDL_TRUE,x,y,pressure);
390
} else {
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
/* the relative motion is calculated regarding the last position */
if (relative) {
xrel = x;
yrel = y;
x = (finger->last_x + x);
y = (finger->last_y + y);
} else {
if(x < 0) x = finger->last_x; /*If movement is only in one axis,*/
if(y < 0) y = finger->last_y; /*The other is marked as -1*/
if(pressure < 0) pressure = finger->last_pressure;
xrel = x - finger->last_x;
yrel = y - finger->last_y;
}
/* Drop events that don't change state */
if (!xrel && !yrel) {
407
#if 0
408
printf("Touch event didn't change state - dropped!\n");
409
#endif
410
return 0;
411
}
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
/* Update internal touch coordinates */
finger->x = x;
finger->y = y;
/*Should scale to window? Normalize? Maintain Aspect?*/
//SDL_GetWindowSize(touch->focus, &x_max, &y_max);
/* make sure that the pointers find themselves inside the windows */
/* only check if touch->xmax is set ! */
/*
if (x_max && touch->x > x_max) {
touch->x = x_max;
} else if (touch->x < 0) {
touch->x = 0;
}
if (y_max && touch->y > y_max) {
touch->y = y_max;
} else if (touch->y < 0) {
touch->y = 0;
}
*/
finger->xdelta = xrel;
finger->ydelta = yrel;
finger->pressure = pressure;
/* Post the event, if desired */
posted = 0;
if (SDL_GetEventState(SDL_FINGERMOTION) == SDL_ENABLE) {
SDL_Event event;
event.tfinger.type = SDL_FINGERMOTION;
447
448
event.tfinger.touchId = id;
event.tfinger.fingerId = fingerid;
449
450
451
452
453
454
455
456
457
458
459
460
event.tfinger.x = x;
event.tfinger.y = y;
event.tfinger.pressure = pressure;
event.tfinger.state = touch->buttonstate;
event.tfinger.windowID = touch->focus ? touch->focus->id : 0;
posted = (SDL_PushEvent(&event) > 0);
}
finger->last_x = finger->x;
finger->last_y = finger->y;
finger->last_pressure = finger->pressure;
return posted;
}
461
462
}
int
463
SDL_SendTouchButton(long id, Uint8 state, Uint8 button)
464
{
465
SDL_Touch *touch = SDL_GetTouch(id);
466
467
468
int posted;
Uint32 type;
469
470
if (!touch) {
471
return SDL_TouchNotFoundError(id);
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
}
/* Figure out which event to perform */
switch (state) {
case SDL_PRESSED:
if (touch->buttonstate & SDL_BUTTON(button)) {
/* Ignore this event, no state change */
return 0;
}
type = SDL_TOUCHBUTTONDOWN;
touch->buttonstate |= SDL_BUTTON(button);
break;
case SDL_RELEASED:
if (!(touch->buttonstate & SDL_BUTTON(button))) {
/* Ignore this event, no state change */
return 0;
}
type = SDL_TOUCHBUTTONUP;
touch->buttonstate &= ~SDL_BUTTON(button);
break;
default:
/* Invalid state -- bail */
return 0;
}
/* Post the event, if desired */
posted = 0;
if (SDL_GetEventState(type) == SDL_ENABLE) {
SDL_Event event;
event.type = type;
502
event.tbutton.touchId = touch->id;
503
504
505
event.tbutton.state = state;
event.tbutton.button = button;
event.tbutton.windowID = touch->focus ? touch->focus->id : 0;
506
507
508
509
510
511
posted = (SDL_PushEvent(&event) > 0);
}
return posted;
}
char *
512
SDL_GetTouchName(long id)
513
{
514
SDL_Touch *touch = SDL_GetTouch(id);
515
516
517
518
519
520
if (!touch) {
return NULL;
}
return touch->name;
}
521
522
int SDL_TouchNotFoundError(long id) {
printf("ERROR: Cannot send touch on non-existent device with id: %li make sure SDL_AddTouch has been called\n",id);
523
524
525
printf("ERROR: There are %i touches installed with Id's:\n",SDL_num_touch);
int i;
for(i=0;i < SDL_num_touch;i++) {
526
printf("ERROR: %li\n",SDL_touchPads[i]->id);
527
528
529
}
return 0;
}
530
/* vi: set ts=4 sw=4 expandtab: */