/* -*- Mode: C++; tab-width: 3; indent-tabs-mode: nil; c-basic-offset: 2 -*- * * ***** BEGIN LICENSE BLOCK ***** * Version: MPL 1.1/GPL 2.0/LGPL 2.1 * * The contents of this file are subject to the Mozilla Public License Version * 1.1 (the "License"); you may not use this file except in compliance with * the License. You may obtain a copy of the License at * http://www.mozilla.org/MPL/ * * Software distributed under the License is distributed on an "AS IS" basis, * WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License * for the specific language governing rights and limitations under the * License. * * The Original Code is the Mozilla browser. * * The Initial Developer of the Original Code is * Netscape Communications, Inc. * Portions created by the Initial Developer are Copyright (C) 1999 * the Initial Developer. All Rights Reserved. * * Contributor(s): * Scott MacGregor * Christian Biesinger * Dan Mosedale * * Alternatively, the contents of this file may be used under the terms of * either of the GNU General Public License Version 2 or later (the "GPL"), * or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"), * in which case the provisions of the GPL or the LGPL are applicable instead * of those above. If you wish to allow use of your version of this file only * under the terms of either the GPL or the LGPL, and not to allow others to * use your version of this file under the terms of the MPL, indicate your * decision by deleting the provisions above and replace them with the notice * and other provisions required by the GPL or the LGPL. If you do not delete * the provisions above, a recipient may use your version of this file under * the terms of any one of the MPL, the GPL or the LGPL. * * ***** END LICENSE BLOCK ***** */ #include #include "nsMIMEInfoMac.h" #include "nsILocalFileMac.h" #include "nsIFileURL.h" #include "nsIInternetConfigService.h" NS_IMETHODIMP nsMIMEInfoMac::LaunchWithFile(nsIFile *aFile) { nsCOMPtr application; nsresult rv; NS_ASSERTION(mClass == eMIMEInfo, "only MIME infos are currently allowed" "to pass content by value"); if (mPreferredAction == useHelperApp) { // we don't yet support passing content by value (rather than reference) // to web apps. at some point, we will probably want to. nsCOMPtr localHandlerApp = do_QueryInterface(mPreferredApplication, &rv); NS_ENSURE_SUCCESS(rv, rv); rv = localHandlerApp->GetExecutable(getter_AddRefs(application)); NS_ENSURE_SUCCESS(rv, rv); } else if (mPreferredAction == useSystemDefault) { application = mDefaultApplication; } else return NS_ERROR_INVALID_ARG; // if we've already got an app, just QI so we have the launchWithDoc method nsCOMPtr app; if (application) { app = do_QueryInterface(application, &rv); if (NS_FAILED(rv)) return rv; } else { // otherwise ask LaunchServices for an app directly nsCOMPtr tempFile = do_QueryInterface(aFile, &rv); if (NS_FAILED(rv)) return rv; FSRef tempFileRef; tempFile->GetFSRef(&tempFileRef); FSRef appFSRef; if (::LSGetApplicationForItem(&tempFileRef, kLSRolesAll, &appFSRef, nsnull) == noErr) { app = (do_CreateInstance("@mozilla.org/file/local;1")); if (!app) return NS_ERROR_FAILURE; app->InitWithFSRef(&appFSRef); } else { return NS_ERROR_FAILURE; } } nsCOMPtr localFile = do_QueryInterface(aFile); return app->LaunchWithDoc(localFile, PR_FALSE); } NS_IMETHODIMP nsMIMEInfoMac::LaunchWithURI(nsIURI* aURI, nsIInterfaceRequestor* aWindowContext) { nsCOMPtr application; nsresult rv; // for now, this is only being called with protocol handlers; that // will change once we get to more general registerContentHandler // support NS_ASSERTION(mClass == eProtocolInfo, "nsMIMEInfoBase should be a protocol handler"); if (mPreferredAction == useHelperApp) { // check for and launch with web handler app nsCOMPtr webHandlerApp = do_QueryInterface(mPreferredApplication, &rv); if (NS_SUCCEEDED(rv)) { return LaunchWithWebHandler(webHandlerApp, aURI, aWindowContext); } // otherwise, get the application executable from the handler nsCOMPtr localHandlerApp = do_QueryInterface(mPreferredApplication, &rv); NS_ENSURE_SUCCESS(rv, rv); rv = localHandlerApp->GetExecutable(getter_AddRefs(application)); NS_ENSURE_SUCCESS(rv, rv); // pass the entire URI to the handler. nsCAutoString spec; aURI->GetSpec(spec); return OpenApplicationWithURI(application, spec); } if (mPreferredAction == useSystemDefault) { return LoadUriInternal(aURI); } return NS_ERROR_INVALID_ARG; } nsresult nsMIMEInfoMac::LoadUriInternal(nsIURI *aURI) { NS_ENSURE_ARG_POINTER(aURI); nsresult rv = NS_ERROR_FAILURE; nsCAutoString uri; aURI->GetSpec(uri); if (!uri.IsEmpty()) { nsCOMPtr icService = do_GetService(NS_INTERNETCONFIGSERVICE_CONTRACTID); if (icService) rv = icService->LaunchURL(uri.get()); } return rv; } NS_IMETHODIMP nsMIMEInfoMac::GetHasDefaultHandler(PRBool *_retval) { // We have a default application if we have a description *_retval = !mDefaultAppDescription.IsEmpty(); return NS_OK; } /** * static; mostly copy/pasted from nsMacShellService.cpp (which is in browser/, * so we can't depend on it here). This code probably really wants to live * somewhere more central; see bug 389922. */ nsresult nsMIMEInfoMac::OpenApplicationWithURI(nsIFile* aApplication, const nsCString& aURI) { nsresult rv; nsCOMPtr lfm(do_QueryInterface(aApplication, &rv)); NS_ENSURE_SUCCESS(rv, rv); CFURLRef appURL; rv = lfm->GetCFURL(&appURL); if (NS_FAILED(rv)) return rv; const UInt8* uriString = (const UInt8*)aURI.get(); CFURLRef uri = ::CFURLCreateWithBytes(NULL, uriString, aURI.Length(), kCFStringEncodingUTF8, NULL); if (!uri) { ::CFRelease(appURL); return NS_ERROR_OUT_OF_MEMORY; } CFArrayRef uris = ::CFArrayCreate(NULL, (const void**)&uri, 1, NULL); if (!uris) { ::CFRelease(uri); ::CFRelease(appURL); return NS_ERROR_OUT_OF_MEMORY; } LSLaunchURLSpec launchSpec; launchSpec.appURL = appURL; launchSpec.itemURLs = uris; launchSpec.passThruParams = NULL; launchSpec.launchFlags = kLSLaunchDefaults; launchSpec.asyncRefCon = NULL; OSErr err = ::LSOpenFromURLSpec(&launchSpec, NULL); ::CFRelease(uris); ::CFRelease(uri); ::CFRelease(appURL); return err != noErr ? NS_ERROR_FAILURE : NS_OK; }