1
0
mirror of https://gitlab.com/OpenMW/openmw.git synced 2025-01-25 06:35:30 +00:00
OpenMW/components/files/windowspath.cpp

131 lines
3.4 KiB
C++
Raw Normal View History

#include "windowspath.hpp"
#if defined(_WIN32) || defined(__WINDOWS__)
#include <cstring>
#include <shlobj.h>
#include <shlwapi.h>
2015-09-24 15:21:42 +02:00
#include <WinReg.h>
2014-01-14 17:20:41 +04:00
#include <boost/locale.hpp>
namespace bconv = boost::locale::conv;
#include <components/debug/debuglog.hpp>
/**
* FIXME: Someone with Windows system should check this and correct if necessary
2014-01-14 17:20:41 +04:00
* FIXME: MAX_PATH is irrelevant for extended-length paths, i.e. \\?\...
*/
/**
* \namespace Files
*/
namespace Files
{
WindowsPath::WindowsPath(const std::string& application_name)
: mName(application_name)
{
2014-05-22 16:35:57 +04:00
/* Since on Windows boost::path.string() returns string of narrow
characters in local encoding, it is required to path::imbue()
with UTF-8 encoding (generated for empty name from boost::locale)
to handle Unicode in platform-agnostic way using std::string.
See boost::filesystem and boost::locale reference for details.
*/
boost::filesystem::path::imbue(boost::locale::generator().generate(""));
boost::filesystem::path localPath = getLocalPath();
if (!SetCurrentDirectoryA(localPath.string().c_str()))
Log(Debug::Warning) << "Error " << GetLastError() << " when changing current directory";
}
2013-12-26 00:28:19 +01:00
boost::filesystem::path WindowsPath::getUserConfigPath() const
{
boost::filesystem::path userPath(".");
2014-01-14 17:20:41 +04:00
WCHAR path[MAX_PATH + 1];
memset(path, 0, sizeof(path));
2018-10-09 10:21:12 +04:00
if(SUCCEEDED(SHGetFolderPathW(nullptr, CSIDL_PERSONAL | CSIDL_FLAG_CREATE, nullptr, 0, path)))
{
2014-01-14 17:20:41 +04:00
userPath = boost::filesystem::path(bconv::utf_to_utf<char>(path));
}
return userPath / "My Games" / mName;
}
2013-12-26 01:24:43 +01:00
boost::filesystem::path WindowsPath::getUserDataPath() const
{
// Have some chaos, windows people!
return getUserConfigPath();
}
2013-12-26 00:28:19 +01:00
boost::filesystem::path WindowsPath::getGlobalConfigPath() const
{
boost::filesystem::path globalPath(".");
2014-01-14 17:20:41 +04:00
WCHAR path[MAX_PATH + 1];
memset(path, 0, sizeof(path));
2018-10-09 10:21:12 +04:00
if(SUCCEEDED(SHGetFolderPathW(nullptr, CSIDL_PROGRAM_FILES | CSIDL_FLAG_CREATE, nullptr, 0, path)))
{
2014-01-14 17:20:41 +04:00
globalPath = boost::filesystem::path(bconv::utf_to_utf<char>(path));
}
return globalPath / mName;
}
boost::filesystem::path WindowsPath::getLocalPath() const
{
boost::filesystem::path localPath("./");
WCHAR path[MAX_PATH + 1];
memset(path, 0, sizeof(path));
if (GetModuleFileNameW(nullptr, path, MAX_PATH + 1) > 0)
{
localPath = boost::filesystem::path(bconv::utf_to_utf<char>(path)).parent_path() / "/";
}
// lookup exe path
return localPath;
}
boost::filesystem::path WindowsPath::getGlobalDataPath() const
{
2013-12-28 18:16:01 +01:00
return getGlobalConfigPath();
}
2012-09-02 19:40:26 +02:00
boost::filesystem::path WindowsPath::getCachePath() const
{
2013-12-28 18:16:01 +01:00
return getUserConfigPath() / "cache";
2012-09-02 19:40:26 +02:00
}
boost::filesystem::path WindowsPath::getInstallPath() const
{
boost::filesystem::path installPath("");
HKEY hKey;
LPCTSTR regkey = TEXT("SOFTWARE\\Bethesda Softworks\\Morrowind");
if (RegOpenKeyEx(HKEY_LOCAL_MACHINE, regkey, 0, KEY_READ | KEY_WOW64_32KEY, &hKey) == ERROR_SUCCESS)
{
//Key existed, let's try to read the install dir
2012-01-17 09:02:45 +01:00
std::vector<char> buf(512);
int len = 512;
2018-10-09 10:21:12 +04:00
if (RegQueryValueEx(hKey, TEXT("Installed Path"), nullptr, nullptr, (LPBYTE)&buf[0], (LPDWORD)&len) == ERROR_SUCCESS)
{
2012-01-17 09:02:45 +01:00
installPath = &buf[0];
}
RegCloseKey(hKey);
}
return installPath;
}
} /* namespace Files */
#endif /* defined(_WIN32) || defined(__WINDOWS__) */