mirror of
https://github.com/WerWolv/ImHex.git
synced 2026-03-28 07:47:03 -05:00
<!-- Please provide as much information as possible about what your PR aims to do. PRs with no description will most likely be closed until more information is provided. If you're planing on changing fundamental behaviour or add big new features, please open a GitHub Issue first before starting to work on it. If it's not something big and you still want to contact us about it, feel free to do so ! --> ### Problem description <!-- Describe the bug that you fixed/feature request that you implemented, or link to an existing issue describing it --> ### Implementation description <!-- Explain what you did to correct the problem --> ### Screenshots <!-- If your change is visual, take a screenshot showing it. Ideally, make before/after sceenshots --> ### Additional things <!-- Anything else you would like to say -->
163 lines
4.9 KiB
C++
163 lines
4.9 KiB
C++
#include <algorithm>
|
|
#include <hex/helpers/default_paths.hpp>
|
|
|
|
#include <hex/api/imhex_api/system.hpp>
|
|
#include <hex/api/project_file_manager.hpp>
|
|
|
|
#include <ranges>
|
|
|
|
#if defined(OS_WINDOWS)
|
|
#include <windows.h>
|
|
#include <shlobj.h>
|
|
#elif defined(OS_LINUX) || defined(OS_WEB)
|
|
#include <xdg.hpp>
|
|
# endif
|
|
|
|
namespace hex::paths {
|
|
|
|
std::vector<std::fs::path> getDataPaths(bool includeSystemFolders) {
|
|
std::vector<std::fs::path> paths;
|
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
// In the portable Windows version, we just use the executable directory
|
|
// Prevent the use of the AppData folder here
|
|
if (!ImHexApi::System::isPortableVersion()) {
|
|
PWSTR wAppDataPath = nullptr;
|
|
if (SUCCEEDED(SHGetKnownFolderPath(FOLDERID_LocalAppData, KF_FLAG_CREATE, nullptr, &wAppDataPath))) {
|
|
paths.emplace_back(wAppDataPath);
|
|
CoTaskMemFree(wAppDataPath);
|
|
}
|
|
}
|
|
|
|
#elif defined(OS_MACOS)
|
|
|
|
paths.push_back(wolv::io::fs::getApplicationSupportDirectoryPath() / "imhex");
|
|
|
|
#elif defined(OS_LINUX) || defined(OS_WEB)
|
|
|
|
paths.push_back(xdg::DataHomeDir());
|
|
|
|
auto dataDirs = xdg::DataDirs();
|
|
std::ranges::copy(dataDirs, std::back_inserter(paths));
|
|
|
|
#endif
|
|
|
|
#if defined(OS_MACOS)
|
|
|
|
if (includeSystemFolders) {
|
|
if (auto executablePath = wolv::io::fs::getExecutablePath(); executablePath.has_value()) {
|
|
paths.push_back(executablePath->parent_path());
|
|
}
|
|
}
|
|
|
|
#else
|
|
|
|
for (auto &path : paths)
|
|
path = path / "imhex";
|
|
|
|
if (ImHexApi::System::isPortableVersion() || includeSystemFolders) {
|
|
if (auto executablePath = wolv::io::fs::getExecutablePath(); executablePath.has_value())
|
|
paths.push_back(executablePath->parent_path());
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
// Add additional data directories to the path
|
|
auto additionalDirs = ImHexApi::System::getAdditionalFolderPaths();
|
|
std::ranges::copy(additionalDirs, std::back_inserter(paths));
|
|
|
|
// Add the project file directory to the path, if one is loaded
|
|
if (ProjectFile::hasPath()) {
|
|
paths.push_back(ProjectFile::getPath().parent_path());
|
|
}
|
|
|
|
return paths;
|
|
}
|
|
|
|
std::vector<std::fs::path> getConfigPaths(bool includeSystemFolders) {
|
|
#if defined(OS_WINDOWS)
|
|
return getDataPaths(includeSystemFolders);
|
|
#elif defined(OS_MACOS)
|
|
return getDataPaths(includeSystemFolders);
|
|
#elif defined(OS_LINUX) || defined(OS_WEB)
|
|
std::ignore = includeSystemFolders;
|
|
return {xdg::ConfigHomeDir() / "imhex"};
|
|
#endif
|
|
}
|
|
|
|
static std::vector<std::fs::path> appendPath(std::vector<std::fs::path> paths, std::fs::path folder) {
|
|
folder.make_preferred();
|
|
|
|
for (auto &path : paths)
|
|
path = path / folder;
|
|
|
|
return paths;
|
|
}
|
|
|
|
static std::vector<std::fs::path> getPluginPaths() {
|
|
// If running from an AppImage, only allow loaded plugins from inside it
|
|
#if defined(OS_LINUX)
|
|
if(const char* appdir = std::getenv("APPDIR")) { // check for AppImage environment
|
|
return {std::string(appdir) + "/usr/lib/imhex"};
|
|
}
|
|
#endif
|
|
|
|
std::vector<std::fs::path> paths = getDataPaths(true);
|
|
|
|
// Add the system plugin directory to the path if one was provided at compile time
|
|
#if defined(OS_LINUX) && defined(SYSTEM_PLUGINS_LOCATION)
|
|
paths.emplace_back(SYSTEM_PLUGINS_LOCATION);
|
|
#endif
|
|
|
|
return paths;
|
|
}
|
|
|
|
namespace impl {
|
|
|
|
std::vector<std::fs::path> DefaultPath::read() const {
|
|
auto result = this->all();
|
|
|
|
std::erase_if(result, [](const auto &entryPath) {
|
|
return !wolv::io::fs::isDirectory(entryPath);
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
std::vector<std::fs::path> DefaultPath::write() const {
|
|
auto result = this->read();
|
|
|
|
std::erase_if(result, [](const auto &entryPath) {
|
|
return !hex::fs::isPathWritable(entryPath);
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
std::vector<std::fs::path> ConfigPath::all() const {
|
|
return appendPath(getConfigPaths(false), m_postfix);
|
|
}
|
|
|
|
std::vector<std::fs::path> DataPath::all() const {
|
|
return appendPath(getDataPaths(true), m_postfix);
|
|
}
|
|
|
|
std::vector<std::fs::path> DataPath::write() const {
|
|
auto result = appendPath(getDataPaths(false), m_postfix);
|
|
|
|
std::erase_if(result, [](const auto &entryPath) {
|
|
return !hex::fs::isPathWritable(entryPath);
|
|
});
|
|
|
|
return result;
|
|
}
|
|
|
|
std::vector<std::fs::path> PluginPath::all() const {
|
|
return appendPath(getPluginPaths(), m_postfix);
|
|
}
|
|
|
|
}
|
|
|
|
} |