This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_uikitview.m
416 lines (342 loc) · 11.6 KB
1
/*
2
Simple DirectMedia Layer
3
Copyright (C) 1997-2012 Sam Lantinga <slouken@libsdl.org>
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
This software is provided 'as-is', without any express or implied
warranty. In no event will the authors be held liable for any damages
arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it
freely, subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not
claim that you wrote the original software. If you use this software
in a product, an acknowledgment in the product documentation would be
appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be
misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
*/
21
22
23
#include "SDL_config.h"
#if SDL_VIDEO_DRIVER_UIKIT
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
#import "SDL_uikitview.h"
#include "../../events/SDL_keyboard_c.h"
#include "../../events/SDL_mouse_c.h"
#include "../../events/SDL_touch_c.h"
#if SDL_IPHONE_KEYBOARD
#import "keyinfotable.h"
#import "SDL_uikitappdelegate.h"
#import "SDL_uikitwindow.h"
#endif
@implementation SDL_uikitview
39
40
- (void)dealloc
{
41
42
43
[super dealloc];
}
44
45
- (id)initWithFrame:(CGRect)frame
{
46
self = [super initWithFrame: frame];
47
48
49
#if SDL_IPHONE_KEYBOARD
[self initializeKeyboard];
50
#endif
51
52
53
54
55
56
57
58
59
#ifdef FIXED_MULTITOUCH
self.multipleTouchEnabled = YES;
SDL_Touch touch;
touch.id = 0; //TODO: Should be -1?
//touch.driverdata = SDL_malloc(sizeof(EventTouchData));
//EventTouchData* data = (EventTouchData*)(touch.driverdata);
60
61
touch.x_min = 0;
62
touch.x_max = 1;
63
64
touch.native_xres = touch.x_max - touch.x_min;
touch.y_min = 0;
65
touch.y_max = 1;
66
67
68
69
70
71
72
73
74
75
76
77
78
touch.native_yres = touch.y_max - touch.y_min;
touch.pressure_min = 0;
touch.pressure_max = 1;
touch.native_pressureres = touch.pressure_max - touch.pressure_min;
touchId = SDL_AddTouch(&touch, "IPHONE SCREEN");
#endif
return self;
}
79
- (CGPoint)touchLocation:(UITouch *)touch shouldNormalize:(BOOL)normalize
80
81
{
CGPoint point = [touch locationInView: self];
82
83
84
85
86
87
88
89
90
// Get the display scale and apply that to the input coordinates
SDL_Window *window = self->viewcontroller.window;
SDL_VideoDisplay *display = SDL_GetDisplayForWindow(window);
SDL_DisplayModeData *displaymodedata = (SDL_DisplayModeData *) display->current_mode.driverdata;
point.x *= displaymodedata->scale;
point.y *= displaymodedata->scale;
if (normalize) {
91
92
93
CGRect bounds = [self bounds];
point.x /= bounds.size.width;
point.y /= bounds.size.height;
94
}
95
96
97
return point;
}
98
99
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
100
101
102
103
NSEnumerator *enumerator = [touches objectEnumerator];
UITouch *touch = (UITouch*)[enumerator nextObject];
if (touch) {
104
CGPoint locationInView = [self touchLocation:touch shouldNormalize:NO];
105
106
107
108
109
110
111
112
113
114
/* send moved event */
SDL_SendMouseMotion(NULL, 0, locationInView.x, locationInView.y);
/* send mouse down event */
SDL_SendMouseButton(NULL, SDL_PRESSED, SDL_BUTTON_LEFT);
}
#ifdef FIXED_MULTITOUCH
while(touch) {
115
CGPoint locationInView = [self touchLocation:touch shouldNormalize:YES];
116
117
#ifdef IPHONE_TOUCH_EFFICIENT_DANGEROUS
118
119
120
121
122
123
//FIXME: TODO: Using touch as the fingerId is potentially dangerous
//It is also much more efficient than storing the UITouch pointer
//and comparing it to the incoming event.
SDL_SendFingerDown(touchId, (long)touch,
SDL_TRUE, locationInView.x, locationInView.y,
1);
124
#else
125
126
127
128
129
130
131
132
133
int i;
for(i = 0; i < MAX_SIMULTANEOUS_TOUCHES; i++) {
if (finger[i] == NULL) {
finger[i] = touch;
SDL_SendFingerDown(touchId, i,
SDL_TRUE, locationInView.x, locationInView.y,
1);
break;
}
134
135
136
}
#endif
137
touch = (UITouch*)[enumerator nextObject];
138
139
140
141
}
#endif
}
142
143
- (void)touchesEnded:(NSSet *)touches withEvent:(UIEvent *)event
{
144
145
NSEnumerator *enumerator = [touches objectEnumerator];
UITouch *touch = (UITouch*)[enumerator nextObject];
146
147
148
149
150
151
152
153
if (touch) {
/* send mouse up */
SDL_SendMouseButton(NULL, SDL_RELEASED, SDL_BUTTON_LEFT);
}
#ifdef FIXED_MULTITOUCH
while(touch) {
154
CGPoint locationInView = [self touchLocation:touch shouldNormalize:YES];
155
156
#ifdef IPHONE_TOUCH_EFFICIENT_DANGEROUS
157
158
159
SDL_SendFingerDown(touchId, (long)touch,
SDL_FALSE, locationInView.x, locationInView.y,
1);
160
#else
161
162
163
164
165
166
167
168
169
int i;
for (i = 0; i < MAX_SIMULTANEOUS_TOUCHES; i++) {
if (finger[i] == touch) {
SDL_SendFingerDown(touchId, i,
SDL_FALSE, locationInView.x, locationInView.y,
1);
finger[i] = NULL;
break;
}
170
171
172
}
#endif
173
touch = (UITouch*)[enumerator nextObject];
174
175
176
177
}
#endif
}
178
179
- (void)touchesCancelled:(NSSet *)touches withEvent:(UIEvent *)event
{
180
181
182
183
184
185
186
187
/*
this can happen if the user puts more than 5 touches on the screen
at once, or perhaps in other circumstances. Usually (it seems)
all active touches are canceled.
*/
[self touchesEnded: touches withEvent: event];
}
188
189
- (void)touchesMoved:(NSSet *)touches withEvent:(UIEvent *)event
{
190
191
NSEnumerator *enumerator = [touches objectEnumerator];
UITouch *touch = (UITouch*)[enumerator nextObject];
192
193
if (touch) {
194
CGPoint locationInView = [self touchLocation:touch shouldNormalize:NO];
195
196
197
198
199
200
201
/* send moved event */
SDL_SendMouseMotion(NULL, 0, locationInView.x, locationInView.y);
}
#ifdef FIXED_MULTITOUCH
while(touch) {
202
CGPoint locationInView = [self touchLocation:touch shouldNormalize:YES];
203
204
#ifdef IPHONE_TOUCH_EFFICIENT_DANGEROUS
205
206
207
SDL_SendTouchMotion(touchId, (long)touch,
SDL_FALSE, locationInView.x, locationInView.y,
1);
208
#else
209
210
211
212
213
214
215
216
int i;
for (i = 0; i < MAX_SIMULTANEOUS_TOUCHES; i++) {
if (finger[i] == touch) {
SDL_SendTouchMotion(touchId, i,
SDL_FALSE, locationInView.x, locationInView.y,
1);
break;
}
217
218
219
}
#endif
220
touch = (UITouch*)[enumerator nextObject];
221
222
223
224
225
226
227
228
229
230
}
#endif
}
/*
---- Keyboard related functionality below this line ----
*/
#if SDL_IPHONE_KEYBOARD
/* Is the iPhone virtual keyboard visible onscreen? */
231
232
- (BOOL)keyboardVisible
{
233
234
235
236
return keyboardVisible;
}
/* Set ourselves up as a UITextFieldDelegate */
237
238
- (void)initializeKeyboard
{
239
240
241
textField = [[UITextField alloc] initWithFrame: CGRectZero];
textField.delegate = self;
/* placeholder so there is something to delete! */
242
243
textField.text = @" ";
244
245
246
247
248
249
250
/* set UITextInputTrait properties, mostly to defaults */
textField.autocapitalizationType = UITextAutocapitalizationTypeNone;
textField.autocorrectionType = UITextAutocorrectionTypeNo;
textField.enablesReturnKeyAutomatically = NO;
textField.keyboardAppearance = UIKeyboardAppearanceDefault;
textField.keyboardType = UIKeyboardTypeDefault;
textField.returnKeyType = UIReturnKeyDefault;
251
252
textField.secureTextEntry = NO;
253
254
255
256
257
258
259
260
textField.hidden = YES;
keyboardVisible = NO;
/* add the UITextField (hidden) to our view */
[self addSubview: textField];
[textField release];
}
/* reveal onscreen virtual keyboard */
261
262
- (void)showKeyboard
{
263
264
265
266
267
keyboardVisible = YES;
[textField becomeFirstResponder];
}
/* hide onscreen virtual keyboard */
268
269
- (void)hideKeyboard
{
270
271
272
273
274
keyboardVisible = NO;
[textField resignFirstResponder];
}
/* UITextFieldDelegate method. Invoked when user types something. */
275
276
- (BOOL)textField:(UITextField *)_textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string
{
277
278
279
280
281
282
283
284
285
if ([string length] == 0) {
/* it wants to replace text with nothing, ie a delete */
SDL_SendKeyboardKey(SDL_PRESSED, SDL_SCANCODE_DELETE);
SDL_SendKeyboardKey(SDL_RELEASED, SDL_SCANCODE_DELETE);
}
else {
/* go through all the characters in the string we've been sent
and convert them to key presses */
int i;
286
for (i = 0; i < [string length]; i++) {
287
288
unichar c = [string characterAtIndex: i];
289
290
291
Uint16 mod = 0;
SDL_Scancode code;
292
293
294
295
296
297
298
299
300
301
302
if (c < 127) {
/* figure out the SDL_Scancode and SDL_keymod for this unichar */
code = unicharToUIKeyInfoTable[c].code;
mod = unicharToUIKeyInfoTable[c].mod;
}
else {
/* we only deal with ASCII right now */
code = SDL_SCANCODE_UNKNOWN;
mod = 0;
}
303
304
305
306
307
308
309
310
311
312
313
if (mod & KMOD_SHIFT) {
/* If character uses shift, press shift down */
SDL_SendKeyboardKey(SDL_PRESSED, SDL_SCANCODE_LSHIFT);
}
/* send a keydown and keyup even for the character */
SDL_SendKeyboardKey(SDL_PRESSED, code);
SDL_SendKeyboardKey(SDL_RELEASED, code);
if (mod & KMOD_SHIFT) {
/* If character uses shift, press shift back up */
SDL_SendKeyboardKey(SDL_RELEASED, SDL_SCANCODE_LSHIFT);
314
}
315
316
317
318
319
320
321
}
SDL_SendKeyboardText([string UTF8String]);
}
return NO; /* don't allow the edit! (keep placeholder text there) */
}
/* Terminates the editing session */
322
323
- (BOOL)textFieldShouldReturn:(UITextField*)_textField
{
324
SDL_SendKeyboardKey(SDL_PRESSED, SDL_SCANCODE_RETURN);
325
SDL_SendKeyboardKey(SDL_RELEASED, SDL_SCANCODE_RETURN);
326
327
328
329
330
331
332
333
334
335
336
[self hideKeyboard];
return YES;
}
#endif
@end
/* iPhone keyboard addition functions */
#if SDL_IPHONE_KEYBOARD
337
static SDL_uikitview * getWindowView(SDL_Window * window)
338
{
339
if (window == NULL) {
340
SDL_SetError("Window does not exist");
341
return nil;
342
}
343
344
345
SDL_WindowData *data = (SDL_WindowData *)window->driverdata;
SDL_uikitview *view = data != NULL ? data->view : nil;
346
347
if (view == nil) {
348
349
SDL_SetError("Window has no view");
}
350
351
return view;
352
353
}
354
355
356
357
358
359
360
361
362
363
364
SDL_bool UIKit_HasScreenKeyboardSupport(_THIS, SDL_Window *window)
{
SDL_uikitview *view = getWindowView(window);
if (view == nil) {
return SDL_FALSE;
}
return SDL_TRUE;
}
int UIKit_ShowScreenKeyboard(_THIS, SDL_Window *window)
365
{
366
367
SDL_uikitview *view = getWindowView(window);
if (view == nil) {
368
return -1;
369
370
}
371
372
[view showKeyboard];
return 0;
373
374
}
375
int UIKit_HideScreenKeyboard(_THIS, SDL_Window *window)
376
{
377
378
SDL_uikitview *view = getWindowView(window);
if (view == nil) {
379
return -1;
380
381
}
382
383
384
[view hideKeyboard];
return 0;
}
385
386
SDL_bool UIKit_IsScreenKeyboardShown(_THIS, SDL_Window *window)
387
388
389
{
SDL_uikitview *view = getWindowView(window);
if (view == nil) {
390
391
return 0;
}
392
393
return view.keyboardVisible;
394
395
}
396
int UIKit_ToggleScreenKeyboard(_THIS, SDL_Window *window)
397
{
398
399
SDL_uikitview *view = getWindowView(window);
if (view == nil) {
400
return -1;
401
402
}
403
404
if (UIKit_IsScreenKeyboardShown(_this, window)) {
UIKit_HideScreenKeyboard(_this, window);
405
406
}
else {
407
UIKit_ShowScreenKeyboard(_this, window);
408
}
409
return 0;
410
411
412
413
}
#endif /* SDL_IPHONE_KEYBOARD */
414
415
#endif /* SDL_VIDEO_DRIVER_UIKIT */
416
/* vi: set ts=4 sw=4 expandtab: */