This repository has been archived by the owner on Feb 11, 2021. It is now read-only.
/
SDL_cocoawindow.m
625 lines (531 loc) · 17.5 KB
1
2
/*
SDL - Simple DirectMedia Layer
3
Copyright (C) 1997-2009 Sam Lantinga
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
30
31
32
33
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"
#include "SDL_syswm.h"
#include "../SDL_sysvideo.h"
#include "../../events/SDL_keyboard_c.h"
#include "../../events/SDL_mouse_c.h"
#include "../../events/SDL_windowevents_c.h"
#include "SDL_cocoavideo.h"
static __inline__ void ConvertNSRect(NSRect *r)
{
34
/* FIXME: Cache the display used for this window */
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
r->origin.y = CGDisplayPixelsHigh(kCGDirectMainDisplay) - r->origin.y - r->size.height;
}
@implementation Cocoa_WindowListener
- (void)listen:(SDL_WindowData *)data
{
NSNotificationCenter *center;
_data = data;
center = [NSNotificationCenter defaultCenter];
[_data->window setNextResponder:self];
if ([_data->window delegate] != nil) {
[center addObserver:self selector:@selector(windowDisExpose:) name:NSWindowDidExposeNotification object:_data->window];
[center addObserver:self selector:@selector(windowDidMove:) name:NSWindowDidMoveNotification object:_data->window];
[center addObserver:self selector:@selector(windowDidResize:) name:NSWindowDidResizeNotification object:_data->window];
[center addObserver:self selector:@selector(windowDidMiniaturize:) name:NSWindowDidMiniaturizeNotification object:_data->window];
[center addObserver:self selector:@selector(windowDidDeminiaturize:) name:NSWindowDidDeminiaturizeNotification object:_data->window];
[center addObserver:self selector:@selector(windowDidBecomeKey:) name:NSWindowDidBecomeKeyNotification object:_data->window];
[center addObserver:self selector:@selector(windowDidResignKey:) name:NSWindowDidResignKeyNotification object:_data->window];
} else {
[_data->window setDelegate:self];
}
[center addObserver:self selector:@selector(windowDidHide:) name:NSApplicationDidHideNotification object:NSApp];
[center addObserver:self selector:@selector(windowDidUnhide:) name:NSApplicationDidUnhideNotification object:NSApp];
[_data->window setAcceptsMouseMovedEvents:YES];
}
- (void)close
{
NSNotificationCenter *center;
center = [NSNotificationCenter defaultCenter];
[_data->window setNextResponder:nil];
if ([_data->window delegate] != self) {
[center removeObserver:self name:NSWindowDidExposeNotification object:_data->window];
[center removeObserver:self name:NSWindowDidMoveNotification object:_data->window];
[center removeObserver:self name:NSWindowDidResizeNotification object:_data->window];
[center removeObserver:self name:NSWindowDidMiniaturizeNotification object:_data->window];
[center removeObserver:self name:NSWindowDidDeminiaturizeNotification object:_data->window];
[center removeObserver:self name:NSWindowDidBecomeKeyNotification object:_data->window];
[center removeObserver:self name:NSWindowDidResignKeyNotification object:_data->window];
} else {
[_data->window setDelegate:nil];
}
[center removeObserver:self name:NSApplicationDidHideNotification object:NSApp];
[center removeObserver:self name:NSApplicationDidUnhideNotification object:NSApp];
}
- (BOOL)windowShouldClose:(id)sender
{
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_CLOSE, 0, 0);
return NO;
}
- (void)windowDidExpose:(NSNotification *)aNotification
{
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_EXPOSED, 0, 0);
}
- (void)windowDidMove:(NSNotification *)aNotification
{
int x, y;
NSRect rect = [_data->window contentRectForFrameRect:[_data->window frame]];
ConvertNSRect(&rect);
x = (int)rect.origin.x;
y = (int)rect.origin.y;
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_MOVED, x, y);
}
- (void)windowDidResize:(NSNotification *)aNotification
{
int w, h;
NSRect rect = [_data->window contentRectForFrameRect:[_data->window frame]];
w = (int)rect.size.width;
h = (int)rect.size.height;
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_RESIZED, w, h);
}
- (void)windowDidMiniaturize:(NSNotification *)aNotification
{
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_MINIMIZED, 0, 0);
}
- (void)windowDidDeminiaturize:(NSNotification *)aNotification
{
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_RESTORED, 0, 0);
}
- (void)windowDidBecomeKey:(NSNotification *)aNotification
{
130
131
int index;
132
/* We're going to get keyboard events, since we're key. */
133
134
index = _data->videodata->keyboard;
SDL_SetKeyboardFocus(index, _data->windowID);
135
136
137
138
}
- (void)windowDidResignKey:(NSNotification *)aNotification
{
139
int index;
140
141
142
143
144
145
146
147
SDL_Mouse *mouse;
/* Some other window will get mouse events, since we're not key. */
index = _data->videodata->mouse;
mouse = SDL_GetMouse(index);
if (mouse->focus == _data->windowID) {
SDL_SetMouseFocus(index, 0);
}
148
149
/* Some other window will get keyboard events, since we're not key. */
150
151
index = _data->videodata->keyboard;
SDL_SetKeyboardFocus(index, 0);
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
}
- (void)windowDidHide:(NSNotification *)aNotification
{
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_HIDDEN, 0, 0);
}
- (void)windowDidUnhide:(NSNotification *)aNotification
{
SDL_SendWindowEvent(_data->windowID, SDL_WINDOWEVENT_SHOWN, 0, 0);
}
- (void)mouseDown:(NSEvent *)theEvent
{
int index;
167
int button;
168
169
index = _data->videodata->mouse;
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
switch ([theEvent buttonNumber]) {
case 0:
button = SDL_BUTTON_LEFT;
break;
case 1:
button = SDL_BUTTON_RIGHT;
break;
case 2:
button = SDL_BUTTON_MIDDLE;
break;
default:
button = [theEvent buttonNumber];
break;
}
SDL_SendMouseButton(index, SDL_PRESSED, button);
185
186
187
188
}
- (void)rightMouseDown:(NSEvent *)theEvent
{
189
[self mouseDown:theEvent];
190
191
192
193
}
- (void)otherMouseDown:(NSEvent *)theEvent
{
194
[self mouseDown:theEvent];
195
196
197
198
199
}
- (void)mouseUp:(NSEvent *)theEvent
{
int index;
200
int button;
201
202
index = _data->videodata->mouse;
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
switch ([theEvent buttonNumber]) {
case 0:
button = SDL_BUTTON_LEFT;
break;
case 1:
button = SDL_BUTTON_RIGHT;
break;
case 2:
button = SDL_BUTTON_MIDDLE;
break;
default:
button = [theEvent buttonNumber];
break;
}
SDL_SendMouseButton(index, SDL_RELEASED, button);
218
219
220
221
}
- (void)rightMouseUp:(NSEvent *)theEvent
{
222
[self mouseUp:theEvent];
223
224
225
226
}
- (void)otherMouseUp:(NSEvent *)theEvent
{
227
[self mouseUp:theEvent];
228
229
230
231
}
- (void)mouseMoved:(NSEvent *)theEvent
{
232
SDL_Window *window = SDL_GetWindowFromID(_data->windowID);
233
234
235
int index;
SDL_Mouse *mouse;
NSPoint point;
236
NSRect rect;
237
238
239
240
241
index = _data->videodata->mouse;
mouse = SDL_GetMouse(index);
point = [NSEvent mouseLocation];
242
243
244
if ( (window->flags & SDL_WINDOW_FULLSCREEN) ) {
rect.size.width = CGDisplayPixelsWide(kCGDirectMainDisplay);
rect.size.height = CGDisplayPixelsHigh(kCGDirectMainDisplay);
245
point.x = point.x - rect.origin.x;
246
247
248
249
250
point.y = rect.size.height - point.y;
} else {
rect = [_data->window contentRectForFrameRect:[_data->window frame]];
point.y = rect.size.height - (point.y - rect.origin.y);
}
251
252
253
254
255
256
257
258
259
if ( point.x < 0 || point.x >= rect.size.width ||
point.y < 0 || point.y >= rect.size.height ) {
if (mouse->focus != 0) {
SDL_SetMouseFocus(index, 0);
}
} else {
if (mouse->focus != _data->windowID) {
SDL_SetMouseFocus(index, _data->windowID);
}
260
SDL_SendMouseMotion(index, 0, (int)point.x, (int)point.y, 0);
261
}
262
263
}
264
265
266
267
268
- (void)mouseDragged:(NSEvent *)theEvent
{
[self mouseMoved:theEvent];
}
269
270
271
272
273
274
275
276
277
278
- (void)rightMouseDragged:(NSEvent *)theEvent
{
[self mouseMoved:theEvent];
}
- (void)otherMouseDragged:(NSEvent *)theEvent
{
[self mouseMoved:theEvent];
}
279
280
- (void)scrollWheel:(NSEvent *)theEvent
{
281
282
283
int index;
index = _data->videodata->mouse;
284
SDL_SendMouseWheel(index, (int)([theEvent deltaX]+0.9f), (int)([theEvent deltaY]+0.9f));
285
286
287
288
}
@end
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
@interface SDLWindow : NSWindow
/* These are needed for borderless/fullscreen windows */
- (BOOL)canBecomeKeyWindow;
- (BOOL)canBecomeMainWindow;
@end
@implementation SDLWindow
- (BOOL)canBecomeKeyWindow
{
return YES;
}
- (BOOL)canBecomeMainWindow
{
return YES;
}
@end
307
static int
308
SetupWindowData(_THIS, SDL_Window * window, NSWindow *nswindow, SDL_bool created)
309
310
{
NSAutoreleasePool *pool;
311
SDL_VideoData *videodata = (SDL_VideoData *) _this->driverdata;
312
313
314
315
316
317
318
319
320
321
322
SDL_WindowData *data;
/* Allocate the window data */
data = (SDL_WindowData *) SDL_malloc(sizeof(*data));
if (!data) {
SDL_OutOfMemory();
return -1;
}
data->windowID = window->id;
data->window = nswindow;
data->created = created;
323
data->videodata = videodata;
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
pool = [[NSAutoreleasePool alloc] init];
/* Create an event listener for the window */
data->listener = [[Cocoa_WindowListener alloc] init];
[data->listener listen:data];
/* Fill in the SDL window with the window data */
{
NSRect rect = [nswindow contentRectForFrameRect:[nswindow frame]];
ConvertNSRect(&rect);
window->x = (int)rect.origin.x;
window->y = (int)rect.origin.y;
window->w = (int)rect.size.width;
window->h = (int)rect.size.height;
}
if ([nswindow isVisible]) {
window->flags |= SDL_WINDOW_SHOWN;
} else {
window->flags &= ~SDL_WINDOW_SHOWN;
}
{
unsigned int style = [nswindow styleMask];
348
if ((style & ~NSResizableWindowMask) == NSBorderlessWindowMask) {
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
window->flags |= SDL_WINDOW_BORDERLESS;
} else {
window->flags &= ~SDL_WINDOW_BORDERLESS;
}
if (style & NSResizableWindowMask) {
window->flags |= SDL_WINDOW_RESIZABLE;
} else {
window->flags &= ~SDL_WINDOW_RESIZABLE;
}
}
if ([nswindow isZoomed]) {
window->flags |= SDL_WINDOW_MAXIMIZED;
} else {
window->flags &= ~SDL_WINDOW_MAXIMIZED;
}
if ([nswindow isMiniaturized]) {
window->flags |= SDL_WINDOW_MINIMIZED;
} else {
window->flags &= ~SDL_WINDOW_MINIMIZED;
}
if ([nswindow isKeyWindow]) {
int index = data->videodata->keyboard;
window->flags |= SDL_WINDOW_INPUT_FOCUS;
SDL_SetKeyboardFocus(index, data->windowID);
if (window->flags & SDL_WINDOW_INPUT_GRABBED) {
/* FIXME */
}
}
/* All done! */
[pool release];
window->driverdata = data;
return 0;
}
int
Cocoa_CreateWindow(_THIS, SDL_Window * window)
{
NSAutoreleasePool *pool;
NSWindow *nswindow;
NSRect rect;
unsigned int style;
NSString *title;
393
int status;
394
395
396
pool = [[NSAutoreleasePool alloc] init];
397
398
if ((window->flags & SDL_WINDOW_FULLSCREEN)
|| window->x == SDL_WINDOWPOS_CENTERED) {
399
400
401
402
403
404
rect.origin.x = (CGDisplayPixelsWide(kCGDirectMainDisplay) - window->w) / 2;
} else if (window->x == SDL_WINDOWPOS_UNDEFINED) {
rect.origin.x = 0;
} else {
rect.origin.x = window->x;
}
405
406
if ((window->flags & SDL_WINDOW_FULLSCREEN)
|| window->y == SDL_WINDOWPOS_CENTERED) {
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
rect.origin.y = (CGDisplayPixelsHigh(kCGDirectMainDisplay) - window->h) / 2;
} else if (window->y == SDL_WINDOWPOS_UNDEFINED) {
rect.origin.y = 0;
} else {
rect.origin.y = window->y;
}
rect.size.width = window->w;
rect.size.height = window->h;
ConvertNSRect(&rect);
if (window->flags & SDL_WINDOW_BORDERLESS) {
style = NSBorderlessWindowMask;
} else {
style = (NSTitledWindowMask|NSClosableWindowMask|NSMiniaturizableWindowMask);
}
if (window->flags & SDL_WINDOW_RESIZABLE) {
style |= NSResizableWindowMask;
}
426
nswindow = [[SDLWindow alloc] initWithContentRect:rect styleMask:style backing:NSBackingStoreBuffered defer:FALSE];
427
428
429
[pool release];
430
if (SetupWindowData(_this, window, nswindow, SDL_TRUE) < 0) {
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
[nswindow release];
return -1;
}
return 0;
}
int
Cocoa_CreateWindowFrom(_THIS, SDL_Window * window, const void *data)
{
NSAutoreleasePool *pool;
NSWindow *nswindow = (NSWindow *) data;
NSString *title;
int status;
pool = [[NSAutoreleasePool alloc] init];
/* Query the title from the existing window */
title = [nswindow title];
if (title) {
window->title = SDL_strdup([title UTF8String]);
}
[pool release];
455
return SetupWindowData(_this, window, nswindow, SDL_FALSE);
456
457
458
459
460
}
void
Cocoa_SetWindowTitle(_THIS, SDL_Window * window)
{
461
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
462
463
464
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
NSString *string;
465
466
467
468
469
if(window->title) {
string = [[NSString alloc] initWithUTF8String:window->title];
} else {
string = [[NSString alloc] init];
}
470
471
[nswindow setTitle:string];
[string release];
472
473
[pool release];
474
475
476
477
478
}
void
Cocoa_SetWindowPosition(_THIS, SDL_Window * window)
{
479
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
480
481
482
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
NSRect rect;
483
484
485
486
487
488
489
490
491
492
493
494
if ((window->flags & SDL_WINDOW_FULLSCREEN)
|| window->x == SDL_WINDOWPOS_CENTERED) {
rect.origin.x = (CGDisplayPixelsWide(kCGDirectMainDisplay) - window->w) / 2;
} else {
rect.origin.x = window->x;
}
if ((window->flags & SDL_WINDOW_FULLSCREEN)
|| window->y == SDL_WINDOWPOS_CENTERED) {
rect.origin.y = (CGDisplayPixelsHigh(kCGDirectMainDisplay) - window->h) / 2;
} else {
rect.origin.y = window->y;
}
495
496
497
498
499
rect.size.width = window->w;
rect.size.height = window->h;
ConvertNSRect(&rect);
rect = [nswindow frameRectForContentRect:rect];
[nswindow setFrameOrigin:rect.origin];
500
[pool release];
501
502
503
504
505
}
void
Cocoa_SetWindowSize(_THIS, SDL_Window * window)
{
506
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
507
508
509
510
511
512
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
NSSize size;
size.width = window->w;
size.height = window->h;
[nswindow setContentSize:size];
513
[pool release];
514
515
516
517
518
}
void
Cocoa_ShowWindow(_THIS, SDL_Window * window)
{
519
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
520
521
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
522
523
524
if (![nswindow isMiniaturized]) {
[nswindow makeKeyAndOrderFront:nil];
}
525
[pool release];
526
527
528
529
530
}
void
Cocoa_HideWindow(_THIS, SDL_Window * window)
{
531
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
532
533
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
534
535
[nswindow orderOut:nil];
[pool release];
536
537
538
539
540
}
void
Cocoa_RaiseWindow(_THIS, SDL_Window * window)
{
541
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
542
543
544
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
[nswindow makeKeyAndOrderFront:nil];
545
[pool release];
546
547
548
549
550
}
void
Cocoa_MaximizeWindow(_THIS, SDL_Window * window)
{
551
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
552
553
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
554
[nswindow zoom:nil];
555
[pool release];
556
557
558
559
560
}
void
Cocoa_MinimizeWindow(_THIS, SDL_Window * window)
{
561
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
562
563
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
564
[nswindow miniaturize:nil];
565
[pool release];
566
567
568
569
570
}
void
Cocoa_RestoreWindow(_THIS, SDL_Window * window)
{
571
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
572
573
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
574
575
576
577
578
579
if ([nswindow isMiniaturized]) {
[nswindow deminiaturize:nil];
} else if ([nswindow isZoomed]) {
[nswindow zoom:nil];
}
[pool release];
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
}
void
Cocoa_SetWindowGrab(_THIS, SDL_Window * window)
{
if ((window->flags & SDL_WINDOW_INPUT_GRABBED) &&
(window->flags & SDL_WINDOW_INPUT_FOCUS)) {
/* FIXME: Grab mouse */
} else {
/* FIXME: Release mouse */
}
}
void
Cocoa_DestroyWindow(_THIS, SDL_Window * window)
{
596
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
597
598
599
600
601
602
603
604
605
606
SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
if (data) {
[data->listener close];
[data->listener release];
if (data->created) {
[data->window close];
}
SDL_free(data);
}
607
[pool release];
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
}
SDL_bool
Cocoa_GetWindowWMInfo(_THIS, SDL_Window * window, SDL_SysWMinfo * info)
{
NSWindow *nswindow = ((SDL_WindowData *) window->driverdata)->window;
if (info->version.major <= SDL_MAJOR_VERSION) {
//info->window = nswindow;
return SDL_TRUE;
} else {
SDL_SetError("Application not compiled with SDL %d.%d\n",
SDL_MAJOR_VERSION, SDL_MINOR_VERSION);
return SDL_FALSE;
}
}
/* vi: set ts=4 sw=4 expandtab: */