Streamline the archive loading mechanism
[tennix.git] / SDLMain.m
blob2eaa1c11ef70105dd764a19468ca5e871819c94f
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>
5 Feel free to customize this file to suit your needs
6 */
8 #import "SDL.h"
9 #import "SDLMain.h"
10 #import <sys/param.h> /* for MAXPATHLEN */
11 #import <unistd.h>
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
20 /* Use this flag to determine whether we use SDLMain.nib or not */
21 #define SDL_USE_NIB_FILE 0
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
29 UInt32 lo;
30 UInt32 hi;
31 } CPSProcessSerNum;
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);
37 #endif /* SDL_USE_CPS */
39 static int gArgc;
40 static char **gArgv;
41 static BOOL gFinderLaunch;
42 static BOOL gCalledAppMainline = FALSE;
44 static NSString *getApplicationName(void)
46 NSDictionary *dict;
47 NSString *appName = 0;
49 /* Determine the application name */
50 dict = (NSDictionary *)CFBundleGetInfoDictionary(CFBundleGetMainBundle());
51 if (dict)
52 appName = [dict objectForKey: @"CFBundleName"];
54 if (![appName length])
55 appName = [[NSProcessInfo processInfo] processName];
57 return appName;
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
67 @interface SDLApplication : NSApplication
68 @end
70 @implementation SDLApplication
71 /* Invoked from the Quit menu item */
72 - (void)terminate:(id)sender
74 /* Post a SDL_QUIT event */
75 SDL_Event event;
76 event.type = SDL_QUIT;
77 SDL_PushEvent(&event);
79 @end
81 /* The main class of the application, the application's delegate */
82 @implementation SDLMain
84 /* Set the working directory to the .app's parent directory */
85 - (void) setupWorkingDirectory:(BOOL)shouldChdir
87 if (shouldChdir)
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 */
95 CFRelease(url);
96 CFRelease(url2);
101 #if SDL_USE_NIB_FILE
103 /* Fix menu to contain the real app name instead of "SDL App" */
104 - (void)fixMenu:(NSMenu *)aMenu withAppName:(NSString *)appName
106 NSRange aRange;
107 NSEnumerator *enumerator;
108 NSMenuItem *menuItem;
110 aRange = [[aMenu title] rangeOfString:@"SDL App"];
111 if (aRange.length != 0)
112 [aMenu setTitle: [[aMenu title] stringByReplacingRange:aRange with:appName]];
114 enumerator = [[aMenu itemArray] objectEnumerator];
115 while ((menuItem = [enumerator nextObject]))
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];
123 [ aMenu sizeToFit ];
126 #else
128 static void setApplicationMenu(void)
130 /* warning: this code is very odd */
131 NSMenu *appleMenu;
132 NSMenuItem *menuItem;
133 NSString *title;
134 NSString *appName;
136 appName = getApplicationName();
137 appleMenu = [[NSMenu alloc] initWithTitle:@""];
139 /* Add menu items */
140 title = [@"About " stringByAppendingString:appName];
141 [appleMenu addItemWithTitle:title action:@selector(orderFrontStandardAboutPanel:) keyEquivalent:@""];
143 [appleMenu addItem:[NSMenuItem separatorItem]];
145 title = [@"Hide " stringByAppendingString:appName];
146 [appleMenu addItemWithTitle:title action:@selector(hide:) keyEquivalent:@"h"];
148 menuItem = (NSMenuItem *)[appleMenu addItemWithTitle:@"Hide Others" action:@selector(hideOtherApplications:) keyEquivalent:@"h"];
149 [menuItem setKeyEquivalentModifierMask:(NSAlternateKeyMask|NSCommandKeyMask)];
151 [appleMenu addItemWithTitle:@"Show All" action:@selector(unhideAllApplications:) keyEquivalent:@""];
153 [appleMenu addItem:[NSMenuItem separatorItem]];
155 title = [@"Quit " stringByAppendingString:appName];
156 [appleMenu addItemWithTitle:title action:@selector(terminate:) keyEquivalent:@"q"];
159 /* Put menu into the menubar */
160 menuItem = [[NSMenuItem alloc] initWithTitle:@"" action:nil keyEquivalent:@""];
161 [menuItem setSubmenu:appleMenu];
162 [[NSApp mainMenu] addItem:menuItem];
164 /* Tell the application object that this is now the application menu */
165 [NSApp setAppleMenu:appleMenu];
167 /* Finally give up our references to the objects */
168 [appleMenu release];
169 [menuItem release];
172 /* Create a window menu */
173 static void setupWindowMenu(void)
175 NSMenu *windowMenu;
176 NSMenuItem *windowMenuItem;
177 NSMenuItem *menuItem;
179 windowMenu = [[NSMenu alloc] initWithTitle:@"Window"];
181 /* "Minimize" item */
182 menuItem = [[NSMenuItem alloc] initWithTitle:@"Minimize" action:@selector(performMiniaturize:) keyEquivalent:@"m"];
183 [windowMenu addItem:menuItem];
184 [menuItem release];
186 /* Put menu into the menubar */
187 windowMenuItem = [[NSMenuItem alloc] initWithTitle:@"Window" action:nil keyEquivalent:@""];
188 [windowMenuItem setSubmenu:windowMenu];
189 [[NSApp mainMenu] addItem:windowMenuItem];
191 /* Tell the application object that this is now the window menu */
192 [NSApp setWindowsMenu:windowMenu];
194 /* Finally give up our references to the objects */
195 [windowMenu release];
196 [windowMenuItem release];
199 /* Replacement for NSApplicationMain */
200 static void CustomApplicationMain (int argc, char **argv)
202 NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
203 SDLMain *sdlMain;
205 /* Ensure the application object is initialised */
206 [SDLApplication sharedApplication];
208 #ifdef SDL_USE_CPS
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];
217 #endif /* SDL_USE_CPS */
219 /* Set up the menubar */
220 [NSApp setMainMenu:[[NSMenu alloc] init]];
221 setApplicationMenu();
222 setupWindowMenu();
224 /* Create SDLMain and make it the app delegate */
225 sdlMain = [[SDLMain alloc] init];
226 [NSApp setDelegate:sdlMain];
228 /* Start the main event loop */
229 [NSApp run];
231 [sdlMain release];
232 [pool release];
235 #endif
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.
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].
249 * This message may be received multiple times to open several docs on launch.
251 * This message is ignored once the app's mainline has been called.
253 - (BOOL)application:(NSApplication *)theApplication openFile:(NSString *)filename
255 const char *temparg;
256 size_t arglen;
257 char *arg;
258 char **newargv;
260 if (!gFinderLaunch) /* MacOS is passing command line args. */
261 return FALSE;
263 if (gCalledAppMainline) /* app has started, ignore this document. */
264 return FALSE;
266 temparg = [filename UTF8String];
267 arglen = SDL_strlen(temparg) + 1;
268 arg = (char *) SDL_malloc(arglen);
269 if (arg == NULL)
270 return FALSE;
272 newargv = (char **) realloc(gArgv, sizeof (char *) * (gArgc + 2));
273 if (newargv == NULL)
275 SDL_free(arg);
276 return FALSE;
278 gArgv = newargv;
280 SDL_strlcpy(arg, temparg, arglen);
281 gArgv[gArgc++] = arg;
282 gArgv[gArgc] = NULL;
283 return TRUE;
287 /* Called when the internal event loop has just started running */
288 - (void) applicationDidFinishLaunching: (NSNotification *) note
290 int status;
292 /* Set the working directory to the .app's parent directory */
293 [self setupWorkingDirectory:gFinderLaunch];
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
300 /* Hand off to main application code */
301 gCalledAppMainline = TRUE;
302 status = SDL_main (gArgc, gArgv);
304 /* We're done, thank you for playing */
305 exit(status);
307 @end
310 @implementation NSString (ReplaceSubString)
312 - (NSString *)stringByReplacingRange:(NSRange)aRange with:(NSString *)aString
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;
321 bufferSize = selfLen + aStringLen - aRange.length;
322 buffer = NSAllocateMemoryPages(bufferSize*sizeof(unichar));
324 /* Get first part into buffer */
325 localRange.location = 0;
326 localRange.length = aRange.location;
327 [self getCharacters:buffer range:localRange];
329 /* Get middle part into buffer */
330 localRange.location = 0;
331 localRange.length = aStringLen;
332 [aString getCharacters:(buffer+aRange.location) range:localRange];
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];
339 /* Build output string */
340 result = [NSString stringWithCharacters:buffer length:bufferSize];
342 NSDeallocateMemoryPages(buffer, bufferSize);
344 return result;
347 @end
351 #ifdef main
352 # undef main
353 #endif
356 /* Main entry point to executable - should *not* be SDL_main! */
357 int main (int argc, char **argv)
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;
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;