597 lines
18 KiB
C++
597 lines
18 KiB
C++
/* -*- Mode: C++; tab-width: 2; 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 Shell Service.
|
|
*
|
|
* The Initial Developer of the Original Code is mozilla.org.
|
|
* Portions created by the Initial Developer are Copyright (C) 2004
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
*
|
|
* Alternatively, the contents of this file may be used under the terms of
|
|
* either 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 "nsCOMPtr.h"
|
|
#include "nsGNOMEShellService.h"
|
|
#include "nsShellService.h"
|
|
#include "nsIServiceManager.h"
|
|
#include "nsILocalFile.h"
|
|
#include "nsIProperties.h"
|
|
#include "nsDirectoryServiceDefs.h"
|
|
#include "nsIPrefService.h"
|
|
#include "prenv.h"
|
|
#include "nsStringAPI.h"
|
|
#include "nsIGConfService.h"
|
|
#include "nsIGnomeVFSService.h"
|
|
#include "nsIStringBundle.h"
|
|
#include "gfxIImageFrame.h"
|
|
#include "nsIOutputStream.h"
|
|
#include "nsIProcess.h"
|
|
#include "nsNetUtil.h"
|
|
#include "nsIDOMHTMLImageElement.h"
|
|
#include "nsIImageLoadingContent.h"
|
|
#include "imgIRequest.h"
|
|
#include "imgIContainer.h"
|
|
#include "nsIImage.h"
|
|
#include "prprf.h"
|
|
#ifdef MOZ_WIDGET_GTK2
|
|
#include "nsIImageToPixbuf.h"
|
|
#endif
|
|
|
|
#include <glib.h>
|
|
#include <glib-object.h>
|
|
#include <gtk/gtkversion.h>
|
|
#include <gdk/gdk.h>
|
|
#include <gdk-pixbuf/gdk-pixbuf.h>
|
|
#include <limits.h>
|
|
#include <stdlib.h>
|
|
|
|
struct ProtocolAssociation
|
|
{
|
|
const char *name;
|
|
PRBool essential;
|
|
};
|
|
|
|
struct MimeTypeAssociation
|
|
{
|
|
const char *mimeType;
|
|
const char *extensions;
|
|
};
|
|
|
|
static const ProtocolAssociation appProtocols[] = {
|
|
{ "http", PR_TRUE },
|
|
{ "https", PR_TRUE },
|
|
{ "ftp", PR_FALSE },
|
|
{ "chrome", PR_FALSE }
|
|
};
|
|
|
|
static const MimeTypeAssociation appTypes[] = {
|
|
{ "text/html", "htm html shtml" },
|
|
{ "application/xhtml+xml", "xhtml xht" }
|
|
};
|
|
|
|
static const char kDocumentIconPath[] = "firefox-document.png";
|
|
|
|
// GConf registry key constants
|
|
#define DG_BACKGROUND "/desktop/gnome/background"
|
|
|
|
static const char kDesktopImageKey[] = DG_BACKGROUND "/picture_filename";
|
|
static const char kDesktopOptionsKey[] = DG_BACKGROUND "/picture_options";
|
|
static const char kDesktopDrawBGKey[] = DG_BACKGROUND "/draw_background";
|
|
static const char kDesktopColorKey[] = DG_BACKGROUND "/primary_color";
|
|
|
|
nsresult
|
|
nsGNOMEShellService::Init()
|
|
{
|
|
nsresult rv;
|
|
|
|
// GConf and GnomeVFS _must_ be available, or we do not allow
|
|
// CreateInstance to succeed.
|
|
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
nsCOMPtr<nsIGnomeVFSService> vfs =
|
|
do_GetService(NS_GNOMEVFSSERVICE_CONTRACTID);
|
|
|
|
if (!gconf || !vfs)
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
// Check G_BROKEN_FILENAMES. If it's set, then filenames in glib use
|
|
// the locale encoding. If it's not set, they use UTF-8.
|
|
mUseLocaleFilenames = PR_GetEnv("G_BROKEN_FILENAMES") != nsnull;
|
|
|
|
nsCOMPtr<nsIProperties> dirSvc
|
|
(do_GetService("@mozilla.org/file/directory_service;1"));
|
|
NS_ENSURE_TRUE(dirSvc, NS_ERROR_NOT_AVAILABLE);
|
|
|
|
nsCOMPtr<nsILocalFile> appPath;
|
|
rv = dirSvc->Get(NS_XPCOM_CURRENT_PROCESS_DIR, NS_GET_IID(nsILocalFile),
|
|
getter_AddRefs(appPath));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
rv = appPath->AppendNative(NS_LITERAL_CSTRING(MOZ_APP_NAME));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
|
|
return appPath->GetNativePath(mAppPath);
|
|
}
|
|
|
|
NS_IMPL_ISUPPORTS1(nsGNOMEShellService, nsIShellService)
|
|
|
|
PRBool
|
|
nsGNOMEShellService::KeyMatchesAppName(const char *aKeyValue) const
|
|
{
|
|
|
|
gchar *commandPath;
|
|
if (mUseLocaleFilenames) {
|
|
gchar *nativePath = g_filename_from_utf8(aKeyValue, -1, NULL, NULL, NULL);
|
|
if (!nativePath) {
|
|
NS_ERROR("Error converting path to filesystem encoding");
|
|
return PR_FALSE;
|
|
}
|
|
|
|
commandPath = g_find_program_in_path(nativePath);
|
|
g_free(nativePath);
|
|
} else {
|
|
commandPath = g_find_program_in_path(aKeyValue);
|
|
}
|
|
|
|
if (!commandPath)
|
|
return PR_FALSE;
|
|
|
|
PRBool matches = mAppPath.Equals(commandPath);
|
|
g_free(commandPath);
|
|
return matches;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::IsDefaultBrowser(PRBool aStartupCheck,
|
|
PRBool* aIsDefaultBrowser)
|
|
{
|
|
*aIsDefaultBrowser = PR_FALSE;
|
|
if (aStartupCheck)
|
|
mCheckedThisSession = PR_TRUE;
|
|
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
|
|
PRBool enabled;
|
|
nsCAutoString handler;
|
|
|
|
for (unsigned int i = 0; i < NS_ARRAY_LENGTH(appProtocols); ++i) {
|
|
if (!appProtocols[i].essential)
|
|
continue;
|
|
|
|
handler.Truncate();
|
|
gconf->GetAppForProtocol(nsDependentCString(appProtocols[i].name),
|
|
&enabled, handler);
|
|
|
|
// The string will be something of the form: [/path/to/]browser "%s"
|
|
// We want to remove all of the parameters and get just the binary name.
|
|
|
|
gint argc;
|
|
gchar **argv;
|
|
|
|
if (g_shell_parse_argv(handler.get(), &argc, &argv, NULL) && argc > 0) {
|
|
handler.Assign(argv[0]);
|
|
g_strfreev(argv);
|
|
}
|
|
|
|
if (!KeyMatchesAppName(handler.get()) || !enabled)
|
|
return NS_OK; // the handler is disabled or set to another app
|
|
}
|
|
|
|
*aIsDefaultBrowser = PR_TRUE;
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::SetDefaultBrowser(PRBool aClaimAllTypes,
|
|
PRBool aForAllUsers)
|
|
{
|
|
#ifdef DEBUG
|
|
if (aForAllUsers)
|
|
NS_WARNING("Setting the default browser for all users is not yet supported");
|
|
#endif
|
|
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
|
|
nsCAutoString schemeList;
|
|
nsCAutoString appKeyValue(mAppPath);
|
|
appKeyValue.Append(" \"%s\"");
|
|
unsigned int i;
|
|
|
|
for (i = 0; i < NS_ARRAY_LENGTH(appProtocols); ++i) {
|
|
schemeList.Append(nsDependentCString(appProtocols[i].name));
|
|
schemeList.Append(',');
|
|
|
|
if (appProtocols[i].essential || aClaimAllTypes) {
|
|
gconf->SetAppForProtocol(nsDependentCString(appProtocols[i].name),
|
|
appKeyValue);
|
|
}
|
|
}
|
|
|
|
if (aClaimAllTypes) {
|
|
nsCOMPtr<nsIGnomeVFSService> vfs =
|
|
do_GetService(NS_GNOMEVFSSERVICE_CONTRACTID);
|
|
|
|
nsCOMPtr<nsIStringBundleService> bundleService =
|
|
do_GetService(NS_STRINGBUNDLE_CONTRACTID);
|
|
NS_ENSURE_TRUE(bundleService, NS_ERROR_OUT_OF_MEMORY);
|
|
|
|
nsCOMPtr<nsIStringBundle> brandBundle;
|
|
bundleService->CreateBundle(BRAND_PROPERTIES, getter_AddRefs(brandBundle));
|
|
NS_ENSURE_TRUE(brandBundle, NS_ERROR_FAILURE);
|
|
|
|
nsString brandShortName, brandFullName;
|
|
brandBundle->GetStringFromName(NS_LITERAL_STRING("brandShortName").get(),
|
|
getter_Copies(brandShortName));
|
|
brandBundle->GetStringFromName(NS_LITERAL_STRING("brandFullName").get(),
|
|
getter_Copies(brandFullName));
|
|
|
|
// use brandShortName as the application id.
|
|
NS_ConvertUTF16toUTF8 id(brandShortName);
|
|
|
|
vfs->SetAppStringKey(id, nsIGnomeVFSService::APP_KEY_COMMAND, mAppPath);
|
|
vfs->SetAppStringKey(id, nsIGnomeVFSService::APP_KEY_NAME,
|
|
NS_ConvertUTF16toUTF8(brandFullName));
|
|
|
|
// We don't want to be the default handler for "file:", but we do
|
|
// want Nautilus to know that we support file: if the MIME type is
|
|
// one that we can handle.
|
|
|
|
schemeList.Append("file");
|
|
|
|
vfs->SetAppStringKey(id, nsIGnomeVFSService::APP_KEY_SUPPORTED_URI_SCHEMES,
|
|
schemeList);
|
|
|
|
vfs->SetAppStringKey(id, nsIGnomeVFSService::APP_KEY_EXPECTS_URIS,
|
|
NS_LITERAL_CSTRING("true"));
|
|
|
|
vfs->SetAppBoolKey(id, nsIGnomeVFSService::APP_KEY_CAN_OPEN_MULTIPLE,
|
|
PR_FALSE);
|
|
|
|
vfs->SetAppBoolKey(id, nsIGnomeVFSService::APP_KEY_REQUIRES_TERMINAL,
|
|
PR_FALSE);
|
|
|
|
// Copy icons/document.png to ~/.icons/firefox-document.png
|
|
nsCAutoString iconFilePath(mAppPath);
|
|
PRInt32 lastSlash = iconFilePath.RFindChar(PRUnichar('/'));
|
|
if (lastSlash == -1) {
|
|
NS_ERROR("no slash in executable path?");
|
|
} else {
|
|
iconFilePath.SetLength(lastSlash);
|
|
nsCOMPtr<nsILocalFile> iconFile;
|
|
NS_NewNativeLocalFile(iconFilePath, PR_FALSE, getter_AddRefs(iconFile));
|
|
if (iconFile) {
|
|
iconFile->AppendRelativeNativePath(NS_LITERAL_CSTRING("icons/document.png"));
|
|
|
|
nsCOMPtr<nsILocalFile> userIconPath;
|
|
NS_NewNativeLocalFile(nsDependentCString(PR_GetEnv("HOME")), PR_FALSE,
|
|
getter_AddRefs(userIconPath));
|
|
if (userIconPath) {
|
|
userIconPath->AppendNative(NS_LITERAL_CSTRING(".icons"));
|
|
iconFile->CopyToNative(userIconPath,
|
|
nsDependentCString(kDocumentIconPath));
|
|
}
|
|
}
|
|
}
|
|
|
|
for (i = 0; i < NS_ARRAY_LENGTH(appTypes); ++i) {
|
|
vfs->AddMimeType(id, nsDependentCString(appTypes[i].mimeType));
|
|
vfs->SetMimeExtensions(nsDependentCString(appTypes[i].mimeType),
|
|
nsDependentCString(appTypes[i].extensions));
|
|
vfs->SetAppForMimeType(nsDependentCString(appTypes[i].mimeType), id);
|
|
vfs->SetIconForMimeType(nsDependentCString(appTypes[i].mimeType),
|
|
NS_LITERAL_CSTRING(kDocumentIconPath));
|
|
}
|
|
|
|
vfs->SyncAppRegistry();
|
|
}
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::GetShouldCheckDefaultBrowser(PRBool* aResult)
|
|
{
|
|
// If we've already checked, the browser has been started and this is a
|
|
// new window open, and we don't want to check again.
|
|
if (mCheckedThisSession) {
|
|
*aResult = PR_FALSE;
|
|
return NS_OK;
|
|
}
|
|
|
|
nsCOMPtr<nsIPrefBranch> prefs;
|
|
nsCOMPtr<nsIPrefService> pserve(do_GetService(NS_PREFSERVICE_CONTRACTID));
|
|
if (pserve)
|
|
pserve->GetBranch("", getter_AddRefs(prefs));
|
|
|
|
prefs->GetBoolPref(PREF_CHECKDEFAULTBROWSER, aResult);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::SetShouldCheckDefaultBrowser(PRBool aShouldCheck)
|
|
{
|
|
nsCOMPtr<nsIPrefBranch> prefs;
|
|
nsCOMPtr<nsIPrefService> pserve(do_GetService(NS_PREFSERVICE_CONTRACTID));
|
|
if (pserve)
|
|
pserve->GetBranch("", getter_AddRefs(prefs));
|
|
|
|
prefs->SetBoolPref(PREF_CHECKDEFAULTBROWSER, aShouldCheck);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
static nsresult
|
|
WriteImage(const nsCString& aPath, gfxIImageFrame* aImage)
|
|
{
|
|
nsCOMPtr<nsIImage> img(do_GetInterface(aImage));
|
|
if (!img)
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
#ifndef MOZ_WIDGET_GTK2
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
#else
|
|
nsCOMPtr<nsIImageToPixbuf> imgToPixbuf =
|
|
do_GetService("@mozilla.org/widget/image-to-gdk-pixbuf;1");
|
|
if (!imgToPixbuf)
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
GdkPixbuf* pixbuf = imgToPixbuf->ConvertImageToPixbuf(img);
|
|
if (!pixbuf)
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
gboolean res = gdk_pixbuf_save(pixbuf, aPath.get(), "png", NULL, NULL);
|
|
|
|
g_object_unref(pixbuf);
|
|
return res ? NS_OK : NS_ERROR_FAILURE;
|
|
#endif
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::SetDesktopBackground(nsIDOMElement* aElement,
|
|
PRInt32 aPosition)
|
|
{
|
|
nsresult rv;
|
|
nsCOMPtr<gfxIImageFrame> gfxFrame;
|
|
|
|
nsCOMPtr<nsIImageLoadingContent> imageContent = do_QueryInterface(aElement, &rv);
|
|
if (!imageContent) return rv;
|
|
|
|
// get the image container
|
|
nsCOMPtr<imgIRequest> request;
|
|
rv = imageContent->GetRequest(nsIImageLoadingContent::CURRENT_REQUEST,
|
|
getter_AddRefs(request));
|
|
if (!request) return rv;
|
|
nsCOMPtr<imgIContainer> container;
|
|
rv = request->GetImage(getter_AddRefs(container));
|
|
if (!container) return rv;
|
|
|
|
// get the current frame, which holds the image data
|
|
container->GetCurrentFrame(getter_AddRefs(gfxFrame));
|
|
|
|
if (!gfxFrame)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// Write the background file to the home directory.
|
|
nsCAutoString filePath(PR_GetEnv("HOME"));
|
|
|
|
// get the product brand name from localized strings
|
|
nsString brandName;
|
|
nsCID bundleCID = NS_STRINGBUNDLESERVICE_CID;
|
|
nsCOMPtr<nsIStringBundleService> bundleService(do_GetService(bundleCID));
|
|
if (bundleService) {
|
|
nsCOMPtr<nsIStringBundle> brandBundle;
|
|
rv = bundleService->CreateBundle(BRAND_PROPERTIES,
|
|
getter_AddRefs(brandBundle));
|
|
if (NS_SUCCEEDED(rv) && brandBundle) {
|
|
rv = brandBundle->GetStringFromName(NS_LITERAL_STRING("brandShortName").get(),
|
|
getter_Copies(brandName));
|
|
NS_ENSURE_SUCCESS(rv, rv);
|
|
}
|
|
}
|
|
|
|
// build the file name
|
|
filePath.Append('/');
|
|
filePath.Append(NS_ConvertUTF16toUTF8(brandName));
|
|
filePath.Append("_wallpaper.png");
|
|
|
|
// write the image to a file in the home dir
|
|
rv = WriteImage(filePath, gfxFrame);
|
|
|
|
// if the file was written successfully, set it as the system wallpaper
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
|
|
nsCAutoString options;
|
|
if (aPosition == BACKGROUND_TILE)
|
|
options.Assign("wallpaper");
|
|
else if (aPosition == BACKGROUND_STRETCH)
|
|
options.Assign("stretched");
|
|
else
|
|
options.Assign("centered");
|
|
|
|
gconf->SetString(NS_LITERAL_CSTRING(kDesktopOptionsKey), options);
|
|
|
|
// Set the image to an empty string first to force a refresh
|
|
// (since we could be writing a new image on top of an existing
|
|
// Firefox_wallpaper.png and nautilus doesn't monitor the file for changes)
|
|
gconf->SetString(NS_LITERAL_CSTRING(kDesktopImageKey),
|
|
EmptyCString());
|
|
|
|
gconf->SetString(NS_LITERAL_CSTRING(kDesktopImageKey), filePath);
|
|
gconf->SetBool(NS_LITERAL_CSTRING(kDesktopDrawBGKey), PR_TRUE);
|
|
|
|
return rv;
|
|
}
|
|
|
|
#define COLOR_16_TO_8_BIT(_c) ((_c) >> 8)
|
|
#define COLOR_8_TO_16_BIT(_c) ((_c) << 8)
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::GetDesktopBackgroundColor(PRUint32 *aColor)
|
|
{
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
|
|
nsCAutoString background;
|
|
gconf->GetString(NS_LITERAL_CSTRING(kDesktopColorKey), background);
|
|
|
|
if (background.IsEmpty()) {
|
|
*aColor = 0;
|
|
return NS_OK;
|
|
}
|
|
|
|
GdkColor color;
|
|
gboolean success = gdk_color_parse(background.get(), &color);
|
|
|
|
NS_ENSURE_TRUE(success, NS_ERROR_FAILURE);
|
|
|
|
*aColor = COLOR_16_TO_8_BIT(color.red) << 16 |
|
|
COLOR_16_TO_8_BIT(color.green) << 8 |
|
|
COLOR_16_TO_8_BIT(color.blue);
|
|
return NS_OK;
|
|
}
|
|
|
|
static void
|
|
ColorToCString(PRUint32 aColor, nsCString& aResult)
|
|
{
|
|
#if GTK_CHECK_VERSION(2,12,0)
|
|
GdkColor color;
|
|
color.red = COLOR_8_TO_16_BIT(aColor >> 16);
|
|
color.green = COLOR_8_TO_16_BIT((aColor >> 8) & 0xff);
|
|
color.blue = COLOR_8_TO_16_BIT(aColor & 0xff);
|
|
|
|
gchar *colorString = gdk_color_to_string(&color);
|
|
aResult.Assign(colorString);
|
|
g_free(colorString);
|
|
|
|
#else // GTK 2.12.0
|
|
|
|
// The #rrrrggggbbbb format is used to match gdk_color_to_string()
|
|
char *buf = aResult.BeginWriting(13);
|
|
if (!buf)
|
|
return;
|
|
|
|
PRUint8 red = (aColor >> 16);
|
|
PRUint8 green = (aColor >> 8) & 0xff;
|
|
PRUint8 blue = aColor & 0xff;
|
|
|
|
PR_snprintf(buf, 14, "#%02x00%02x00%02x00", red, green, blue);
|
|
#endif // GTK 2.12.0
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::SetDesktopBackgroundColor(PRUint32 aColor)
|
|
{
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
|
|
nsCString colorString;
|
|
ColorToCString(aColor, colorString);
|
|
|
|
gconf->SetString(NS_LITERAL_CSTRING(kDesktopColorKey), colorString);
|
|
|
|
return NS_OK;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::OpenApplication(PRInt32 aApplication)
|
|
{
|
|
nsCAutoString scheme;
|
|
if (aApplication == APPLICATION_MAIL)
|
|
scheme.Assign("mailto");
|
|
else if (aApplication == APPLICATION_NEWS)
|
|
scheme.Assign("news");
|
|
else
|
|
return NS_ERROR_NOT_AVAILABLE;
|
|
|
|
nsCOMPtr<nsIGConfService> gconf = do_GetService(NS_GCONFSERVICE_CONTRACTID);
|
|
|
|
PRBool enabled;
|
|
nsCAutoString appCommand;
|
|
gconf->GetAppForProtocol(scheme, &enabled, appCommand);
|
|
|
|
if (!enabled)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// XXX we don't currently handle launching a terminal window.
|
|
// If the handler requires a terminal, bail.
|
|
PRBool requiresTerminal;
|
|
gconf->HandlerRequiresTerminal(scheme, &requiresTerminal);
|
|
if (requiresTerminal)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
// Perform shell argument expansion
|
|
int argc;
|
|
char **argv;
|
|
if (!g_shell_parse_argv(appCommand.get(), &argc, &argv, NULL))
|
|
return NS_ERROR_FAILURE;
|
|
|
|
char **newArgv = new char*[argc + 1];
|
|
int newArgc = 0;
|
|
|
|
// Run through the list of arguments. Copy all of them to the new
|
|
// argv except for %s, which we skip.
|
|
for (int i = 0; i < argc; ++i) {
|
|
if (strcmp(argv[i], "%s") != 0)
|
|
newArgv[newArgc++] = argv[i];
|
|
}
|
|
|
|
newArgv[newArgc] = nsnull;
|
|
|
|
gboolean err = g_spawn_async(NULL, newArgv, NULL, G_SPAWN_SEARCH_PATH,
|
|
NULL, NULL, NULL, NULL);
|
|
|
|
g_strfreev(argv);
|
|
delete[] newArgv;
|
|
|
|
return err ? NS_OK : NS_ERROR_FAILURE;
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::OpenApplicationWithURI(nsILocalFile* aApplication, const nsACString& aURI)
|
|
{
|
|
nsresult rv;
|
|
nsCOMPtr<nsIProcess> process =
|
|
do_CreateInstance("@mozilla.org/process/util;1", &rv);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
rv = process->Init(aApplication);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
const nsCString spec(aURI);
|
|
const char* specStr = spec.get();
|
|
PRUint32 pid;
|
|
return process->Run(PR_FALSE, &specStr, 1, &pid);
|
|
}
|
|
|
|
NS_IMETHODIMP
|
|
nsGNOMEShellService::GetDefaultFeedReader(nsILocalFile** _retval)
|
|
{
|
|
return NS_ERROR_NOT_IMPLEMENTED;
|
|
}
|