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