mirror of
https://github.com/mozilla/gecko-dev.git
synced 2024-11-05 16:46:26 +00:00
9b0852aaf7
patch by unknown@simplemachines.org r=timeless rs=brendan
381 lines
14 KiB
C++
381 lines
14 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/* ***** 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 Mozilla Communicator client code, released
|
|
* March 31, 1998.
|
|
*
|
|
* The Initial Developer of the Original Code is
|
|
* Netscape Communications Corporation.
|
|
* Portions created by the Initial Developer are Copyright (C) 2000
|
|
* the Initial Developer. All Rights Reserved.
|
|
*
|
|
* Contributor(s):
|
|
* Conrad Carlen <conrad@ingress.com>
|
|
*
|
|
* 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 "winEmbedFileLocProvider.h"
|
|
#include "nsAppDirectoryServiceDefs.h"
|
|
#include "nsDirectoryServiceDefs.h"
|
|
#include "nsILocalFile.h"
|
|
#include "nsString.h"
|
|
#include "nsXPIDLString.h"
|
|
#include "nsCRT.h"
|
|
|
|
|
|
#include <windows.h>
|
|
#include <shlobj.h>
|
|
|
|
|
|
// WARNING: These hard coded names need to go away. They need to
|
|
// come from localizable resources
|
|
#define APP_REGISTRY_NAME NS_LITERAL_CSTRING("registry.dat")
|
|
|
|
#define PROFILE_ROOT_DIR_NAME NS_LITERAL_CSTRING("Profiles")
|
|
#define DEFAULTS_DIR_NAME NS_LITERAL_CSTRING("defaults")
|
|
#define DEFAULTS_PREF_DIR_NAME NS_LITERAL_CSTRING("pref")
|
|
#define DEFAULTS_PROFILE_DIR_NAME NS_LITERAL_CSTRING("profile")
|
|
#define RES_DIR_NAME NS_LITERAL_CSTRING("res")
|
|
#define CHROME_DIR_NAME NS_LITERAL_CSTRING("chrome")
|
|
#define PLUGINS_DIR_NAME NS_LITERAL_CSTRING("plugins")
|
|
#define SEARCH_DIR_NAME NS_LITERAL_CSTRING("searchplugins")
|
|
#define COMPONENTS_DIR_NAME NS_LITERAL_CSTRING("components")
|
|
|
|
//*****************************************************************************
|
|
// winEmbedFileLocProvider::Constructor/Destructor
|
|
//*****************************************************************************
|
|
|
|
winEmbedFileLocProvider::winEmbedFileLocProvider(const char* productDirName)
|
|
{
|
|
strncpy(mProductDirName, productDirName, sizeof(mProductDirName) - 1);
|
|
mProductDirName[sizeof(mProductDirName) - 1] = '\0';
|
|
}
|
|
|
|
winEmbedFileLocProvider::~winEmbedFileLocProvider()
|
|
{
|
|
}
|
|
|
|
|
|
//*****************************************************************************
|
|
// winEmbedFileLocProvider::nsISupports
|
|
//*****************************************************************************
|
|
|
|
NS_IMPL_ISUPPORTS1(winEmbedFileLocProvider, nsIDirectoryServiceProvider)
|
|
|
|
//*****************************************************************************
|
|
// winEmbedFileLocProvider::nsIDirectoryServiceProvider
|
|
//*****************************************************************************
|
|
|
|
NS_IMETHODIMP
|
|
winEmbedFileLocProvider::GetFile(const char *prop, PRBool *persistent, nsIFile **_retval)
|
|
{
|
|
nsCOMPtr<nsILocalFile> localFile;
|
|
nsresult rv = NS_ERROR_FAILURE;
|
|
|
|
*_retval = nsnull;
|
|
*persistent = PR_TRUE;
|
|
|
|
if (nsCRT::strcmp(prop, NS_APP_APPLICATION_REGISTRY_DIR) == 0)
|
|
{
|
|
rv = GetProductDirectory(getter_AddRefs(localFile));
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_APPLICATION_REGISTRY_FILE) == 0)
|
|
{
|
|
rv = GetProductDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendNative(APP_REGISTRY_NAME);
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_DEFAULTS_50_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(DEFAULTS_DIR_NAME);
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_PREF_DEFAULTS_50_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv)) {
|
|
rv = localFile->AppendRelativeNativePath(DEFAULTS_DIR_NAME);
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(DEFAULTS_PREF_DIR_NAME);
|
|
}
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_PROFILE_DEFAULTS_NLOC_50_DIR) == 0 ||
|
|
nsCRT::strcmp(prop, NS_APP_PROFILE_DEFAULTS_50_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv)) {
|
|
rv = localFile->AppendRelativeNativePath(DEFAULTS_DIR_NAME);
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(DEFAULTS_PROFILE_DIR_NAME);
|
|
}
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_USER_PROFILES_ROOT_DIR) == 0)
|
|
{
|
|
rv = GetDefaultUserProfileRoot(getter_AddRefs(localFile));
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_RES_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(RES_DIR_NAME);
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_CHROME_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(CHROME_DIR_NAME);
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_PLUGINS_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(PLUGINS_DIR_NAME);
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_APP_SEARCH_DIR) == 0)
|
|
{
|
|
rv = CloneMozBinDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(SEARCH_DIR_NAME);
|
|
}
|
|
//---------------------------------------------------------------
|
|
// Note that by returning a valid localFile's for NS_GRE_DIR and
|
|
// NS_GRE_COMPONENT_DIR your app is indicating to XPCOM that
|
|
// it found an GRE version with which it's compatible with and
|
|
// it intends to be "run against" that GRE
|
|
//
|
|
// Please see http://www.mozilla.org/projects/embedding/MRE.html
|
|
// for more info. on GRE
|
|
//---------------------------------------------------------------
|
|
else if (nsCRT::strcmp(prop, NS_GRE_DIR) == 0)
|
|
{
|
|
rv = GetGreDirectory(getter_AddRefs(localFile));
|
|
}
|
|
else if (nsCRT::strcmp(prop, NS_GRE_COMPONENT_DIR) == 0)
|
|
{
|
|
rv = GetGreDirectory(getter_AddRefs(localFile));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localFile->AppendRelativeNativePath(COMPONENTS_DIR_NAME);
|
|
}
|
|
|
|
if (localFile && NS_SUCCEEDED(rv))
|
|
return localFile->QueryInterface(NS_GET_IID(nsIFile), (void**)_retval);
|
|
|
|
return rv;
|
|
}
|
|
|
|
// Get the location of the GRE version we're compatible with from
|
|
// the registry
|
|
//
|
|
static char * GetGreLocationFromRegistry()
|
|
{
|
|
char szKey[256];
|
|
HKEY hRegKey = NULL;
|
|
DWORD dwLength = _MAX_PATH * sizeof(char);
|
|
long rc;
|
|
char keyValue[_MAX_PATH + 1];
|
|
char *pGreLocation = NULL;
|
|
|
|
// A couple of key points here:
|
|
// 1. Note the usage of the "Software\\Mozilla\\GRE" subkey - this allows
|
|
// us to have multiple versions of GREs on the same machine by having
|
|
// subkeys such as 1.0, 1.1, 2.0 etc. under it.
|
|
// 2. In this sample below we're looking for the location of GRE version 1.3
|
|
// i.e. we're compatible with GRE 1.3 and we're trying to find it's install
|
|
// location.
|
|
//
|
|
// Please see http://www.mozilla.org/projects/embedding/MRE.html for
|
|
// more info.
|
|
//
|
|
strcpy(szKey, "Software\\Mozilla\\GRE\\" MOZILLA_VERSION);
|
|
|
|
if (::RegOpenKeyEx(HKEY_LOCAL_MACHINE, szKey, 0, KEY_QUERY_VALUE, &hRegKey) == ERROR_SUCCESS)
|
|
{
|
|
if ((rc = ::RegQueryValueEx(hRegKey, "GreHome", NULL, NULL, (BYTE *)keyValue, &dwLength))==ERROR_SUCCESS)
|
|
{
|
|
pGreLocation = ::strdup(keyValue);
|
|
::RegCloseKey(hRegKey);
|
|
}
|
|
}
|
|
|
|
return pGreLocation;
|
|
}
|
|
|
|
// Create and return the location of the GRE the application is
|
|
// currently using, if any, via the |aLocalFile| param
|
|
//
|
|
// If an embedding application is written to use an GRE it determines
|
|
// the compatible GRE's location by looking in the Windows registry
|
|
// In this case GetGreDirectory() creates a new localFile based on the
|
|
// GRE path it just read from the registry
|
|
//
|
|
// If the embedding application is not using an GRE and is running in
|
|
// a "regular" embedding scenario GetGreDirectory() simply returns a
|
|
// failure code indicating to the caller to fallback to a non-GRE
|
|
// based operation - which is the default mode of operation.
|
|
//
|
|
// Please see http://www.mozilla.org/projects/embedding/MRE.html for
|
|
// more information on the Mozilla Runtime Environment(GRE) and for
|
|
// the actual registry key whichs contains the GRE path, if any.
|
|
|
|
NS_METHOD winEmbedFileLocProvider::GetGreDirectory(nsILocalFile **aLocalFile)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aLocalFile);
|
|
nsresult rv = NS_ERROR_FAILURE;
|
|
|
|
// Get the path of the GRE which is compatible with our embedding application
|
|
// from the registry
|
|
//
|
|
char *pGreDir = GetGreLocationFromRegistry();
|
|
if(pGreDir)
|
|
{
|
|
nsCOMPtr<nsILocalFile> tempLocal;
|
|
rv = NS_NewNativeLocalFile(nsDependentCString(pGreDir), TRUE, getter_AddRefs(tempLocal));
|
|
|
|
if (tempLocal)
|
|
{
|
|
*aLocalFile = tempLocal;
|
|
NS_ADDREF(*aLocalFile);
|
|
rv = NS_OK;
|
|
}
|
|
|
|
::free(pGreDir);
|
|
}
|
|
|
|
return rv;
|
|
}
|
|
|
|
NS_METHOD winEmbedFileLocProvider::CloneMozBinDirectory(nsILocalFile **aLocalFile)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aLocalFile);
|
|
nsresult rv;
|
|
|
|
if (!mMozBinDirectory)
|
|
{
|
|
// Get the mozilla bin directory
|
|
// 1. Check the directory service first for NS_XPCOM_CURRENT_PROCESS_DIR
|
|
// This will be set if a directory was passed to NS_InitXPCOM
|
|
// 2. If that doesn't work, set it to be the current process directory
|
|
|
|
nsCOMPtr<nsIProperties> directoryService =
|
|
do_GetService(NS_DIRECTORY_SERVICE_CONTRACTID, &rv);
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
rv = directoryService->Get(NS_XPCOM_CURRENT_PROCESS_DIR, NS_GET_IID(nsIFile), getter_AddRefs(mMozBinDirectory));
|
|
if (NS_FAILED(rv)) {
|
|
rv = directoryService->Get(NS_OS_CURRENT_PROCESS_DIR, NS_GET_IID(nsIFile), getter_AddRefs(mMozBinDirectory));
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
}
|
|
}
|
|
|
|
nsCOMPtr<nsIFile> aFile;
|
|
rv = mMozBinDirectory->Clone(getter_AddRefs(aFile));
|
|
if (NS_FAILED(rv))
|
|
return rv;
|
|
|
|
nsCOMPtr<nsILocalFile> lfile = do_QueryInterface (aFile);
|
|
if (!lfile)
|
|
return NS_ERROR_FAILURE;
|
|
|
|
NS_IF_ADDREF(*aLocalFile = lfile);
|
|
return NS_OK;
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------------------------------------
|
|
// GetProductDirectory - Gets the directory which contains the application data folder
|
|
//
|
|
// WIN : <Application Data folder on user's machine>\Mozilla
|
|
//----------------------------------------------------------------------------------------
|
|
NS_METHOD winEmbedFileLocProvider::GetProductDirectory(nsILocalFile **aLocalFile)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aLocalFile);
|
|
|
|
nsresult rv;
|
|
PRBool exists;
|
|
nsCOMPtr<nsILocalFile> localDir;
|
|
|
|
nsCOMPtr<nsIProperties> directoryService =
|
|
do_GetService(NS_DIRECTORY_SERVICE_CONTRACTID, &rv);
|
|
if (NS_FAILED(rv)) return rv;
|
|
rv = directoryService->Get(NS_WIN_APPDATA_DIR, NS_GET_IID(nsILocalFile), getter_AddRefs(localDir));
|
|
if (NS_SUCCEEDED(rv))
|
|
rv = localDir->Exists(&exists);
|
|
if (NS_FAILED(rv) || !exists)
|
|
{
|
|
// On some Win95 machines, NS_WIN_APPDATA_DIR does not exist - revert to NS_WIN_WINDOWS_DIR
|
|
localDir = nsnull;
|
|
rv = directoryService->Get(NS_WIN_WINDOWS_DIR, NS_GET_IID(nsILocalFile), getter_AddRefs(localDir));
|
|
}
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
rv = localDir->AppendRelativeNativePath(nsDependentCString(mProductDirName));
|
|
if (NS_FAILED(rv)) return rv;
|
|
rv = localDir->Exists(&exists);
|
|
if (NS_SUCCEEDED(rv) && !exists)
|
|
rv = localDir->Create(nsIFile::DIRECTORY_TYPE, 0775);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
*aLocalFile = localDir;
|
|
NS_ADDREF(*aLocalFile);
|
|
|
|
return rv;
|
|
}
|
|
|
|
|
|
//----------------------------------------------------------------------------------------
|
|
// GetDefaultUserProfileRoot - Gets the directory which contains each user profile dir
|
|
//
|
|
// WIN : <Application Data folder on user's machine>\Mozilla\Users50
|
|
//----------------------------------------------------------------------------------------
|
|
NS_METHOD winEmbedFileLocProvider::GetDefaultUserProfileRoot(nsILocalFile **aLocalFile)
|
|
{
|
|
NS_ENSURE_ARG_POINTER(aLocalFile);
|
|
|
|
nsresult rv;
|
|
PRBool exists;
|
|
nsCOMPtr<nsILocalFile> localDir;
|
|
|
|
rv = GetProductDirectory(getter_AddRefs(localDir));
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
// These 3 platforms share this part of the path - do them as one
|
|
rv = localDir->AppendRelativeNativePath(PROFILE_ROOT_DIR_NAME);
|
|
if (NS_FAILED(rv)) return rv;
|
|
rv = localDir->Exists(&exists);
|
|
if (NS_SUCCEEDED(rv) && !exists)
|
|
rv = localDir->Create(nsIFile::DIRECTORY_TYPE, 0775);
|
|
if (NS_FAILED(rv)) return rv;
|
|
|
|
*aLocalFile = localDir;
|
|
NS_ADDREF(*aLocalFile);
|
|
|
|
return rv;
|
|
}
|
|
|