2014-12-28 14:35:00 +00:00
|
|
|
/*
|
|
|
|
** i_main.mm
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
2015-01-04 12:08:53 +00:00
|
|
|
** Copyright 2012-2015 Alexey Lysiuk
|
2014-12-28 14:35:00 +00:00
|
|
|
** 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.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
2015-01-05 15:24:54 +00:00
|
|
|
#include "i_common.h"
|
2017-05-19 10:20:33 +00:00
|
|
|
#include "s_sound.h"
|
2014-12-28 14:35:00 +00:00
|
|
|
|
2015-01-05 15:24:54 +00:00
|
|
|
#include <sys/sysctl.h>
|
2014-12-28 15:11:30 +00:00
|
|
|
|
2014-12-28 14:35:00 +00:00
|
|
|
#include "c_console.h"
|
|
|
|
#include "c_cvars.h"
|
|
|
|
#include "cmdlib.h"
|
|
|
|
#include "d_main.h"
|
|
|
|
#include "i_system.h"
|
|
|
|
#include "m_argv.h"
|
2015-12-29 13:19:42 +00:00
|
|
|
#include "st_console.h"
|
2014-12-28 14:35:00 +00:00
|
|
|
#include "version.h"
|
|
|
|
|
|
|
|
|
|
|
|
#define ZD_UNUSED(VARIABLE) ((void)(VARIABLE))
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
2017-12-24 23:01:45 +00:00
|
|
|
CVAR (Bool, i_soundinbackground, false, CVAR_ARCHIVE|CVAR_GLOBALCONFIG)
|
2014-12-28 14:35:00 +00:00
|
|
|
EXTERN_CVAR(Int, vid_defwidth )
|
|
|
|
EXTERN_CVAR(Int, vid_defheight)
|
|
|
|
EXTERN_CVAR(Bool, vid_vsync )
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
|
|
|
// The maximum number of functions that can be registered with atterm.
|
2014-12-29 10:10:18 +00:00
|
|
|
const size_t MAX_TERMS = 64;
|
2014-12-28 14:35:00 +00:00
|
|
|
|
2014-12-29 10:10:18 +00:00
|
|
|
void (*TermFuncs[MAX_TERMS])();
|
|
|
|
const char *TermNames[MAX_TERMS];
|
|
|
|
size_t NumTerms;
|
2014-12-28 14:35:00 +00:00
|
|
|
|
2016-03-13 05:34:35 +00:00
|
|
|
} // unnamed namespace
|
|
|
|
|
|
|
|
// Expose this for i_main_except.cpp
|
2014-12-28 14:35:00 +00:00
|
|
|
void call_terms()
|
|
|
|
{
|
|
|
|
while (NumTerms > 0)
|
|
|
|
{
|
|
|
|
TermFuncs[--NumTerms]();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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);
|
2017-05-19 10:20:33 +00:00
|
|
|
S_StopMusic(true);
|
2014-12-28 14:35:00 +00:00
|
|
|
|
2015-12-29 13:19:42 +00:00
|
|
|
FConsoleWindow::GetInstance().ShowFatalError(message);
|
2014-12-28 14:35:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-07-15 07:43:52 +00:00
|
|
|
#if MAC_OS_X_VERSION_MAX_ALLOWED < 101000
|
|
|
|
|
|
|
|
// Available since 10.9 with no public declaration/definition until 10.10
|
|
|
|
|
|
|
|
struct NSOperatingSystemVersion
|
|
|
|
{
|
|
|
|
NSInteger majorVersion;
|
|
|
|
NSInteger minorVersion;
|
|
|
|
NSInteger patchVersion;
|
|
|
|
};
|
|
|
|
|
|
|
|
@interface NSProcessInfo(OperatingSystemVersion)
|
|
|
|
- (NSOperatingSystemVersion)operatingSystemVersion;
|
|
|
|
@end
|
|
|
|
|
|
|
|
#endif // before 10.10
|
|
|
|
|
2016-10-08 12:47:16 +00:00
|
|
|
static void I_DetectOS()
|
|
|
|
{
|
2018-07-15 07:43:52 +00:00
|
|
|
NSOperatingSystemVersion version = {};
|
|
|
|
NSProcessInfo* const processInfo = [NSProcessInfo processInfo];
|
|
|
|
|
|
|
|
if ([processInfo respondsToSelector:@selector(operatingSystemVersion)])
|
|
|
|
{
|
|
|
|
version = [processInfo operatingSystemVersion];
|
|
|
|
}
|
|
|
|
|
2016-10-08 12:47:16 +00:00
|
|
|
const char* name = "Unknown version";
|
|
|
|
|
2018-07-15 07:43:52 +00:00
|
|
|
if (10 == version.majorVersion) switch (version.minorVersion)
|
2016-10-08 12:47:16 +00:00
|
|
|
{
|
|
|
|
case 7: name = "Mac OS X Lion"; break;
|
|
|
|
case 8: name = "OS X Mountain Lion"; break;
|
|
|
|
case 9: name = "OS X Mavericks"; break;
|
|
|
|
case 10: name = "OS X Yosemite"; break;
|
|
|
|
case 11: name = "OS X El Capitan"; break;
|
|
|
|
case 12: name = "macOS Sierra"; break;
|
2017-07-01 09:29:25 +00:00
|
|
|
case 13: name = "macOS High Sierra"; break;
|
2018-06-28 08:06:19 +00:00
|
|
|
case 14: name = "macOS Mojave"; break;
|
2016-10-08 12:47:16 +00:00
|
|
|
}
|
|
|
|
|
2016-11-19 10:44:35 +00:00
|
|
|
char release[16] = "unknown";
|
2016-10-08 12:47:16 +00:00
|
|
|
size_t size = sizeof release - 1;
|
|
|
|
sysctlbyname("kern.osversion", release, &size, nullptr, 0);
|
2018-07-15 11:26:29 +00:00
|
|
|
|
|
|
|
char model[64] = "Unknown Mac model";
|
|
|
|
size = sizeof model - 1;
|
|
|
|
sysctlbyname("hw.model", model, &size, nullptr, 0);
|
|
|
|
|
2016-10-08 12:47:16 +00:00
|
|
|
const char* const architecture =
|
|
|
|
#ifdef __i386__
|
|
|
|
"32-bit Intel";
|
|
|
|
#elif defined __x86_64__
|
|
|
|
"64-bit Intel";
|
|
|
|
#else
|
|
|
|
"Unknown";
|
|
|
|
#endif
|
|
|
|
|
2018-07-15 11:26:29 +00:00
|
|
|
Printf("%s running %s %d.%d.%d (%s) %s\n", model, name,
|
2018-07-15 07:43:52 +00:00
|
|
|
int(version.majorVersion), int(version.minorVersion), int(version.patchVersion),
|
|
|
|
release, architecture);
|
2016-10-08 12:47:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-04-14 11:31:58 +00:00
|
|
|
FArgs* Args; // command line arguments
|
2014-12-29 10:10:18 +00:00
|
|
|
|
|
|
|
|
2016-03-13 05:34:35 +00:00
|
|
|
// Newer versions of GCC than 4.2 have a bug with C++ exceptions in Objective-C++ code.
|
|
|
|
// To work around we'll implement the try and catch in standard C++.
|
|
|
|
// https://gcc.gnu.org/bugzilla/show_bug.cgi?id=61759
|
|
|
|
void OriginalMainExcept(int argc, char** argv);
|
|
|
|
void OriginalMainTry(int argc, char** argv)
|
|
|
|
{
|
2017-04-14 11:31:58 +00:00
|
|
|
Args = new FArgs(argc, argv);
|
2016-03-13 05:34:35 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
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);
|
|
|
|
|
|
|
|
NSString* exePath = [[NSBundle mainBundle] executablePath];
|
|
|
|
progdir = [[exePath stringByDeletingLastPathComponent] UTF8String];
|
|
|
|
progdir += "/";
|
|
|
|
|
|
|
|
C_InitConsole(80 * 8, 25 * 8, false);
|
2016-10-08 12:47:16 +00:00
|
|
|
|
|
|
|
I_DetectOS();
|
2016-03-13 05:34:35 +00:00
|
|
|
D_DoomMain();
|
|
|
|
}
|
|
|
|
|
2014-12-28 14:35:00 +00:00
|
|
|
namespace
|
|
|
|
{
|
|
|
|
|
2018-01-14 09:29:45 +00:00
|
|
|
TArray<FString> s_argv;
|
2014-12-28 14:35:00 +00:00
|
|
|
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
2016-03-13 05:34:35 +00:00
|
|
|
OriginalMainExcept(argc, argv);
|
2014-12-28 14:35:00 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
} // unnamed namespace
|
|
|
|
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------------
|
|
|
|
|
|
|
|
|
2017-10-07 10:37:13 +00:00
|
|
|
@interface ApplicationController : NSResponder<NSApplicationDelegate>
|
2014-12-28 14:35:00 +00:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
- (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);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-05-04 08:24:37 +00:00
|
|
|
extern bool AppActive;
|
|
|
|
|
2014-12-28 14:35:00 +00:00
|
|
|
- (void)applicationDidBecomeActive:(NSNotification*)aNotification
|
|
|
|
{
|
|
|
|
ZD_UNUSED(aNotification);
|
|
|
|
|
|
|
|
S_SetSoundPaused(1);
|
2018-05-04 08:24:37 +00:00
|
|
|
|
|
|
|
AppActive = true;
|
2014-12-28 14:35:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
- (void)applicationWillResignActive:(NSNotification*)aNotification
|
|
|
|
{
|
|
|
|
ZD_UNUSED(aNotification);
|
|
|
|
|
2018-05-04 08:24:37 +00:00
|
|
|
S_SetSoundPaused(i_soundinbackground);
|
|
|
|
|
|
|
|
AppActive = false;
|
2014-12-28 14:35:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
- (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];
|
|
|
|
|
2015-12-29 13:19:42 +00:00
|
|
|
FConsoleWindow::CreateInstance();
|
|
|
|
atterm(FConsoleWindow::DeleteInstance);
|
|
|
|
|
2018-01-14 09:29:45 +00:00
|
|
|
const size_t argc = s_argv.Size();
|
|
|
|
TArray<char*> argv(argc + 1, true);
|
|
|
|
|
|
|
|
for (size_t i = 0; i < argc; ++i)
|
|
|
|
{
|
|
|
|
argv[i] = s_argv[i].LockBuffer();
|
|
|
|
}
|
|
|
|
|
|
|
|
argv[argc] = nullptr;
|
|
|
|
|
|
|
|
exit(OriginalMain(argc, &argv[0]));
|
2014-12-28 14:35:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
- (BOOL)application:(NSApplication*)theApplication openFile:(NSString*)filename
|
|
|
|
{
|
|
|
|
ZD_UNUSED(theApplication);
|
|
|
|
|
|
|
|
// 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];
|
|
|
|
|
2018-01-14 09:29:45 +00:00
|
|
|
for (size_t i = 0, count = s_argv.Size(); i < count; ++i)
|
2014-12-28 14:35:00 +00:00
|
|
|
{
|
|
|
|
if (0 == strcmp(s_argv[i], charFileName))
|
|
|
|
{
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-09 09:47:11 +00:00
|
|
|
bool iwad = false;
|
|
|
|
|
|
|
|
if (const char* const extPos = strrchr(charFileName, '.'))
|
|
|
|
{
|
|
|
|
iwad = 0 == stricmp(extPos, ".iwad")
|
|
|
|
|| 0 == stricmp(extPos, ".ipk3")
|
|
|
|
|| 0 == stricmp(extPos, ".ipk7");
|
|
|
|
}
|
|
|
|
|
|
|
|
s_argv.Push(iwad ? "-iwad" : "-file");
|
|
|
|
s_argv.Push(charFileName);
|
2014-12-28 14:35:00 +00:00
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
2014-12-29 10:16:17 +00:00
|
|
|
|
|
|
|
I_ProcessEvent(event);
|
|
|
|
|
2014-12-28 14:35:00 +00:00
|
|
|
[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];
|
|
|
|
}
|
|
|
|
|
|
|
|
void ReleaseApplicationController()
|
|
|
|
{
|
|
|
|
if (NULL != appCtrl)
|
|
|
|
{
|
|
|
|
[NSApp setDelegate:nil];
|
|
|
|
[NSApp deactivate];
|
|
|
|
|
|
|
|
[appCtrl release];
|
|
|
|
appCtrl = NULL;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} // unnamed namespace
|
|
|
|
|
|
|
|
|
|
|
|
int main(int argc, char** argv)
|
|
|
|
{
|
2018-01-14 09:29:45 +00:00
|
|
|
for (int i = 0; i < argc; ++i)
|
2014-12-28 14:35:00 +00:00
|
|
|
{
|
|
|
|
const char* const argument = argv[i];
|
|
|
|
|
2017-11-02 12:49:08 +00:00
|
|
|
#if _DEBUG
|
2018-01-14 09:29:45 +00:00
|
|
|
if (0 == strcmp(argument, "-wait_for_debugger"))
|
2017-11-02 12:49:08 +00:00
|
|
|
{
|
|
|
|
NSAlert* alert = [[NSAlert alloc] init];
|
|
|
|
[alert setMessageText:@GAMENAME];
|
|
|
|
[alert setInformativeText:@"Waiting for debugger..."];
|
|
|
|
[alert addButtonWithTitle:@"Continue"];
|
|
|
|
[alert runModal];
|
|
|
|
}
|
|
|
|
#endif // _DEBUG
|
2018-01-14 09:29:45 +00:00
|
|
|
|
|
|
|
s_argv.Push(argument);
|
2014-12-28 14:35:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|