2015-06-21 15:33:46 +00:00
|
|
|
/*
|
|
|
|
Simple DirectMedia Layer
|
2016-01-02 18:10:34 +00:00
|
|
|
Copyright (C) 1997-2016 Sam Lantinga <slouken@libsdl.org>
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
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.
|
|
|
|
*/
|
|
|
|
#include "../../SDL_internal.h"
|
|
|
|
|
|
|
|
#if SDL_VIDEO_DRIVER_COCOA
|
|
|
|
#include "SDL_timer.h"
|
|
|
|
|
|
|
|
#include "SDL_cocoavideo.h"
|
|
|
|
#include "../../events/SDL_events_c.h"
|
|
|
|
#include "SDL_assert.h"
|
2015-07-30 00:19:15 +00:00
|
|
|
#include "SDL_hints.h"
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
/* This define was added in the 10.9 SDK. */
|
|
|
|
#ifndef kIOPMAssertPreventUserIdleDisplaySleep
|
|
|
|
#define kIOPMAssertPreventUserIdleDisplaySleep kIOPMAssertionTypePreventUserIdleDisplaySleep
|
|
|
|
#endif
|
|
|
|
|
|
|
|
@interface SDLApplication : NSApplication
|
|
|
|
|
|
|
|
- (void)terminate:(id)sender;
|
2016-10-01 21:34:52 +00:00
|
|
|
- (void)sendEvent:(NSEvent *)theEvent;
|
2015-06-21 15:33:46 +00:00
|
|
|
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation SDLApplication
|
|
|
|
|
|
|
|
// Override terminate to handle Quit and System Shutdown smoothly.
|
|
|
|
- (void)terminate:(id)sender
|
|
|
|
{
|
|
|
|
SDL_SendQuit();
|
|
|
|
}
|
|
|
|
|
2016-10-14 15:15:39 +00:00
|
|
|
static SDL_bool s_bShouldHandleEventsInSDLApplication = SDL_FALSE;
|
|
|
|
|
|
|
|
static void Cocoa_DispatchEvent(NSEvent *theEvent)
|
2016-10-01 21:34:52 +00:00
|
|
|
{
|
|
|
|
SDL_VideoDevice *_this = SDL_GetVideoDevice();
|
2016-10-14 15:15:39 +00:00
|
|
|
|
2016-10-01 21:34:52 +00:00
|
|
|
switch ([theEvent type]) {
|
|
|
|
case NSLeftMouseDown:
|
|
|
|
case NSOtherMouseDown:
|
|
|
|
case NSRightMouseDown:
|
|
|
|
case NSLeftMouseUp:
|
|
|
|
case NSOtherMouseUp:
|
|
|
|
case NSRightMouseUp:
|
|
|
|
case NSLeftMouseDragged:
|
|
|
|
case NSRightMouseDragged:
|
|
|
|
case NSOtherMouseDragged: /* usually middle mouse dragged */
|
|
|
|
case NSMouseMoved:
|
|
|
|
case NSScrollWheel:
|
|
|
|
Cocoa_HandleMouseEvent(_this, theEvent);
|
|
|
|
break;
|
|
|
|
case NSKeyDown:
|
|
|
|
case NSKeyUp:
|
|
|
|
case NSFlagsChanged:
|
|
|
|
Cocoa_HandleKeyEvent(_this, theEvent);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
2016-10-14 15:15:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Dispatch events here so that we can handle events caught by
|
|
|
|
// nextEventMatchingMask in SDL, as well as events caught by other
|
|
|
|
// processes (such as CEF) that are passed down to NSApp.
|
|
|
|
- (void)sendEvent:(NSEvent *)theEvent
|
|
|
|
{
|
|
|
|
if (s_bShouldHandleEventsInSDLApplication) {
|
|
|
|
Cocoa_DispatchEvent(theEvent);
|
|
|
|
}
|
|
|
|
|
2016-10-01 21:34:52 +00:00
|
|
|
[super sendEvent:theEvent];
|
|
|
|
}
|
|
|
|
|
2015-06-21 15:33:46 +00:00
|
|
|
@end // SDLApplication
|
|
|
|
|
|
|
|
/* setAppleMenu disappeared from the headers in 10.4 */
|
|
|
|
@interface NSApplication(NSAppleMenu)
|
|
|
|
- (void)setAppleMenu:(NSMenu *)menu;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@interface SDLAppDelegate : NSObject <NSApplicationDelegate> {
|
|
|
|
@public
|
|
|
|
BOOL seenFirstActivate;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (id)init;
|
|
|
|
@end
|
|
|
|
|
|
|
|
@implementation SDLAppDelegate : NSObject
|
|
|
|
- (id)init
|
|
|
|
{
|
|
|
|
self = [super init];
|
|
|
|
if (self) {
|
|
|
|
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
|
|
|
|
|
|
|
|
seenFirstActivate = NO;
|
|
|
|
|
|
|
|
[center addObserver:self
|
|
|
|
selector:@selector(windowWillClose:)
|
|
|
|
name:NSWindowWillCloseNotification
|
|
|
|
object:nil];
|
|
|
|
|
|
|
|
[center addObserver:self
|
|
|
|
selector:@selector(focusSomeWindow:)
|
|
|
|
name:NSApplicationDidBecomeActiveNotification
|
|
|
|
object:nil];
|
|
|
|
}
|
|
|
|
|
|
|
|
return self;
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)dealloc
|
|
|
|
{
|
|
|
|
NSNotificationCenter *center = [NSNotificationCenter defaultCenter];
|
|
|
|
|
|
|
|
[center removeObserver:self name:NSWindowWillCloseNotification object:nil];
|
|
|
|
[center removeObserver:self name:NSApplicationDidBecomeActiveNotification object:nil];
|
|
|
|
|
|
|
|
[super dealloc];
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)windowWillClose:(NSNotification *)notification;
|
|
|
|
{
|
|
|
|
NSWindow *win = (NSWindow*)[notification object];
|
|
|
|
|
|
|
|
if (![win isKeyWindow]) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* HACK: Make the next window in the z-order key when the key window is
|
|
|
|
* closed. The custom event loop and/or windowing code we have seems to
|
|
|
|
* prevent the normal behavior: https://bugzilla.libsdl.org/show_bug.cgi?id=1825
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* +[NSApp orderedWindows] never includes the 'About' window, but we still
|
|
|
|
* want to try its list first since the behavior in other apps is to only
|
|
|
|
* make the 'About' window key if no other windows are on-screen.
|
|
|
|
*/
|
|
|
|
for (NSWindow *window in [NSApp orderedWindows]) {
|
|
|
|
if (window != win && [window canBecomeKeyWindow]) {
|
2016-05-21 03:20:52 +00:00
|
|
|
if (![window isOnActiveSpace]) {
|
|
|
|
continue;
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
[window makeKeyAndOrderFront:self];
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-05-21 03:20:52 +00:00
|
|
|
/* If a window wasn't found above, iterate through all visible windows in
|
|
|
|
* the active Space in z-order (including the 'About' window, if it's shown)
|
|
|
|
* and make the first one key.
|
2015-06-21 15:33:46 +00:00
|
|
|
*/
|
2016-05-21 03:20:52 +00:00
|
|
|
for (NSNumber *num in [NSWindow windowNumbersWithOptions:0]) {
|
|
|
|
NSWindow *window = [NSApp windowWithWindowNumber:[num integerValue]];
|
|
|
|
if (window && window != win && [window canBecomeKeyWindow]) {
|
|
|
|
[window makeKeyAndOrderFront:self];
|
|
|
|
return;
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (void)focusSomeWindow:(NSNotification *)aNotification
|
|
|
|
{
|
|
|
|
/* HACK: Ignore the first call. The application gets a
|
|
|
|
* applicationDidBecomeActive: a little bit after the first window is
|
|
|
|
* created, and if we don't ignore it, a window that has been created with
|
|
|
|
* SDL_WINDOW_MINIMIZED will ~immediately be restored.
|
|
|
|
*/
|
|
|
|
if (!seenFirstActivate) {
|
|
|
|
seenFirstActivate = YES;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
SDL_VideoDevice *device = SDL_GetVideoDevice();
|
|
|
|
if (device && device->windows) {
|
|
|
|
SDL_Window *window = device->windows;
|
|
|
|
int i;
|
|
|
|
for (i = 0; i < device->num_displays; ++i) {
|
|
|
|
SDL_Window *fullscreen_window = device->displays[i].fullscreen_window;
|
|
|
|
if (fullscreen_window) {
|
|
|
|
if (fullscreen_window->flags & SDL_WINDOW_MINIMIZED) {
|
|
|
|
SDL_RestoreWindow(fullscreen_window);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (window->flags & SDL_WINDOW_MINIMIZED) {
|
|
|
|
SDL_RestoreWindow(window);
|
|
|
|
} else {
|
|
|
|
SDL_RaiseWindow(window);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
- (BOOL)application:(NSApplication *)theApplication openFile:(NSString *)filename
|
|
|
|
{
|
2016-01-05 06:42:00 +00:00
|
|
|
return (BOOL)SDL_SendDropFile(NULL, [filename UTF8String]) && SDL_SendDropComplete(NULL);
|
2015-06-21 15:33:46 +00:00
|
|
|
}
|
|
|
|
@end
|
|
|
|
|
|
|
|
static SDLAppDelegate *appDelegate = nil;
|
|
|
|
|
|
|
|
static NSString *
|
|
|
|
GetApplicationName(void)
|
|
|
|
{
|
|
|
|
NSString *appName;
|
|
|
|
|
|
|
|
/* Determine the application name */
|
|
|
|
appName = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleDisplayName"];
|
|
|
|
if (!appName) {
|
|
|
|
appName = [[NSBundle mainBundle] objectForInfoDictionaryKey:@"CFBundleName"];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (![appName length]) {
|
|
|
|
appName = [[NSProcessInfo processInfo] processName];
|
|
|
|
}
|
|
|
|
|
|
|
|
return appName;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void
|
|
|
|
CreateApplicationMenus(void)
|
|
|
|
{
|
|
|
|
NSString *appName;
|
|
|
|
NSString *title;
|
|
|
|
NSMenu *appleMenu;
|
|
|
|
NSMenu *serviceMenu;
|
|
|
|
NSMenu *windowMenu;
|
|
|
|
NSMenu *viewMenu;
|
|
|
|
NSMenuItem *menuItem;
|
|
|
|
NSMenu *mainMenu;
|
|
|
|
|
|
|
|
if (NSApp == nil) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
mainMenu = [[NSMenu alloc] init];
|
|
|
|
|
|
|
|
/* Create the main menu bar */
|
|
|
|
[NSApp setMainMenu:mainMenu];
|
|
|
|
|
|
|
|
[mainMenu release]; /* we're done with it, let NSApp own it. */
|
|
|
|
mainMenu = nil;
|
|
|
|
|
|
|
|
/* Create the application menu */
|
|
|
|
appName = GetApplicationName();
|
|
|
|
appleMenu = [[NSMenu alloc] initWithTitle:@""];
|
|
|
|
|
|
|
|
/* Add menu items */
|
|
|
|
title = [@"About " stringByAppendingString:appName];
|
|
|
|
[appleMenu addItemWithTitle:title action:@selector(orderFrontStandardAboutPanel:) keyEquivalent:@""];
|
|
|
|
|
|
|
|
[appleMenu addItem:[NSMenuItem separatorItem]];
|
|
|
|
|
|
|
|
[appleMenu addItemWithTitle:@"Preferences…" action:nil keyEquivalent:@","];
|
|
|
|
|
|
|
|
[appleMenu addItem:[NSMenuItem separatorItem]];
|
|
|
|
|
|
|
|
serviceMenu = [[NSMenu alloc] initWithTitle:@""];
|
|
|
|
menuItem = (NSMenuItem *)[appleMenu addItemWithTitle:@"Services" action:nil keyEquivalent:@""];
|
|
|
|
[menuItem setSubmenu:serviceMenu];
|
|
|
|
|
|
|
|
[NSApp setServicesMenu:serviceMenu];
|
|
|
|
[serviceMenu release];
|
|
|
|
|
|
|
|
[appleMenu addItem:[NSMenuItem separatorItem]];
|
|
|
|
|
|
|
|
title = [@"Hide " stringByAppendingString:appName];
|
|
|
|
[appleMenu addItemWithTitle:title action:@selector(hide:) keyEquivalent:@"h"];
|
|
|
|
|
|
|
|
menuItem = (NSMenuItem *)[appleMenu addItemWithTitle:@"Hide Others" action:@selector(hideOtherApplications:) keyEquivalent:@"h"];
|
|
|
|
[menuItem setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
|
|
|
|
|
|
|
|
[appleMenu addItemWithTitle:@"Show All" action:@selector(unhideAllApplications:) keyEquivalent:@""];
|
|
|
|
|
|
|
|
[appleMenu addItem:[NSMenuItem separatorItem]];
|
|
|
|
|
|
|
|
title = [@"Quit " stringByAppendingString:appName];
|
|
|
|
[appleMenu addItemWithTitle:title action:@selector(terminate:) keyEquivalent:@"q"];
|
|
|
|
|
|
|
|
/* Put menu into the menubar */
|
|
|
|
menuItem = [[NSMenuItem alloc] initWithTitle:@"" action:nil keyEquivalent:@""];
|
|
|
|
[menuItem setSubmenu:appleMenu];
|
|
|
|
[[NSApp mainMenu] addItem:menuItem];
|
|
|
|
[menuItem release];
|
|
|
|
|
|
|
|
/* Tell the application object that this is now the application menu */
|
|
|
|
[NSApp setAppleMenu:appleMenu];
|
|
|
|
[appleMenu release];
|
|
|
|
|
|
|
|
|
|
|
|
/* Create the window menu */
|
|
|
|
windowMenu = [[NSMenu alloc] initWithTitle:@"Window"];
|
|
|
|
|
|
|
|
/* Add menu items */
|
|
|
|
[windowMenu addItemWithTitle:@"Minimize" action:@selector(performMiniaturize:) keyEquivalent:@"m"];
|
|
|
|
|
|
|
|
[windowMenu addItemWithTitle:@"Zoom" action:@selector(performZoom:) keyEquivalent:@""];
|
|
|
|
|
|
|
|
/* Put menu into the menubar */
|
|
|
|
menuItem = [[NSMenuItem alloc] initWithTitle:@"Window" action:nil keyEquivalent:@""];
|
|
|
|
[menuItem setSubmenu:windowMenu];
|
|
|
|
[[NSApp mainMenu] addItem:menuItem];
|
|
|
|
[menuItem release];
|
|
|
|
|
|
|
|
/* Tell the application object that this is now the window menu */
|
|
|
|
[NSApp setWindowsMenu:windowMenu];
|
|
|
|
[windowMenu release];
|
|
|
|
|
|
|
|
|
|
|
|
/* Add the fullscreen view toggle menu option, if supported */
|
2016-05-21 03:20:52 +00:00
|
|
|
if (floor(NSAppKitVersionNumber) > NSAppKitVersionNumber10_6) {
|
2015-06-21 15:33:46 +00:00
|
|
|
/* Create the view menu */
|
|
|
|
viewMenu = [[NSMenu alloc] initWithTitle:@"View"];
|
|
|
|
|
|
|
|
/* Add menu items */
|
|
|
|
menuItem = [viewMenu addItemWithTitle:@"Toggle Full Screen" action:@selector(toggleFullScreen:) keyEquivalent:@"f"];
|
|
|
|
[menuItem setKeyEquivalentModifierMask:NSControlKeyMask | NSCommandKeyMask];
|
|
|
|
|
|
|
|
/* Put menu into the menubar */
|
|
|
|
menuItem = [[NSMenuItem alloc] initWithTitle:@"View" action:nil keyEquivalent:@""];
|
|
|
|
[menuItem setSubmenu:viewMenu];
|
|
|
|
[[NSApp mainMenu] addItem:menuItem];
|
|
|
|
[menuItem release];
|
|
|
|
|
|
|
|
[viewMenu release];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cocoa_RegisterApp(void)
|
|
|
|
{ @autoreleasepool
|
|
|
|
{
|
|
|
|
/* This can get called more than once! Be careful what you initialize! */
|
|
|
|
|
|
|
|
if (NSApp == nil) {
|
|
|
|
[SDLApplication sharedApplication];
|
|
|
|
SDL_assert(NSApp != nil);
|
|
|
|
|
2016-10-14 15:15:39 +00:00
|
|
|
s_bShouldHandleEventsInSDLApplication = SDL_TRUE;
|
|
|
|
|
2016-10-08 06:40:44 +00:00
|
|
|
if (!SDL_GetHintBoolean(SDL_HINT_MAC_BACKGROUND_APP, SDL_FALSE)) {
|
2016-05-21 03:20:52 +00:00
|
|
|
[NSApp setActivationPolicy:NSApplicationActivationPolicyRegular];
|
2015-07-30 00:19:15 +00:00
|
|
|
[NSApp activateIgnoringOtherApps:YES];
|
|
|
|
}
|
|
|
|
|
2015-06-21 15:33:46 +00:00
|
|
|
if ([NSApp mainMenu] == nil) {
|
|
|
|
CreateApplicationMenus();
|
|
|
|
}
|
|
|
|
[NSApp finishLaunching];
|
|
|
|
NSDictionary *appDefaults = [[NSDictionary alloc] initWithObjectsAndKeys:
|
|
|
|
[NSNumber numberWithBool:NO], @"AppleMomentumScrollSupported",
|
|
|
|
[NSNumber numberWithBool:NO], @"ApplePressAndHoldEnabled",
|
|
|
|
[NSNumber numberWithBool:YES], @"ApplePersistenceIgnoreState",
|
|
|
|
nil];
|
|
|
|
[[NSUserDefaults standardUserDefaults] registerDefaults:appDefaults];
|
|
|
|
[appDefaults release];
|
|
|
|
}
|
|
|
|
if (NSApp && !appDelegate) {
|
|
|
|
appDelegate = [[SDLAppDelegate alloc] init];
|
|
|
|
|
|
|
|
/* If someone else has an app delegate, it means we can't turn a
|
|
|
|
* termination into SDL_Quit, and we can't handle application:openFile:
|
|
|
|
*/
|
|
|
|
if (![NSApp delegate]) {
|
|
|
|
[(NSApplication *)NSApp setDelegate:appDelegate];
|
|
|
|
} else {
|
|
|
|
appDelegate->seenFirstActivate = YES;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cocoa_PumpEvents(_THIS)
|
|
|
|
{ @autoreleasepool
|
|
|
|
{
|
|
|
|
/* Update activity every 30 seconds to prevent screensaver */
|
|
|
|
SDL_VideoData *data = (SDL_VideoData *)_this->driverdata;
|
|
|
|
if (_this->suspend_screensaver && !data->screensaver_use_iopm) {
|
|
|
|
Uint32 now = SDL_GetTicks();
|
|
|
|
if (!data->screensaver_activity ||
|
|
|
|
SDL_TICKS_PASSED(now, data->screensaver_activity + 30000)) {
|
|
|
|
UpdateSystemActivity(UsrActivity);
|
|
|
|
data->screensaver_activity = now;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for ( ; ; ) {
|
|
|
|
NSEvent *event = [NSApp nextEventMatchingMask:NSAnyEventMask untilDate:[NSDate distantPast] inMode:NSDefaultRunLoopMode dequeue:YES ];
|
|
|
|
if ( event == nil ) {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2016-10-14 15:15:39 +00:00
|
|
|
if (!s_bShouldHandleEventsInSDLApplication) {
|
|
|
|
Cocoa_DispatchEvent(event);
|
|
|
|
}
|
|
|
|
|
2016-10-01 21:34:52 +00:00
|
|
|
// Pass events down to SDLApplication to be handled in sendEvent:
|
2015-06-21 15:33:46 +00:00
|
|
|
[NSApp sendEvent:event];
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
|
|
|
|
void
|
|
|
|
Cocoa_SuspendScreenSaver(_THIS)
|
|
|
|
{ @autoreleasepool
|
|
|
|
{
|
|
|
|
SDL_VideoData *data = (SDL_VideoData *)_this->driverdata;
|
|
|
|
|
|
|
|
if (!data->screensaver_use_iopm) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data->screensaver_assertion) {
|
|
|
|
IOPMAssertionRelease(data->screensaver_assertion);
|
|
|
|
data->screensaver_assertion = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (_this->suspend_screensaver) {
|
|
|
|
/* FIXME: this should ideally describe the real reason why the game
|
|
|
|
* called SDL_DisableScreenSaver. Note that the name is only meant to be
|
|
|
|
* seen by OS X power users. there's an additional optional human-readable
|
|
|
|
* (localized) reason parameter which we don't set.
|
|
|
|
*/
|
|
|
|
NSString *name = [GetApplicationName() stringByAppendingString:@" using SDL_DisableScreenSaver"];
|
|
|
|
IOPMAssertionCreateWithDescription(kIOPMAssertPreventUserIdleDisplaySleep,
|
|
|
|
(CFStringRef) name,
|
|
|
|
NULL, NULL, NULL, 0, NULL,
|
|
|
|
&data->screensaver_assertion);
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
|
|
|
|
#endif /* SDL_VIDEO_DRIVER_COCOA */
|
|
|
|
|
|
|
|
/* vi: set ts=4 sw=4 expandtab: */
|