2022-03-04 11:44:11 +01:00
|
|
|
#include <hex/helpers/fs.hpp>
|
|
|
|
|
2022-02-18 22:34:54 +01:00
|
|
|
#include <hex/api/content_registry.hpp>
|
2023-01-11 23:31:25 +01:00
|
|
|
#include <hex/api/project_file_manager.hpp>
|
2023-03-26 11:02:51 +02:00
|
|
|
#include <hex/helpers/logger.hpp>
|
2023-05-21 13:21:53 +02:00
|
|
|
#include <hex/helpers/fmt.hpp>
|
2023-07-05 20:50:46 +02:00
|
|
|
#include <hex/helpers/utils_linux.hpp>
|
2023-01-11 23:31:25 +01:00
|
|
|
|
2021-12-03 00:00:25 +01:00
|
|
|
#include <xdg.hpp>
|
|
|
|
|
2021-09-03 02:33:15 +02:00
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
#include <windows.h>
|
|
|
|
#include <shlobj.h>
|
|
|
|
#elif defined(OS_LINUX)
|
|
|
|
#include <xdg.hpp>
|
2021-09-10 15:26:19 +02:00
|
|
|
#include <linux/limits.h>
|
2021-09-03 02:33:15 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <filesystem>
|
2023-07-09 12:53:31 +02:00
|
|
|
#include <utility>
|
2021-09-03 02:33:15 +02:00
|
|
|
|
2023-03-12 18:27:29 +01:00
|
|
|
#include <wolv/io/file.hpp>
|
2023-03-13 09:25:07 +01:00
|
|
|
#include <wolv/io/fs.hpp>
|
2023-07-09 12:53:31 +02:00
|
|
|
#include <wolv/utils/string.hpp>
|
2022-03-04 11:36:37 +01:00
|
|
|
|
2023-03-12 18:27:29 +01:00
|
|
|
namespace hex::fs {
|
2022-03-04 11:36:37 +01:00
|
|
|
|
2023-03-26 11:02:51 +02:00
|
|
|
static std::function<void(const std::string&)> s_fileBrowserErrorCallback;
|
|
|
|
void setFileBrowserErrorCallback(const std::function<void(const std::string&)> &callback) {
|
2022-09-20 15:47:59 +02:00
|
|
|
s_fileBrowserErrorCallback = callback;
|
|
|
|
}
|
|
|
|
|
2023-05-21 13:21:53 +02:00
|
|
|
// With help from https://github.com/owncloud/client/blob/cba22aa34b3677406e0499aadd126ce1d94637a2/src/gui/openfilemanager.cpp
|
|
|
|
|
|
|
|
void openFileExternal(const std::fs::path &filePath) {
|
|
|
|
if (!wolv::io::fs::exists(filePath))
|
|
|
|
return;
|
|
|
|
|
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
hex::unused(
|
|
|
|
ShellExecute(nullptr, "open", wolv::util::toUTF8String(filePath).c_str(), nullptr, nullptr, SW_SHOWNORMAL)
|
|
|
|
);
|
|
|
|
#elif defined(OS_MACOS)
|
|
|
|
hex::unused(system(
|
|
|
|
hex::format("open {}", wolv::util::toUTF8String(filePath)).c_str()
|
|
|
|
));
|
|
|
|
#elif defined(OS_LINUX)
|
2023-07-05 20:50:46 +02:00
|
|
|
executeCmd({"xdg-open", wolv::util::toUTF8String(filePath)});
|
2023-05-21 13:21:53 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void openFolderExternal(const std::fs::path &dirPath) {
|
|
|
|
if (!wolv::io::fs::exists(dirPath))
|
|
|
|
return;
|
|
|
|
|
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
hex::unused(system(
|
|
|
|
hex::format("explorer.exe {}", wolv::util::toUTF8String(dirPath)).c_str()
|
|
|
|
));
|
|
|
|
#elif defined(OS_MACOS)
|
|
|
|
hex::unused(system(
|
|
|
|
hex::format("open {}", wolv::util::toUTF8String(dirPath)).c_str()
|
|
|
|
));
|
|
|
|
#elif defined(OS_LINUX)
|
2023-07-05 20:50:46 +02:00
|
|
|
executeCmd({"xdg-open", wolv::util::toUTF8String(dirPath)});
|
2023-05-21 13:21:53 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void openFolderWithSelectionExternal(const std::fs::path &selectedFilePath) {
|
|
|
|
if (!wolv::io::fs::exists(selectedFilePath))
|
|
|
|
return;
|
|
|
|
|
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
hex::unused(system(
|
|
|
|
hex::format(R"(explorer.exe /select,"{}")", wolv::util::toUTF8String(selectedFilePath)).c_str()
|
|
|
|
));
|
|
|
|
#elif defined(OS_MACOS)
|
|
|
|
hex::unused(system(
|
|
|
|
hex::format(
|
|
|
|
R"(osascript -e 'tell application "Finder" to reveal POSIX file "{}"')",
|
|
|
|
wolv::util::toUTF8String(selectedFilePath)
|
|
|
|
).c_str()
|
|
|
|
));
|
|
|
|
system(R"(osascript -e 'tell application "Finder" to activate')");
|
|
|
|
#elif defined(OS_LINUX)
|
|
|
|
// fallback to only opening the folder for now
|
|
|
|
// TODO actually select the file
|
2023-07-05 20:50:46 +02:00
|
|
|
executeCmd({"xdg-open", wolv::util::toUTF8String(selectedFilePath.parent_path())});
|
2023-05-21 13:21:53 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-01-28 00:01:53 +01:00
|
|
|
bool openFileBrowser(DialogMode mode, const std::vector<nfdfilteritem_t> &validExtensions, const std::function<void(std::fs::path)> &callback, const std::string &defaultPath, bool multiple) {
|
2023-03-26 11:02:51 +02:00
|
|
|
NFD::ClearError();
|
|
|
|
|
|
|
|
if (NFD::Init() != NFD_OKAY) {
|
|
|
|
log::error("NFD init returned an error: {}", NFD::GetError());
|
|
|
|
if (s_fileBrowserErrorCallback != nullptr)
|
|
|
|
s_fileBrowserErrorCallback(NFD::GetError() ? NFD::GetError() : "No details");
|
|
|
|
return false;
|
|
|
|
}
|
2022-03-04 11:36:37 +01:00
|
|
|
|
2023-01-28 00:01:53 +01:00
|
|
|
NFD::UniquePathU8 outPath;
|
|
|
|
NFD::UniquePathSet outPaths;
|
2022-03-04 11:36:37 +01:00
|
|
|
nfdresult_t result;
|
|
|
|
switch (mode) {
|
|
|
|
case DialogMode::Open:
|
2023-01-28 00:01:53 +01:00
|
|
|
if (multiple)
|
|
|
|
result = NFD::OpenDialogMultiple(outPaths, validExtensions.data(), validExtensions.size(), defaultPath.empty() ? nullptr : defaultPath.c_str());
|
|
|
|
else
|
|
|
|
result = NFD::OpenDialog(outPath, validExtensions.data(), validExtensions.size(), defaultPath.empty() ? nullptr : defaultPath.c_str());
|
2022-03-04 11:36:37 +01:00
|
|
|
break;
|
|
|
|
case DialogMode::Save:
|
2023-01-28 00:01:53 +01:00
|
|
|
result = NFD::SaveDialog(outPath, validExtensions.data(), validExtensions.size(), defaultPath.empty() ? nullptr : defaultPath.c_str());
|
2022-03-04 11:36:37 +01:00
|
|
|
break;
|
|
|
|
case DialogMode::Folder:
|
2023-01-28 00:01:53 +01:00
|
|
|
result = NFD::PickFolder(outPath, defaultPath.empty() ? nullptr : defaultPath.c_str());
|
2022-03-04 11:36:37 +01:00
|
|
|
break;
|
|
|
|
default:
|
2023-02-09 23:07:04 +01:00
|
|
|
std::unreachable();
|
2022-03-04 11:36:37 +01:00
|
|
|
}
|
|
|
|
|
2022-09-20 15:47:59 +02:00
|
|
|
if (result == NFD_OKAY){
|
|
|
|
if(outPath != nullptr) {
|
2023-01-28 00:01:53 +01:00
|
|
|
callback(reinterpret_cast<char8_t*>(outPath.get()));
|
2022-09-20 15:47:59 +02:00
|
|
|
}
|
2023-01-28 00:01:53 +01:00
|
|
|
if (outPaths != nullptr) {
|
|
|
|
nfdpathsetsize_t numPaths = 0;
|
|
|
|
if (NFD::PathSet::Count(outPaths, numPaths) == NFD_OKAY) {
|
|
|
|
for (size_t i = 0; i < numPaths; i++) {
|
|
|
|
NFD::UniquePathSetPath path;
|
|
|
|
if (NFD::PathSet::GetPath(outPaths, i, path) == NFD_OKAY)
|
|
|
|
callback(reinterpret_cast<char8_t*>(path.get()));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (result == NFD_ERROR) {
|
2023-03-26 11:02:51 +02:00
|
|
|
log::error("Requested file dialog returned an error: {}", NFD::GetError());
|
2022-09-20 15:47:59 +02:00
|
|
|
if (s_fileBrowserErrorCallback != nullptr)
|
2023-03-26 11:02:51 +02:00
|
|
|
s_fileBrowserErrorCallback(NFD::GetError() ? NFD::GetError() : "No details");
|
2022-03-04 11:36:37 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
NFD::Quit();
|
|
|
|
|
|
|
|
return result == NFD_OKAY;
|
|
|
|
}
|
|
|
|
|
2023-03-26 11:02:23 +02:00
|
|
|
std::vector<std::fs::path> getDataPaths() {
|
2022-08-01 14:51:08 +02:00
|
|
|
std::vector<std::fs::path> paths;
|
2022-01-24 20:53:17 +01:00
|
|
|
|
2022-08-02 22:55:01 +02:00
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
|
|
|
|
if (!ImHexApi::System::isPortableVersion()) {
|
2022-08-01 14:51:08 +02:00
|
|
|
PWSTR wAppDataPath = nullptr;
|
2022-08-02 22:55:01 +02:00
|
|
|
if (SUCCEEDED(SHGetKnownFolderPath(FOLDERID_LocalAppData, KF_FLAG_CREATE, nullptr, &wAppDataPath))) {
|
|
|
|
paths.emplace_back(wAppDataPath);
|
|
|
|
CoTaskMemFree(wAppDataPath);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#elif defined(OS_MACOS)
|
2022-01-24 20:53:17 +01:00
|
|
|
|
2023-03-13 09:25:07 +01:00
|
|
|
paths.push_back(wolv::io::fs::getApplicationSupportDirectoryPath());
|
2022-03-22 09:34:26 +01:00
|
|
|
|
2022-08-02 22:55:01 +02:00
|
|
|
#elif defined(OS_LINUX)
|
2022-01-24 20:53:17 +01:00
|
|
|
|
2022-08-03 19:52:02 +02:00
|
|
|
paths.push_back(xdg::DataHomeDir());
|
2022-08-02 22:55:01 +02:00
|
|
|
|
2022-08-03 19:52:02 +02:00
|
|
|
auto dataDirs = xdg::DataDirs();
|
2022-08-02 22:55:01 +02:00
|
|
|
std::copy(dataDirs.begin(), dataDirs.end(), std::back_inserter(paths));
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2022-08-07 23:14:06 +02:00
|
|
|
for (auto &path : paths)
|
2022-08-02 22:55:01 +02:00
|
|
|
path = path / "imhex";
|
2022-08-04 08:56:58 +02:00
|
|
|
|
|
|
|
#if defined(OS_MACOS)
|
|
|
|
|
2023-03-13 09:25:07 +01:00
|
|
|
if (auto executablePath = wolv::io::fs::getExecutablePath(); executablePath.has_value())
|
2022-08-04 08:56:58 +02:00
|
|
|
paths.push_back(*executablePath);
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2023-03-12 18:27:29 +01:00
|
|
|
if (auto executablePath = wolv::io::fs::getExecutablePath(); executablePath.has_value())
|
2022-08-04 08:56:58 +02:00
|
|
|
paths.push_back(executablePath->parent_path());
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2022-08-02 22:55:01 +02:00
|
|
|
|
|
|
|
auto additionalDirs = ImHexApi::System::getAdditionalFolderPaths();
|
|
|
|
std::copy(additionalDirs.begin(), additionalDirs.end(), std::back_inserter(paths));
|
2022-03-22 09:34:26 +01:00
|
|
|
|
2023-01-11 23:31:25 +01:00
|
|
|
if (ProjectFile::hasPath()) {
|
|
|
|
paths.push_back(ProjectFile::getPath().parent_path());
|
|
|
|
}
|
|
|
|
|
2022-08-02 22:55:01 +02:00
|
|
|
return paths;
|
|
|
|
}
|
|
|
|
|
|
|
|
static std::vector<std::fs::path> getConfigPaths() {
|
|
|
|
#if defined(OS_WINDOWS)
|
|
|
|
return getDataPaths();
|
|
|
|
#elif defined(OS_MACOS)
|
|
|
|
return getDataPaths();
|
|
|
|
#elif defined(OS_LINUX)
|
2023-03-26 11:02:23 +02:00
|
|
|
return {xdg::ConfigHomeDir() / "imhex"};
|
2022-08-02 22:55:01 +02:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2023-01-01 01:01:24 +01:00
|
|
|
std::vector<std::fs::path> appendPath(std::vector<std::fs::path> paths, const std::fs::path &folder) {
|
2022-08-04 20:37:57 +02:00
|
|
|
for (auto &path : paths)
|
|
|
|
path = path / folder;
|
2022-08-02 22:55:01 +02:00
|
|
|
|
2022-08-04 20:37:57 +02:00
|
|
|
return paths;
|
2022-10-02 17:30:26 +02:00
|
|
|
}
|
2022-08-04 20:37:57 +02:00
|
|
|
|
|
|
|
std::vector<std::fs::path> getPluginPaths() {
|
|
|
|
std::vector<std::fs::path> paths = getDataPaths();
|
|
|
|
#if defined(OS_LINUX) && defined(SYSTEM_PLUGINS_LOCATION)
|
|
|
|
paths.push_back(SYSTEM_PLUGINS_LOCATION);
|
|
|
|
#endif
|
|
|
|
return paths;
|
|
|
|
}
|
2022-08-02 22:55:01 +02:00
|
|
|
|
|
|
|
|
2022-08-04 20:37:57 +02:00
|
|
|
std::vector<std::fs::path> getDefaultPaths(ImHexPath path, bool listNonExisting) {
|
|
|
|
std::vector<std::fs::path> result;
|
2022-01-24 20:53:17 +01:00
|
|
|
|
|
|
|
switch (path) {
|
2022-10-02 14:18:40 +02:00
|
|
|
case ImHexPath::END:
|
|
|
|
return { };
|
2022-08-02 22:55:01 +02:00
|
|
|
case ImHexPath::Constants:
|
|
|
|
result = appendPath(getDataPaths(), "constants");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
2022-08-02 22:55:01 +02:00
|
|
|
case ImHexPath::Config:
|
|
|
|
result = appendPath(getConfigPaths(), "config");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
2022-08-02 22:55:01 +02:00
|
|
|
case ImHexPath::Encodings:
|
|
|
|
result = appendPath(getDataPaths(), "encodings");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
2022-08-02 22:55:01 +02:00
|
|
|
case ImHexPath::Logs:
|
2023-03-26 11:02:23 +02:00
|
|
|
result = appendPath(getDataPaths(), "logs");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
|
|
|
case ImHexPath::Plugins:
|
2022-08-04 20:37:57 +02:00
|
|
|
result = appendPath(getPluginPaths(), "plugins");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
2023-01-07 10:32:01 +01:00
|
|
|
case ImHexPath::Libraries:
|
|
|
|
result = appendPath(getPluginPaths(), "lib");
|
|
|
|
break;
|
2022-02-01 22:09:44 +01:00
|
|
|
case ImHexPath::Resources:
|
2022-08-02 22:55:01 +02:00
|
|
|
result = appendPath(getDataPaths(), "resources");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
2022-08-02 22:55:01 +02:00
|
|
|
case ImHexPath::Magic:
|
|
|
|
result = appendPath(getDataPaths(), "magic");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
|
|
|
case ImHexPath::Patterns:
|
2022-08-02 22:55:01 +02:00
|
|
|
result = appendPath(getDataPaths(), "patterns");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
|
|
|
case ImHexPath::PatternsInclude:
|
2022-08-02 22:55:01 +02:00
|
|
|
result = appendPath(getDataPaths(), "includes");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
|
|
|
case ImHexPath::Yara:
|
2022-08-02 22:55:01 +02:00
|
|
|
result = appendPath(getDataPaths(), "yara");
|
2022-02-01 22:09:44 +01:00
|
|
|
break;
|
2022-08-14 10:07:45 +02:00
|
|
|
case ImHexPath::Recent:
|
|
|
|
result = appendPath(getConfigPaths(), "recent");
|
|
|
|
break;
|
2022-10-02 14:18:40 +02:00
|
|
|
case ImHexPath::Scripts:
|
|
|
|
result = appendPath(getDataPaths(), "scripts");
|
|
|
|
break;
|
|
|
|
case ImHexPath::Inspectors:
|
|
|
|
result = appendPath(getDefaultPaths(ImHexPath::Scripts), "inspectors");
|
|
|
|
break;
|
2023-02-09 23:07:04 +01:00
|
|
|
case ImHexPath::Nodes:
|
|
|
|
result = appendPath(getDefaultPaths(ImHexPath::Scripts), "nodes");
|
|
|
|
break;
|
2022-12-29 19:26:00 +01:00
|
|
|
case ImHexPath::Themes:
|
|
|
|
result = appendPath(getDataPaths(), "themes");
|
|
|
|
break;
|
2023-05-11 18:44:50 +02:00
|
|
|
case ImHexPath::Layouts:
|
|
|
|
result = appendPath(getDataPaths(), "layouts");
|
|
|
|
break;
|
2022-01-24 20:53:17 +01:00
|
|
|
}
|
2022-02-21 22:47:56 +01:00
|
|
|
|
2022-01-12 09:02:03 +01:00
|
|
|
if (!listNonExisting) {
|
2022-01-24 20:53:17 +01:00
|
|
|
result.erase(std::remove_if(result.begin(), result.end(), [](const auto &path) {
|
2023-03-12 18:27:29 +01:00
|
|
|
return !wolv::io::fs::isDirectory(path);
|
2022-08-02 22:55:01 +02:00
|
|
|
}), result.end());
|
2022-01-12 09:02:03 +01:00
|
|
|
}
|
2021-12-15 00:21:34 +01:00
|
|
|
|
|
|
|
return result;
|
2021-09-03 02:33:15 +02:00
|
|
|
}
|
|
|
|
|
2023-03-12 18:27:29 +01:00
|
|
|
bool isPathWritable(const std::fs::path &path) {
|
|
|
|
constexpr static auto TestFileName = "__imhex__tmp__";
|
|
|
|
{
|
|
|
|
wolv::io::File file(path / TestFileName, wolv::io::File::Mode::Read);
|
|
|
|
if (file.isValid()) {
|
|
|
|
if (!file.remove())
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
wolv::io::File file(path / TestFileName, wolv::io::File::Mode::Create);
|
|
|
|
bool result = file.isValid();
|
|
|
|
if (!file.remove())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-06-30 19:39:06 +02:00
|
|
|
std::fs::path toShortPath(const std::fs::path &path) {
|
|
|
|
#if defined(OS_WINDOWS)
|
2022-10-30 13:39:40 +01:00
|
|
|
size_t size = GetShortPathNameW(path.c_str(), nullptr, 0);
|
2022-06-30 19:39:06 +02:00
|
|
|
if (size == 0)
|
|
|
|
return path;
|
|
|
|
|
|
|
|
std::wstring newPath(size, 0x00);
|
|
|
|
GetShortPathNameW(path.c_str(), newPath.data(), newPath.size());
|
2022-10-30 13:39:40 +01:00
|
|
|
newPath.pop_back();
|
2022-06-30 19:39:06 +02:00
|
|
|
|
|
|
|
return newPath;
|
|
|
|
#else
|
|
|
|
return path;
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-02-18 22:34:54 +01:00
|
|
|
}
|