1/*   SDLMain.m - main entry point for our Cocoa-ized SDL app
2       Initial Version: Darrell Walisser <dwaliss1@purdue.edu>
3       Non-NIB-Code & other changes: Max Horn <max@quendi.de>
4
5    Feel free to customize this file to suit your needs
6*/
7
8#import "SDL.h"
9#import "SDLMain.h"
10#import <sys/param.h> /* for MAXPATHLEN */
11#import <unistd.h>
12
13/* For some reaon, Apple removed setAppleMenu from the headers in 10.4,
14 but the method still is there and works. To avoid warnings, we declare
15 it ourselves here. */
16@interface NSApplication(SDL_Missing_Methods)
17- (void)setAppleMenu:(NSMenu *)menu;
18@end
19
20/* Use this flag to determine whether we use SDLMain.nib or not */
21#define		SDL_USE_NIB_FILE	0
22
23/* Use this flag to determine whether we use CPS (docking) or not */
24#define		SDL_USE_CPS		1
25#ifdef SDL_USE_CPS
26/* Portions of CPS.h */
27typedef struct CPSProcessSerNum
28{
29	UInt32		lo;
30	UInt32		hi;
31} CPSProcessSerNum;
32
33extern OSErr	CPSGetCurrentProcess( CPSProcessSerNum *psn);
34extern OSErr 	CPSEnableForegroundOperation( CPSProcessSerNum *psn, UInt32 _arg2, UInt32 _arg3, UInt32 _arg4, UInt32 _arg5);
35extern OSErr	CPSSetFrontProcess( CPSProcessSerNum *psn);
36
37#endif /* SDL_USE_CPS */
38
39static int    gArgc;
40static char  **gArgv;
41static BOOL   gFinderLaunch;
42static BOOL   gCalledAppMainline = FALSE;
43
44static NSString *getApplicationName(void)
45{
46    NSDictionary *dict;
47    NSString *appName = 0;
48
49    /* Determine the application name */
50    dict = (NSDictionary *)CFBundleGetInfoDictionary(CFBundleGetMainBundle());
51    if (dict)
52        appName = [dict objectForKey: @"CFBundleName"];
53
54    if (![appName length])
55        appName = [[NSProcessInfo processInfo] processName];
56
57    return appName;
58}
59
60#if SDL_USE_NIB_FILE
61/* A helper category for NSString */
62@interface NSString (ReplaceSubString)
63- (NSString *)stringByReplacingRange:(NSRange)aRange with:(NSString *)aString;
64@end
65#endif
66
67@interface SDLApplication : NSApplication
68@end
69
70@implementation SDLApplication
71/* Invoked from the Quit menu item */
72- (void)terminate:(id)sender
73{
74    /* Post a SDL_QUIT event */
75    SDL_Event event;
76    event.type = SDL_QUIT;
77    SDL_PushEvent(&event);
78}
79@end
80
81/* The main class of the application, the application's delegate */
82@implementation SDLMain
83
84/* Set the working directory to the .app's parent directory */
85- (void) setupWorkingDirectory:(BOOL)shouldChdir
86{
87    if (shouldChdir)
88    {
89        char parentdir[MAXPATHLEN];
90		CFURLRef url = CFBundleCopyBundleURL(CFBundleGetMainBundle());
91		CFURLRef url2 = CFURLCreateCopyDeletingLastPathComponent(0, url);
92		if (CFURLGetFileSystemRepresentation(url2, true, (UInt8 *)parentdir, MAXPATHLEN)) {
93	        assert ( chdir (parentdir) == 0 );   /* chdir to the binary app's parent */
94		}
95		CFRelease(url);
96		CFRelease(url2);
97	}
98
99}
100
101#if SDL_USE_NIB_FILE
102
103/* Fix menu to contain the real app name instead of "SDL App" */
104- (void)fixMenu:(NSMenu *)aMenu withAppName:(NSString *)appName
105{
106    NSRange aRange;
107    NSEnumerator *enumerator;
108    NSMenuItem *menuItem;
109
110    aRange = [[aMenu title] rangeOfString:@"SDL App"];
111    if (aRange.length != 0)
112        [aMenu setTitle: [[aMenu title] stringByReplacingRange:aRange with:appName]];
113
114    enumerator = [[aMenu itemArray] objectEnumerator];
115    while ((menuItem = [enumerator nextObject]))
116    {
117        aRange = [[menuItem title] rangeOfString:@"SDL App"];
118        if (aRange.length != 0)
119            [menuItem setTitle: [[menuItem title] stringByReplacingRange:aRange with:appName]];
120        if ([menuItem hasSubmenu])
121            [self fixMenu:[menuItem submenu] withAppName:appName];
122    }
123    [ aMenu sizeToFit ];
124}
125
126#else
127
128static void setApplicationMenu(void)
129{
130    /* warning: this code is very odd */
131    NSMenu *appleMenu;
132    NSMenuItem *menuItem;
133    NSString *title;
134    NSString *appName;
135
136    appName = getApplicationName();
137    appleMenu = [[NSMenu alloc] initWithTitle:@""];
138
139    /* Add menu items */
140    title = [@"About " stringByAppendingString:appName];
141    [appleMenu addItemWithTitle:title action:@selector(orderFrontStandardAboutPanel:) keyEquivalent:@""];
142
143    [appleMenu addItem:[NSMenuItem separatorItem]];
144
145    title = [@"Hide " stringByAppendingString:appName];
146    [appleMenu addItemWithTitle:title action:@selector(hide:) keyEquivalent:@"h"];
147
148    menuItem = (NSMenuItem *)[appleMenu addItemWithTitle:@"Hide Others" action:@selector(hideOtherApplications:) keyEquivalent:@"h"];
149    [menuItem setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
150
151    [appleMenu addItemWithTitle:@"Show All" action:@selector(unhideAllApplications:) keyEquivalent:@""];
152
153    [appleMenu addItem:[NSMenuItem separatorItem]];
154
155    title = [@"Quit " stringByAppendingString:appName];
156    [appleMenu addItemWithTitle:title action:@selector(terminate:) keyEquivalent:@"q"];
157
158
159    /* Put menu into the menubar */
160    menuItem = [[NSMenuItem alloc] initWithTitle:@"" action:nil keyEquivalent:@""];
161    [menuItem setSubmenu:appleMenu];
162    [[NSApp mainMenu] addItem:menuItem];
163
164    /* Tell the application object that this is now the application menu */
165    [NSApp setAppleMenu:appleMenu];
166
167    /* Finally give up our references to the objects */
168    [appleMenu release];
169    [menuItem release];
170}
171
172/* Create a window menu */
173static void setupWindowMenu(void)
174{
175    NSMenu      *windowMenu;
176    NSMenuItem  *windowMenuItem;
177    NSMenuItem  *menuItem;
178
179    windowMenu = [[NSMenu alloc] initWithTitle:@"Window"];
180
181    /* "Minimize" item */
182    menuItem = [[NSMenuItem alloc] initWithTitle:@"Minimize" action:@selector(performMiniaturize:) keyEquivalent:@"m"];
183    [windowMenu addItem:menuItem];
184    [menuItem release];
185
186    /* Put menu into the menubar */
187    windowMenuItem = [[NSMenuItem alloc] initWithTitle:@"Window" action:nil keyEquivalent:@""];
188    [windowMenuItem setSubmenu:windowMenu];
189    [[NSApp mainMenu] addItem:windowMenuItem];
190
191    /* Tell the application object that this is now the window menu */
192    [NSApp setWindowsMenu:windowMenu];
193
194    /* Finally give up our references to the objects */
195    [windowMenu release];
196    [windowMenuItem release];
197}
198
199/* Replacement for NSApplicationMain */
200static void CustomApplicationMain (int argc, char **argv)
201{
202    NSAutoreleasePool	*pool = [[NSAutoreleasePool alloc] init];
203    SDLMain				*sdlMain;
204
205    /* Ensure the application object is initialised */
206    [SDLApplication sharedApplication];
207
208#ifdef SDL_USE_CPS
209    {
210        CPSProcessSerNum PSN;
211        /* Tell the dock about us */
212        if (!CPSGetCurrentProcess(&PSN))
213            if (!CPSEnableForegroundOperation(&PSN,0x03,0x3C,0x2C,0x1103))
214                if (!CPSSetFrontProcess(&PSN))
215                    [SDLApplication sharedApplication];
216    }
217#endif /* SDL_USE_CPS */
218
219    /* Set up the menubar */
220    [NSApp setMainMenu:[[NSMenu alloc] init]];
221    setApplicationMenu();
222    setupWindowMenu();
223
224    /* Create SDLMain and make it the app delegate */
225    sdlMain = [[SDLMain alloc] init];
226    [NSApp setDelegate:sdlMain];
227
228    /* Start the main event loop */
229    [NSApp run];
230
231    [sdlMain release];
232    [pool release];
233}
234
235#endif
236
237
238/*
239 * Catch document open requests...this lets us notice files when the app
240 *  was launched by double-clicking a document, or when a document was
241 *  dragged/dropped on the app's icon. You need to have a
242 *  CFBundleDocumentsType section in your Info.plist to get this message,
243 *  apparently.
244 *
245 * Files are added to gArgv, so to the app, they'll look like command line
246 *  arguments. Previously, apps launched from the finder had nothing but
247 *  an argv[0].
248 *
249 * This message may be received multiple times to open several docs on launch.
250 *
251 * This message is ignored once the app's mainline has been called.
252 */
253- (BOOL)application:(NSApplication *)theApplication openFile:(NSString *)filename
254{
255    const char *temparg;
256    size_t arglen;
257    char *arg;
258    char **newargv;
259
260    if (!gFinderLaunch)  /* MacOS is passing command line args. */
261        return FALSE;
262
263    if (gCalledAppMainline)  /* app has started, ignore this document. */
264        return FALSE;
265
266    temparg = [filename UTF8String];
267    arglen = SDL_strlen(temparg) + 1;
268    arg = (char *) SDL_malloc(arglen);
269    if (arg == NULL)
270        return FALSE;
271
272    newargv = (char **) realloc(gArgv, sizeof (char *) * (gArgc + 2));
273    if (newargv == NULL)
274    {
275        SDL_free(arg);
276        return FALSE;
277    }
278    gArgv = newargv;
279
280    SDL_strlcpy(arg, temparg, arglen);
281    gArgv[gArgc++] = arg;
282    gArgv[gArgc] = NULL;
283    return TRUE;
284}
285
286
287/* Called when the internal event loop has just started running */
288- (void) applicationDidFinishLaunching: (NSNotification *) note
289{
290    int status;
291
292    /* Set the working directory to the .app's parent directory */
293    [self setupWorkingDirectory:gFinderLaunch];
294
295#if SDL_USE_NIB_FILE
296    /* Set the main menu to contain the real app name instead of "SDL App" */
297    [self fixMenu:[NSApp mainMenu] withAppName:getApplicationName()];
298#endif
299
300    /* Hand off to main application code */
301    gCalledAppMainline = TRUE;
302    status = SDL_main (gArgc, gArgv);
303
304    /* We're done, thank you for playing */
305    exit(status);
306}
307@end
308
309
310@implementation NSString (ReplaceSubString)
311
312- (NSString *)stringByReplacingRange:(NSRange)aRange with:(NSString *)aString
313{
314    unsigned int bufferSize;
315    unsigned int selfLen = [self length];
316    unsigned int aStringLen = [aString length];
317    unichar *buffer;
318    NSRange localRange;
319    NSString *result;
320
321    bufferSize = selfLen + aStringLen - aRange.length;
322    buffer = NSAllocateMemoryPages(bufferSize*sizeof(unichar));
323
324    /* Get first part into buffer */
325    localRange.location = 0;
326    localRange.length = aRange.location;
327    [self getCharacters:buffer range:localRange];
328
329    /* Get middle part into buffer */
330    localRange.location = 0;
331    localRange.length = aStringLen;
332    [aString getCharacters:(buffer+aRange.location) range:localRange];
333
334    /* Get last part into buffer */
335    localRange.location = aRange.location + aRange.length;
336    localRange.length = selfLen - localRange.location;
337    [self getCharacters:(buffer+aRange.location+aStringLen) range:localRange];
338
339    /* Build output string */
340    result = [NSString stringWithCharacters:buffer length:bufferSize];
341
342    NSDeallocateMemoryPages(buffer, bufferSize);
343
344    return result;
345}
346
347@end
348
349
350
351#ifdef main
352#  undef main
353#endif
354
355
356/* Main entry point to executable - should *not* be SDL_main! */
357int main (int argc, char **argv)
358{
359    /* Copy the arguments into a global variable */
360    /* This is passed if we are launched by double-clicking */
361    if ( argc >= 2 && strncmp (argv[1], "-psn", 4) == 0 ) {
362        gArgv = (char **) SDL_malloc(sizeof (char *) * 2);
363        gArgv[0] = argv[0];
364        gArgv[1] = NULL;
365        gArgc = 1;
366        gFinderLaunch = YES;
367    } else {
368        int i;
369        gArgc = argc;
370        gArgv = (char **) SDL_malloc(sizeof (char *) * (argc+1));
371        for (i = 0; i <= argc; i++)
372            gArgv[i] = argv[i];
373        gFinderLaunch = NO;
374    }
375
376#if SDL_USE_NIB_FILE
377    [SDLApplication poseAsClass:[NSApplication class]];
378    NSApplicationMain (argc, argv);
379#else
380    CustomApplicationMain (argc, argv);
381#endif
382    return 0;
383}
384
385