From 179fa413b047bede6e32109e2ce82437c5fb8d34 Mon Sep 17 00:00:00 2001 From: MenTaLguY Date: Mon, 16 Jan 2006 02:36:01 +0000 Subject: moving trunk for module inkscape (bzr r1) --- packaging/macosx/ScriptExec/main.c | 624 +++++++++++++++++++++++++++++++++++++ 1 file changed, 624 insertions(+) create mode 100644 packaging/macosx/ScriptExec/main.c (limited to 'packaging/macosx/ScriptExec/main.c') diff --git a/packaging/macosx/ScriptExec/main.c b/packaging/macosx/ScriptExec/main.c new file mode 100644 index 000000000..89b07ddd0 --- /dev/null +++ b/packaging/macosx/ScriptExec/main.c @@ -0,0 +1,624 @@ +/* + Platypus - create MacOS X application bundles that execute scripts + This is the executable that goes into Platypus apps + Copyright (C) 2003 Sveinbjorn Thordarson + + This program is free software; you can redistribute it and/or modify + it under the terms of the GNU General Public License as published by + the Free Software Foundation; either version 2 of the License, or + (at your option) any later version. + + This program is distributed in the hope that it will be useful, + but WITHOUT ANY WARRANTY; without even the implied warranty of + MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + GNU General Public License for more details. + + You should have received a copy of the GNU General Public License + along with this program; if not, write to the Free Software + Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. + + main.c - main program file + +*/ + +/////////////////////////////////////// +// Includes +/////////////////////////////////////// +#pragma mark Includes + +// Apple stuff +#include +#include +#include +#include + +// Unix stuff +#include +#include +#include +#include + +/////////////////////////////////////// +// Definitions +/////////////////////////////////////// +#pragma mark Definitions + +// name length limits +#define kMaxPathLength 1024 + +// names of files bundled with app +#define kScriptFileName "script" +#define kOpenDocFileName "openDoc" + +// custom carbon events +#define kEventClassRedFatalAlert 911 +#define kEventKindX11Failed 911 +#define kEventKindFCCacheFailed 912 + +//maximum arguments the script accepts +#define kMaxArgumentsToScript 252 + +/////////////////////////////////////// +// Prototypes +/////////////////////////////////////// +#pragma mark Prototypes + +static void *Execute(void *arg); +static void *OpenDoc(void *arg); +static OSErr ExecuteScript(char *script, pid_t *pid); + +static void GetParameters(void); +static char* GetScript(void); +static char* GetOpenDoc(void); + +OSErr LoadMenuBar(char *appName); + +static OSStatus FSMakePath(FSSpec file, char *path, long maxPathSize); +static void RedFatalAlert(Str255 errorString, Str255 expStr); +static short DoesFileExist(char *path); +static OSStatus FixFCCache(void); + +static OSErr AppQuitAEHandler(const AppleEvent *theAppleEvent, + AppleEvent *reply, long refCon); +static OSErr AppOpenDocAEHandler(const AppleEvent *theAppleEvent, + AppleEvent *reply, long refCon); +static OSErr AppOpenAppAEHandler(const AppleEvent *theAppleEvent, + AppleEvent *reply, long refCon); +static OSStatus X11FailedHandler(EventHandlerCallRef theHandlerCall, + EventRef theEvent, void *userData); +static OSStatus FCCacheFailedHandler(EventHandlerCallRef theHandlerCall, + EventRef theEvent, void *userData); +/////////////////////////////////////// +// Globals +/////////////////////////////////////// +#pragma mark Globals + +// process id of forked process +pid_t pid = 0; + +// thread id of threads that start scripts +pthread_t odtid = 0, tid = 0; + +// indicator of whether the script has completed executing +short taskDone = true; + +// execution parameters +char scriptPath[kMaxPathLength]; +char openDocPath[kMaxPathLength]; + +//arguments to the script +char *arguments[kMaxArgumentsToScript+3]; +char *fileArgs[kMaxArgumentsToScript]; +short numArgs = 0; + +extern char **environ; + +#pragma mark - + +/////////////////////////////////////// +// Program entrance point +/////////////////////////////////////// +int main(int argc, char* argv[]) +{ + OSErr err = noErr; + EventTypeSpec X11events = { kEventClassRedFatalAlert, kEventKindX11Failed }; + EventTypeSpec FCCacheEvents = { kEventClassRedFatalAlert, kEventKindFCCacheFailed }; + + InitCursor(); + + //install Apple Event handlers + err += AEInstallEventHandler(kCoreEventClass, kAEQuitApplication, + NewAEEventHandlerUPP(AppQuitAEHandler), + 0, false); + err += AEInstallEventHandler(kCoreEventClass, kAEOpenDocuments, + NewAEEventHandlerUPP(AppOpenDocAEHandler), + 0, false); + err += AEInstallEventHandler(kCoreEventClass, kAEOpenApplication, + NewAEEventHandlerUPP(AppOpenAppAEHandler), + 0, false); + err += InstallEventHandler(GetApplicationEventTarget(), + NewEventHandlerUPP(X11FailedHandler), 1, + &X11events, NULL, NULL); + err += InstallEventHandler(GetApplicationEventTarget(), + NewEventHandlerUPP(FCCacheFailedHandler), 1, + &FCCacheEvents, NULL, NULL); + + if (err) RedFatalAlert("\pInitialization Error", + "\pError initing Apple Event handlers."); + + //create the menu bar + if (err = LoadMenuBar(NULL)) RedFatalAlert("\pInitialization Error", + "\pError loading MenuBar.nib."); + + GetParameters(); //load data from files containing exec settings + + RunApplicationEventLoop(); //Run the event loop + return 0; +} + +#pragma mark - + + +////////////////////////////////// +// Handler for when X11 fails to start +////////////////////////////////// +static OSStatus FCCacheFailedHandler(EventHandlerCallRef theHandlerCall, + EventRef theEvent, void *userData) +{ + + pthread_join(tid, NULL); + if (odtid) pthread_join(odtid, NULL); + + SInt16 itemHit; + + AlertStdAlertParamRec params; + params.movable = true; + params.helpButton = false; + params.filterProc = NULL; + params.defaultText = "\pRun fc-cache"; + params.cancelText = "\pIgnore"; + params.otherText = NULL; + params.defaultButton = kAlertStdAlertOKButton; + params.cancelButton = kAlertStdAlertCancelButton; + params.position = kWindowDefaultPosition; + + StandardAlert(kAlertStopAlert, "\pFont caches may need to be updated", + "\pThere is a problem on OS X 10.4.x where X11 installation does not always generate the necessary fontconfig caches. This can be corrected by running /usr/X11R6/bin/fc-cache as root.\n\nThis may take some time. Please do not close Inkscape.", + ¶ms, &itemHit); + + if (itemHit == kAlertStdAlertOKButton) + { + OSStatus err = FixFCCache(); + + if (err == errAuthorizationSuccess) + { + params.defaultText = (void *) kAlertDefaultOKText; + params.cancelText = NULL; + + StandardAlert(kAlertNoteAlert, "\pFont caches have been updated", + "\pPlease re-run Inkscape.", ¶ms, &itemHit); + system("test -d $HOME/.inkscape || mkdir $HOME/.inkscape; touch $HOME/.inkscape/.fccache"); + } + } + else + { + params.defaultText = (void *) kAlertDefaultOKText; + params.cancelText = NULL; + + StandardAlert(kAlertNoteAlert, "\pFont caches have not been updated", + "\pThey can be updated manually by running the following:\n sudo /usr/X11R6/bin/fc-cache\nOnce you have dealt with this, please re-run Inkscape.", ¶ms, &itemHit); + system("test -d $HOME/.inkscape || mkdir $HOME/.inkscape; touch $HOME/.inkscape/.fccache"); + } + + ExitToShell(); + + return noErr; +} + + +///////////////////////////////////// +// Code to run fc-cache on first run +///////////////////////////////////// +static OSStatus FixFCCache (void) +{ + char* args[1]; + + // Run fc-cache + AuthorizationItem authItems[] = + { + { + kAuthorizationRightExecute, + 23, + "/usr/X11R6/bin/fc-cache", + 0 + } + }; + AuthorizationItemSet authItemSet = + { + 1, + authItems + }; + AuthorizationRef authRef = NULL; + OSStatus err = AuthorizationCreate (NULL, &authItemSet, + kAuthorizationFlagInteractionAllowed | kAuthorizationFlagExtendRights, &authRef); + + if (err == errAuthorizationSuccess) + { + //the arguments parameter to AuthorizationExecuteWithPrivileges is + //a NULL terminated array of C string pointers. + args[0]= NULL; + + AuthorizationExecuteWithPrivileges (authRef, "/usr/X11R6/bin/fc-cache", + kAuthorizationFlagDefaults, args, NULL); + } + AuthorizationFree(authRef, kAuthorizationFlagDestroyRights); + + return err; +} + +/////////////////////////////////// +// Execution thread starts here +/////////////////////////////////// +static void *Execute (void *arg) +{ + EventRef event; + + taskDone = false; + + OSErr err = ExecuteScript(scriptPath, &pid); + if (err == (OSErr)11) { + CreateEvent(NULL, kEventClassRedFatalAlert, kEventKindX11Failed, 0, + kEventAttributeNone, &event); + PostEventToQueue(GetMainEventQueue(), event, kEventPriorityStandard); + } + else if (err == (OSErr)12) { + CreateEvent(NULL, kEventClassRedFatalAlert, kEventKindFCCacheFailed, 0, + kEventAttributeNone, &event); + PostEventToQueue(GetMainEventQueue(), event, kEventPriorityStandard); + } + else ExitToShell(); + return 0; +} + +/////////////////////////////////// +// Open additional documents thread starts here +/////////////////////////////////// +static void *OpenDoc (void *arg) +{ + ExecuteScript(openDocPath, NULL); + return 0; +} + +/////////////////////////////////////// +// Run a script via the system command +/////////////////////////////////////// +static OSErr ExecuteScript (char *script, pid_t *pid) +{ + pid_t wpid = 0, p = 0; + int status, i; + + if (! pid) pid = &p; + + // Generate the array of argument strings before we do any executing + arguments[0] = script; + for (i = 0; i < numArgs; i++) arguments[i + 1] = fileArgs[i]; + arguments[i + 1] = NULL; + + *pid = fork(); //open fork + + if (*pid == (pid_t)-1) exit(13); //error + else if (*pid == 0) { //child process started + execve(arguments[0], arguments, environ); + exit(13); //if we reach this point, there's an error + } + + wpid = waitpid(*pid, &status, 0); //wait while child process finishes + + if (wpid == (pid_t)-1) return wpid; + return (OSErr)WEXITSTATUS(status); +} + +#pragma mark - + +/////////////////////////////////////// +// This function loads all the neccesary settings +// from config files in the Resources folder +/////////////////////////////////////// +static void GetParameters (void) +{ + char *str; + if (! (str = (char *)GetScript())) //get path to script to be executed + RedFatalAlert("\pInitialization Error", + "\pError getting script from application bundle."); + strcpy((char *)&scriptPath, str); + + if (! (str = (char *)GetOpenDoc())) //get path to openDoc + RedFatalAlert("\pInitialization Error", + "\pError getting openDoc from application bundle."); + strcpy((char *)&openDocPath, str); +} + +/////////////////////////////////////// +// Get path to the script in Resources folder +/////////////////////////////////////// +static char* GetScript (void) +{ + CFStringRef fileName; + CFBundleRef appBundle; + CFURLRef scriptFileURL; + FSRef fileRef; + FSSpec fileSpec; + char *path; + + //get CF URL for script + if (! (appBundle = CFBundleGetMainBundle())) return NULL; + if (! (fileName = CFStringCreateWithCString(NULL, kScriptFileName, + kCFStringEncodingASCII))) + return NULL; + if (! (scriptFileURL = CFBundleCopyResourceURL(appBundle, fileName, NULL, + NULL))) return NULL; + + //Get file reference from Core Foundation URL + if (! CFURLGetFSRef(scriptFileURL, &fileRef)) return NULL; + + //dispose of the CF variables + CFRelease(scriptFileURL); + CFRelease(fileName); + + //convert FSRef to FSSpec + if (FSGetCatalogInfo(&fileRef, kFSCatInfoNone, NULL, NULL, &fileSpec, + NULL)) return NULL; + + //create path string + if (! (path = malloc(kMaxPathLength))) return NULL; + if (FSMakePath(fileSpec, path, kMaxPathLength)) return NULL; + if (! DoesFileExist(path)) return NULL; + + return path; +} + +/////////////////////////////////////// +// Gets the path to openDoc in Resources folder +/////////////////////////////////////// +static char* GetOpenDoc (void) +{ + CFStringRef fileName; + CFBundleRef appBundle; + CFURLRef openDocFileURL; + FSRef fileRef; + FSSpec fileSpec; + char *path; + + //get CF URL for openDoc + if (! (appBundle = CFBundleGetMainBundle())) return NULL; + if (! (fileName = CFStringCreateWithCString(NULL, kOpenDocFileName, + kCFStringEncodingASCII))) + return NULL; + if (! (openDocFileURL = CFBundleCopyResourceURL(appBundle, fileName, NULL, + NULL))) return NULL; + + //Get file reference from Core Foundation URL + if (! CFURLGetFSRef( openDocFileURL, &fileRef )) return NULL; + + //dispose of the CF variables + CFRelease(openDocFileURL); + CFRelease(fileName); + + //convert FSRef to FSSpec + if (FSGetCatalogInfo(&fileRef, kFSCatInfoNone, NULL, NULL, &fileSpec, + NULL)) return NULL; + + //create path string + if (! (path = malloc(kMaxPathLength))) return NULL; + if (FSMakePath(fileSpec, path, kMaxPathLength)) return NULL; + if (! DoesFileExist(path)) return NULL; + + return path; +} + +#pragma mark - + +///////////////////////////////////// +// Load menu bar from nib +///////////////////////////////////// +OSErr LoadMenuBar (char *appName) +{ + OSErr err; + IBNibRef nibRef; + + if (err = CreateNibReference(CFSTR("MenuBar"), &nibRef)) return err; + if (err = SetMenuBarFromNib(nibRef, CFSTR("MenuBar"))) return err; + DisposeNibReference(nibRef); + + return noErr; +} + +#pragma mark - + +/////////////////////////////////////// +// Generate path string from FSSpec record +/////////////////////////////////////// +static OSStatus FSMakePath(FSSpec file, char *path, long maxPathSize) +{ + OSErr err = noErr; + FSRef fileRef; + + //create file reference from file spec + if (err = FSpMakeFSRef(&file, &fileRef)) return err; + + // and then convert the FSRef to a path + return FSRefMakePath(&fileRef, path, maxPathSize); +} + +//////////////////////////////////////// +// Standard red error alert, then exit application +//////////////////////////////////////// +static void RedFatalAlert (Str255 errorString, Str255 expStr) +{ + StandardAlert(kAlertStopAlert, errorString, expStr, NULL, NULL); + ExitToShell(); +} + +/////////////////////////////////////// +// Determines whether file exists at path or not +/////////////////////////////////////// +static short DoesFileExist (char *path) +{ + if (access(path, F_OK) == -1) return false; + return true; +} + +#pragma mark - + +/////////////////////////////////////// +// Apple Event handler for Quit i.e. from +// the dock or Application menu item +/////////////////////////////////////// +static OSErr AppQuitAEHandler(const AppleEvent *theAppleEvent, + AppleEvent *reply, long refCon) +{ + #pragma unused (reply, refCon, theAppleEvent) + + while (numArgs > 0) free(fileArgs[numArgs--]); + + if (! taskDone && pid) { //kill the script process brutally + kill(pid, 9); + printf("Platypus App: PID %d killed brutally\n", pid); + } + + pthread_cancel(tid); + if (odtid) pthread_cancel(odtid); + + ExitToShell(); + + return noErr; +} + +///////////////////////////////////// +// Handler for docs dragged on app icon +///////////////////////////////////// +static OSErr AppOpenDocAEHandler(const AppleEvent *theAppleEvent, + AppleEvent *reply, long refCon) +{ + #pragma unused (reply, refCon) + + OSErr err = noErr; + AEDescList fileSpecList; + AEKeyword keyword; + DescType type; + + short i; + long count, actualSize; + + FSSpec fileSpec; + char path[kMaxPathLength]; + + while (numArgs > 0) free(fileArgs[numArgs--]); + + //Read the AppleEvent + err = AEGetParamDesc(theAppleEvent, keyDirectObject, typeAEList, + &fileSpecList); + + err = AECountItems(&fileSpecList, &count); //Count number of files + + for (i = 1; i <= count; i++) { //iteratively process each file + //get fsspec from apple event + if (! (err = AEGetNthPtr(&fileSpecList, i, typeFSS, &keyword, &type, + (Ptr)&fileSpec, sizeof(FSSpec), &actualSize))) + { + //get path from file spec + if ((err = FSMakePath(fileSpec, (unsigned char *)&path, + kMaxPathLength))) return err; + + if (numArgs == kMaxArgumentsToScript) break; + + if (! (fileArgs[numArgs] = malloc(kMaxPathLength))) return true; + + strcpy(fileArgs[numArgs++], (char *)&path); + } + else return err; + } + + if (! taskDone) pthread_create(&odtid, NULL, OpenDoc, NULL); + else pthread_create(&tid, NULL, Execute, NULL); + + return err; +} + +/////////////////////////////// +// Handler for clicking on app icon +/////////////////////////////// +static OSErr AppOpenAppAEHandler(const AppleEvent *theAppleEvent, + AppleEvent *reply, long refCon) +{ + #pragma unused (reply, refCon, theAppleEvent) + + // the app has been opened without any items dragged on to it + pthread_create(&tid, NULL, Execute, NULL); + + return noErr; +} + + +static void OpenURL(Str255 url) +{ + // Use Internet Config to hand the URL to the appropriate application, as + // set by the user in the Internet Preferences pane. + ICInstance icInstance; + // Applications creator code: + OSType signature = 'Inks'; + OSStatus error = ICStart( &icInstance, signature ); + if ( error == noErr ) + { + ConstStr255Param hint = 0x0; + const char* data = url; + long length = strlen(url); + long start = 0; + long end = length; + // Don't bother testing return value (error); launched application will + // report problems. + ICLaunchURL( icInstance, hint, data, length, &start, &end ); + ICStop( icInstance ); + } +} + + +////////////////////////////////// +// Handler for when X11 fails to start +////////////////////////////////// +static OSStatus X11FailedHandler(EventHandlerCallRef theHandlerCall, + EventRef theEvent, void *userData) +{ + #pragma unused(theHanderCall, theEvent, userData) + + pthread_join(tid, NULL); + if (odtid) pthread_join(odtid, NULL); + + SInt16 itemHit; + const char *getX11 = "\pGet X11 for Panther"; + + AlertStdAlertParamRec params; + params.movable = true; + params.helpButton = false; + params.filterProc = NULL; + params.defaultText = (StringPtr) kAlertDefaultOKText; + params.cancelText = getX11; + params.otherText = NULL; + params.defaultButton = kAlertStdAlertOKButton; + params.cancelButton = kAlertStdAlertCancelButton; + params.position = kWindowDefaultPosition; + + StandardAlert(kAlertStopAlert, "\pFailed to start X11", + "\pInkscape.app requires Apple's X11, which is freely downloadable from Apple's website for Panther (10.3.x) users and available as an optional install from the installation DVD for Tiger (10.4.x) users.\n\nPlease install X11 and restart Inkscape.", + ¶ms, &itemHit); + + if (itemHit == kAlertStdAlertCancelButton) + { + OpenURL("http://www.apple.com/downloads/macosx/apple/x11formacosx.html"); + } + + ExitToShell(); + + + return noErr; +} -- cgit v1.2.3