mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-26 02:21:14 +00:00
627 lines
14 KiB
Text
627 lines
14 KiB
Text
|
/*
|
||
|
** 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 <sys/sysctl.h>
|
||
|
|
||
|
#include <AppKit/NSMenu.h>
|
||
|
#include <AppKit/NSScreen.h>
|
||
|
|
||
|
#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"
|
||
|
|
||
|
|
||
|
#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<FString> 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<int>(screenSize.width);
|
||
|
vid_defheight = static_cast<int>(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
|
||
|
<NSFileManagerDelegate>
|
||
|
#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<char*>(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;
|
||
|
}
|