/* ** i_main.mm ** **--------------------------------------------------------------------------- ** Copyright 2012-2014 Alexey Lysiuk ** All rights reserved. ** ** Redistribution and use in source and binary forms, with or without ** modification, are permitted provided that the following conditions ** are met: ** ** 1. Redistributions of source code must retain the above copyright ** notice, this list of conditions and the following disclaimer. ** 2. Redistributions in binary form must reproduce the above copyright ** notice, this list of conditions and the following disclaimer in the ** documentation and/or other materials provided with the distribution. ** 3. The name of the author may not be used to endorse or promote products ** derived from this software without specific prior written permission. ** ** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR ** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES ** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. ** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, ** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT ** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, ** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY ** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT ** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF ** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. **--------------------------------------------------------------------------- ** */ #include #import #import // Avoid collision between DObject class and Objective-C #define Class ObjectClass #include "c_console.h" #include "c_cvars.h" #include "cmdlib.h" #include "d_main.h" #include "doomerrors.h" #include "i_system.h" #include "m_argv.h" #include "s_sound.h" #include "version.h" #include "i_common.h" #include "i_osversion.h" #include "i_rbopts.h" #undef Class #define ZD_UNUSED(VARIABLE) ((void)(VARIABLE)) // --------------------------------------------------------------------------- EXTERN_CVAR(Int, vid_defwidth ) EXTERN_CVAR(Int, vid_defheight) EXTERN_CVAR(Bool, vid_vsync ) EXTERN_CVAR(Bool, fullscreen ) // --------------------------------------------------------------------------- DArgs* Args; // command line arguments namespace { // The maximum number of functions that can be registered with atterm. static const size_t MAX_TERMS = 64; static void (*TermFuncs[MAX_TERMS])(); static const char *TermNames[MAX_TERMS]; static size_t NumTerms; void call_terms() { while (NumTerms > 0) { TermFuncs[--NumTerms](); } } } // unnamed namespace void addterm(void (*func)(), const char *name) { // Make sure this function wasn't already registered. for (size_t i = 0; i < NumTerms; ++i) { if (TermFuncs[i] == func) { return; } } if (NumTerms == MAX_TERMS) { func(); I_FatalError("Too many exit functions registered."); } TermNames[NumTerms] = name; TermFuncs[NumTerms] = func; ++NumTerms; } void popterm() { if (NumTerms) { --NumTerms; } } void Mac_I_FatalError(const char* const message) { I_SetMainWindowVisible(false); const CFStringRef errorString = CFStringCreateWithCStringNoCopy(kCFAllocatorDefault, message, kCFStringEncodingASCII, kCFAllocatorNull); if (NULL != errorString) { CFOptionFlags dummy; CFUserNotificationDisplayAlert( 0, kCFUserNotificationStopAlertLevel, NULL, NULL, NULL, CFSTR("Fatal Error"), errorString, CFSTR("Exit"), NULL, NULL, &dummy); CFRelease(errorString); } } namespace { const int ARGC_MAX = 64; int s_argc; char* s_argv[ARGC_MAX]; TArray s_argvStorage; bool s_restartedFromWADPicker; void NewFailure() { I_FatalError("Failed to allocate memory from system heap"); } int OriginalMain(int argc, char** argv) { printf(GAMENAME" %s - %s - Cocoa version\nCompiled on %s\n\n", GetVersionString(), GetGitTime(), __DATE__); seteuid(getuid()); std::set_new_handler(NewFailure); // Set LC_NUMERIC environment variable in case some library decides to // clear the setlocale call at least this will be correct. // Note that the LANG environment variable is overridden by LC_* setenv("LC_NUMERIC", "C", 1); setlocale(LC_ALL, "C"); // Set reasonable default values for video settings const NSSize screenSize = [[NSScreen mainScreen] frame].size; vid_defwidth = static_cast(screenSize.width); vid_defheight = static_cast(screenSize.height); vid_vsync = true; fullscreen = true; try { Args = new DArgs(argc, argv); /* killough 1/98: This fixes some problems with exit handling during abnormal situations. The old code called I_Quit() to end program, while now I_Quit() is installed as an exit handler and exit() is called to exit, either normally or abnormally. Seg faults are caught and the error handler is used, to prevent being left in graphics mode or having very loud SFX noise because the sound card is left in an unstable state. */ atexit (call_terms); atterm (I_Quit); // Should we even be doing anything with progdir on Unix systems? char program[PATH_MAX]; if (realpath (argv[0], program) == NULL) strcpy (program, argv[0]); char *slash = strrchr (program, '/'); if (slash != NULL) { *(slash + 1) = '\0'; progdir = program; } else { progdir = "./"; } I_StartupJoysticks(); atterm(I_ShutdownJoysticks); C_InitConsole(80 * 8, 25 * 8, false); D_DoomMain(); } catch(const CDoomError& error) { const char* const message = error.GetMessage(); if (NULL != message) { fprintf(stderr, "%s\n", message); Mac_I_FatalError(message); } exit(-1); } catch(...) { call_terms(); throw; } return 0; } } // unnamed namespace // --------------------------------------------------------------------------- @interface ApplicationController : NSResponder #if MAC_OS_X_VERSION_MAX_ALLOWED >= 1070 #endif { } - (void)keyDown:(NSEvent*)theEvent; - (void)keyUp:(NSEvent*)theEvent; - (void)applicationDidBecomeActive:(NSNotification*)aNotification; - (void)applicationWillResignActive:(NSNotification*)aNotification; - (void)applicationDidFinishLaunching:(NSNotification*)aNotification; - (BOOL)application:(NSApplication*)theApplication openFile:(NSString*)filename; - (void)processEvents:(NSTimer*)timer; @end ApplicationController* appCtrl; @implementation ApplicationController - (void)keyDown:(NSEvent*)theEvent { // Empty but present to avoid playing of 'beep' alert sound ZD_UNUSED(theEvent); } - (void)keyUp:(NSEvent*)theEvent { // Empty but present to avoid playing of 'beep' alert sound ZD_UNUSED(theEvent); } - (void)applicationDidBecomeActive:(NSNotification*)aNotification { ZD_UNUSED(aNotification); S_SetSoundPaused(1); } - (void)applicationWillResignActive:(NSNotification*)aNotification { ZD_UNUSED(aNotification); S_SetSoundPaused(0); } - (void)applicationDidFinishLaunching:(NSNotification*)aNotification { // When starting from command line with real executable path, e.g. ZDoom.app/Contents/MacOS/ZDoom // application remains deactivated for an unknown reason. // The following call resolves this issue [NSApp activateIgnoringOtherApps:YES]; // Setup timer for custom event loop NSTimer* timer = [NSTimer timerWithTimeInterval:0 target:self selector:@selector(processEvents:) userInfo:nil repeats:YES]; [[NSRunLoop currentRunLoop] addTimer:timer forMode:NSDefaultRunLoopMode]; exit(OriginalMain(s_argc, s_argv)); } - (BOOL)application:(NSApplication*)theApplication openFile:(NSString*)filename { ZD_UNUSED(theApplication); if (s_restartedFromWADPicker || 0 == [filename length] || s_argc + 2 >= ARGC_MAX) { return FALSE; } // Some parameters from command line are passed to this function // These parameters need to be skipped to avoid duplication // Note: SDL has different approach to fix this issue, see the same method in SDLMain.m const char* const charFileName = [filename UTF8String]; for (int i = 0; i < s_argc; ++i) { if (0 == strcmp(s_argv[i], charFileName)) { return FALSE; } } s_argvStorage.Push("-file"); s_argv[s_argc++] = s_argvStorage.Last().LockBuffer(); s_argvStorage.Push([filename UTF8String]); s_argv[s_argc++] = s_argvStorage.Last().LockBuffer(); return TRUE; } - (void)processEvents:(NSTimer*)timer { ZD_UNUSED(timer); NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]; while (true) { NSEvent* event = [NSApp nextEventMatchingMask:NSAnyEventMask untilDate:[NSDate dateWithTimeIntervalSinceNow:0] inMode:NSDefaultRunLoopMode dequeue:YES]; if (nil == event) { break; } const NSEventType eventType = [event type]; switch (eventType) { case NSMouseMoved: I_ProcessMouseMoveEvent(event); break; case NSLeftMouseDown: case NSLeftMouseUp: case NSRightMouseDown: case NSRightMouseUp: case NSOtherMouseDown: case NSOtherMouseUp: I_ProcessMouseButtonEvent(event); break; case NSLeftMouseDragged: case NSRightMouseDragged: case NSOtherMouseDragged: I_ProcessMouseButtonEvent(event); I_ProcessMouseMoveEvent(event); break; case NSScrollWheel: I_ProcessMouseWheelEvent(event); break; case NSKeyDown: case NSKeyUp: I_ProcessKeyboardEvent(event); break; case NSFlagsChanged: I_ProcessKeyboardFlagsEvent(event); break; default: break; } [NSApp sendEvent:event]; } [NSApp updateWindows]; [pool release]; } @end // --------------------------------------------------------------------------- namespace { NSMenuItem* CreateApplicationMenu() { NSMenu* menu = [NSMenu new]; [menu addItemWithTitle:[@"About " stringByAppendingString:@GAMENAME] action:@selector(orderFrontStandardAboutPanel:) keyEquivalent:@""]; [menu addItem:[NSMenuItem separatorItem]]; [menu addItemWithTitle:[@"Hide " stringByAppendingString:@GAMENAME] action:@selector(hide:) keyEquivalent:@"h"]; [[menu addItemWithTitle:@"Hide Others" action:@selector(hideOtherApplications:) keyEquivalent:@"h"] setKeyEquivalentModifierMask:NSAlternateKeyMask | NSCommandKeyMask]; [menu addItemWithTitle:@"Show All" action:@selector(unhideAllApplications:) keyEquivalent:@""]; [menu addItem:[NSMenuItem separatorItem]]; [menu addItemWithTitle:[@"Quit " stringByAppendingString:@GAMENAME] action:@selector(terminate:) keyEquivalent:@"q"]; NSMenuItem* menuItem = [NSMenuItem new]; [menuItem setSubmenu:menu]; if ([NSApp respondsToSelector:@selector(setAppleMenu:)]) { [NSApp performSelector:@selector(setAppleMenu:) withObject:menu]; } return menuItem; } NSMenuItem* CreateEditMenu() { NSMenu* menu = [[NSMenu alloc] initWithTitle:@"Edit"]; [menu addItemWithTitle:@"Undo" action:@selector(undo:) keyEquivalent:@"z"]; [menu addItemWithTitle:@"Redo" action:@selector(redo:) keyEquivalent:@"Z"]; [menu addItem:[NSMenuItem separatorItem]]; [menu addItemWithTitle:@"Cut" action:@selector(cut:) keyEquivalent:@"x"]; [menu addItemWithTitle:@"Copy" action:@selector(copy:) keyEquivalent:@"c"]; [menu addItemWithTitle:@"Paste" action:@selector(paste:) keyEquivalent:@"v"]; [menu addItemWithTitle:@"Delete" action:@selector(delete:) keyEquivalent:@""]; [menu addItemWithTitle:@"Select All" action:@selector(selectAll:) keyEquivalent:@"a"]; NSMenuItem* menuItem = [NSMenuItem new]; [menuItem setSubmenu:menu]; return menuItem; } NSMenuItem* CreateWindowMenu() { NSMenu* menu = [[NSMenu alloc] initWithTitle:@"Window"]; [NSApp setWindowsMenu:menu]; [menu addItemWithTitle:@"Minimize" action:@selector(performMiniaturize:) keyEquivalent:@"m"]; [menu addItemWithTitle:@"Zoom" action:@selector(performZoom:) keyEquivalent:@""]; [menu addItem:[NSMenuItem separatorItem]]; [menu addItemWithTitle:@"Bring All to Front" action:@selector(arrangeInFront:) keyEquivalent:@""]; NSMenuItem* menuItem = [NSMenuItem new]; [menuItem setSubmenu:menu]; return menuItem; } void CreateMenu() { NSMenu* menuBar = [NSMenu new]; [menuBar addItem:CreateApplicationMenu()]; [menuBar addItem:CreateEditMenu()]; [menuBar addItem:CreateWindowMenu()]; [NSApp setMainMenu:menuBar]; } DarwinVersion GetDarwinVersion() { DarwinVersion result = {}; int mib[2] = { CTL_KERN, KERN_OSRELEASE }; size_t size = 0; if (0 == sysctl(mib, 2, NULL, &size, NULL, 0)) { char* version = static_cast(alloca(size)); if (0 == sysctl(mib, 2, version, &size, NULL, 0)) { sscanf(version, "%hu.%hu.%hu", &result.major, &result.minor, &result.bugfix); } } return result; } void ReleaseApplicationController() { if (NULL != appCtrl) { [NSApp setDelegate:nil]; [NSApp deactivate]; [appCtrl release]; appCtrl = NULL; } } } // unnamed namespace const DarwinVersion darwinVersion = GetDarwinVersion(); int main(int argc, char** argv) { for (int i = 0; i <= argc; ++i) { const char* const argument = argv[i]; if (NULL == argument || '\0' == argument[0]) { continue; } if (0 == strcmp(argument, "-wad_picker_restart")) { s_restartedFromWADPicker = true; } else { s_argvStorage.Push(argument); s_argv[s_argc++] = s_argvStorage.Last().LockBuffer(); } } NSAutoreleasePool* pool = [[NSAutoreleasePool alloc] init]; [NSApplication sharedApplication]; // The following code isn't mandatory, // but it enables to run the application without a bundle if ([NSApp respondsToSelector:@selector(setActivationPolicy:)]) { [NSApp setActivationPolicy:NSApplicationActivationPolicyRegular]; } CreateMenu(); atterm(ReleaseApplicationController); appCtrl = [ApplicationController new]; [NSApp setDelegate:appCtrl]; [NSApp run]; [pool release]; return EXIT_SUCCESS; }