| /*********************************************************** |
| Copyright 1991-1997 by Stichting Mathematisch Centrum, Amsterdam, |
| The Netherlands. |
| |
| All Rights Reserved |
| |
| Permission to use, copy, modify, and distribute this software and its |
| documentation for any purpose and without fee is hereby granted, |
| provided that the above copyright notice appear in all copies and that |
| both that copyright notice and this permission notice appear in |
| supporting documentation, and that the names of Stichting Mathematisch |
| Centrum or CWI not be used in advertising or publicity pertaining to |
| distribution of the software without specific, written prior permission. |
| |
| STICHTING MATHEMATISCH CENTRUM DISCLAIMS ALL WARRANTIES WITH REGARD TO |
| THIS SOFTWARE, INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND |
| FITNESS, IN NO EVENT SHALL STICHTING MATHEMATISCH CENTRUM BE LIABLE |
| FOR ANY SPECIAL, INDIRECT OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES |
| WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN |
| ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT |
| OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE. |
| |
| ******************************************************************/ |
| |
| /* Construct argc and argv for main() by using Apple Events */ |
| /* From Jack's implementation for STDWIN */ |
| |
| #include <stdlib.h> |
| |
| #ifdef WITHOUT_FRAMEWORKS |
| #include <Types.h> |
| #include <Files.h> |
| #include <Events.h> |
| #include <Memory.h> |
| #include <Processes.h> |
| #include <Errors.h> |
| #include <AppleEvents.h> |
| #include <AEObjects.h> |
| #include <Fonts.h> |
| #include <TextEdit.h> |
| #include <Menus.h> |
| #include <Dialogs.h> |
| #include <Windows.h> |
| #else |
| #include <Carbon/Carbon.h> |
| #endif /* WITHOUT_FRAMEWORKS */ |
| |
| typedef long refcontype; |
| |
| #include "Python.h" |
| #include "macglue.h" |
| |
| #define PATHNAMELEN 256 |
| |
| static int arg_count; |
| static char *arg_vector[256]; |
| FSSpec PyMac_ApplicationFSSpec; |
| char PyMac_ApplicationPath[PATHNAMELEN]; |
| |
| /* Duplicate a string to the heap. We also export this since it isn't standard |
| ** and others use it |
| */ |
| #ifndef HAVE_STRDUP |
| char * |
| strdup(const char *src) |
| { |
| char *dst = malloc(strlen(src) + 1); |
| if (dst) |
| strcpy(dst, src); |
| return dst; |
| } |
| #endif |
| |
| /* Initialize FSSpec and full name of current application */ |
| |
| OSErr |
| PyMac_init_process_location(void) |
| { |
| ProcessSerialNumber currentPSN; |
| ProcessInfoRec info; |
| OSErr err; |
| static int applocation_inited; |
| |
| if ( applocation_inited ) return 0; |
| currentPSN.highLongOfPSN = 0; |
| currentPSN.lowLongOfPSN = kCurrentProcess; |
| info.processInfoLength = sizeof(ProcessInfoRec); |
| info.processName = NULL; |
| info.processAppSpec = &PyMac_ApplicationFSSpec; |
| if ( err=GetProcessInformation(¤tPSN, &info)) |
| return err; |
| if ( err=PyMac_GetFullPathname(&PyMac_ApplicationFSSpec, PyMac_ApplicationPath, PATHNAMELEN) ) |
| return err; |
| applocation_inited = 1; |
| return 0; |
| } |
| |
| /* Check that there aren't any args remaining in the event */ |
| |
| static OSErr |
| get_missing_params(const AppleEvent *theAppleEvent) |
| { |
| DescType theType; |
| Size actualSize; |
| OSErr err; |
| |
| err = AEGetAttributePtr(theAppleEvent, keyMissedKeywordAttr, typeWildCard, |
| &theType, nil, 0, &actualSize); |
| if (err == errAEDescNotFound) |
| return noErr; |
| else |
| return errAEEventNotHandled; |
| } |
| |
| static int got_one; /* Flag that we can stop getting events */ |
| |
| /* Handle the Print or Quit events (by failing) */ |
| |
| static pascal OSErr |
| handle_not(const AppleEvent *theAppleEvent, AppleEvent *reply, refcontype refCon) |
| { |
| #pragma unused (reply, refCon) |
| got_one = 1; |
| return errAEEventNotHandled; |
| } |
| |
| /* Handle the Open Application event (by ignoring it) */ |
| |
| static pascal OSErr |
| handle_open_app(const AppleEvent *theAppleEvent, AppleEvent *reply, refcontype refCon) |
| { |
| #pragma unused (reply, refCon) |
| #if 0 |
| /* Test by Jack: would removing this facilitate debugging? */ |
| got_one = 1; |
| #endif |
| return get_missing_params(theAppleEvent); |
| } |
| |
| /* Handle the Open Document event, by adding an argument */ |
| |
| static pascal OSErr |
| handle_open_doc(const AppleEvent *theAppleEvent, AppleEvent *reply, refcontype refCon) |
| { |
| #pragma unused (reply, refCon) |
| OSErr err; |
| AEDescList doclist; |
| AEKeyword keywd; |
| DescType rttype; |
| long i, ndocs, size; |
| FSSpec fss; |
| char path[PATHNAMELEN]; |
| |
| got_one = 1; |
| if ((err = AEGetParamDesc(theAppleEvent, |
| keyDirectObject, typeAEList, &doclist))) |
| return err; |
| if ((err = get_missing_params(theAppleEvent))) |
| return err; |
| if ((err = AECountItems(&doclist, &ndocs))) |
| return err; |
| for(i = 1; i <= ndocs; i++) { |
| err = AEGetNthPtr(&doclist, i, typeFSS, |
| &keywd, &rttype, &fss, sizeof(fss), &size); |
| if (err) |
| break; |
| PyMac_GetFullPathname(&fss, path, PATHNAMELEN); |
| arg_vector[arg_count++] = strdup(path); |
| } |
| return err; |
| } |
| |
| /* Install standard core event handlers */ |
| AEEventHandlerUPP open_doc_upp; |
| AEEventHandlerUPP open_app_upp; |
| AEEventHandlerUPP not_upp; |
| |
| static void |
| set_ae_handlers(void) |
| { |
| open_doc_upp = NewAEEventHandlerUPP(&handle_open_doc); |
| open_app_upp = NewAEEventHandlerUPP(&handle_open_app); |
| not_upp = NewAEEventHandlerUPP(&handle_not); |
| |
| AEInstallEventHandler(kCoreEventClass, kAEOpenApplication, |
| open_app_upp, 0L, false); |
| AEInstallEventHandler(kCoreEventClass, kAEOpenDocuments, |
| open_doc_upp, 0L, false); |
| AEInstallEventHandler(kCoreEventClass, kAEPrintDocuments, |
| not_upp, 0L, false); |
| AEInstallEventHandler(kCoreEventClass, kAEQuitApplication, |
| not_upp, 0L, false); |
| } |
| |
| /* Uninstall standard core event handlers */ |
| |
| static void |
| reset_ae_handlers(void) |
| { |
| AERemoveEventHandler(kCoreEventClass, kAEOpenApplication, |
| open_app_upp, false); |
| AERemoveEventHandler(kCoreEventClass, kAEOpenDocuments, |
| open_doc_upp, false); |
| AERemoveEventHandler(kCoreEventClass, kAEPrintDocuments, |
| not_upp, false); |
| AERemoveEventHandler(kCoreEventClass, kAEQuitApplication, |
| not_upp, false); |
| } |
| |
| /* Wait for events until a core event has been handled */ |
| |
| static void |
| event_loop(void) |
| { |
| EventRecord event; |
| int n; |
| int ok; |
| |
| got_one = 0; |
| for (n = 0; n < 100 && !got_one; n++) { |
| ok = GetNextEvent(everyEvent, &event); |
| if (ok && event.what == kHighLevelEvent) { |
| AEProcessAppleEvent(&event); |
| } |
| } |
| } |
| |
| /* Get the argv vector, return argc */ |
| |
| int |
| PyMac_GetArgv(char ***pargv, int noevents) |
| { |
| arg_count = 0; |
| (void)PyMac_init_process_location(); |
| arg_vector[arg_count++] = strdup(PyMac_ApplicationPath); |
| |
| if( !noevents ) { |
| set_ae_handlers(); |
| event_loop(); |
| reset_ae_handlers(); |
| } |
| |
| arg_vector[arg_count] = NULL; |
| |
| *pargv = arg_vector; |
| return arg_count; |
| } |