2021-08-18 22:36:46 +02:00
|
|
|
#include "window.hpp"
|
|
|
|
|
|
|
|
#if defined(OS_LINUX)
|
|
|
|
|
2022-07-02 17:53:13 +02:00
|
|
|
#include <hex/api/imhex_api.hpp>
|
2022-02-01 18:09:40 +01:00
|
|
|
#include <hex/api/content_registry.hpp>
|
2023-11-18 14:50:43 +01:00
|
|
|
#include <hex/api/event_manager.hpp>
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2021-09-16 22:34:48 +02:00
|
|
|
#include <hex/helpers/utils.hpp>
|
2023-07-05 20:50:46 +02:00
|
|
|
#include <hex/helpers/utils_linux.hpp>
|
2022-01-17 20:06:00 +01:00
|
|
|
#include <hex/helpers/logger.hpp>
|
2024-06-22 12:57:09 +02:00
|
|
|
#include <hex/helpers/default_paths.hpp>
|
2022-01-17 20:06:00 +01:00
|
|
|
|
2023-03-18 11:47:42 +01:00
|
|
|
#include <wolv/utils/core.hpp>
|
|
|
|
|
2021-09-16 22:23:51 +02:00
|
|
|
#include <nlohmann/json.hpp>
|
|
|
|
#include <sys/wait.h>
|
2022-01-17 20:06:00 +01:00
|
|
|
#include <unistd.h>
|
2021-09-16 22:23:51 +02:00
|
|
|
|
2022-08-04 09:51:07 +02:00
|
|
|
#include <imgui_impl_glfw.h>
|
2023-05-27 17:45:41 +02:00
|
|
|
#include <string.h>
|
|
|
|
#include <ranges>
|
2022-08-04 09:51:07 +02:00
|
|
|
|
2024-07-21 20:28:37 +02:00
|
|
|
#if defined(IMHEX_HAS_FONTCONFIG)
|
|
|
|
#include <fontconfig/fontconfig.h>
|
|
|
|
#endif
|
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
namespace hex {
|
2021-08-18 22:36:46 +02:00
|
|
|
|
2023-05-27 17:45:41 +02:00
|
|
|
bool isFileInPath(const std::fs::path &filename) {
|
|
|
|
auto optPathVar = hex::getEnvironmentVariable("PATH");
|
|
|
|
if (!optPathVar.has_value()) {
|
|
|
|
log::error("Could not find variable named PATH");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
for (auto dir : std::views::split(optPathVar.value(), ':')) {
|
|
|
|
if (std::fs::exists(std::fs::path(std::string_view(dir)) / filename)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void nativeErrorMessage(const std::string &message) {
|
|
|
|
log::fatal(message);
|
|
|
|
if (isFileInPath("zenity")) {
|
|
|
|
executeCmd({"zenity", "--error", "--text", message});
|
2023-11-10 20:47:08 +01:00
|
|
|
} else if (isFileInPath("notify-send")) {
|
2023-05-27 17:45:41 +02:00
|
|
|
executeCmd({"notify-send", "-i", "script-error", "Error", message});
|
2023-10-04 12:00:32 +02:00
|
|
|
} // Hopefully one of these commands is installed
|
2023-05-27 17:45:41 +02:00
|
|
|
}
|
|
|
|
|
2024-07-21 20:28:37 +02:00
|
|
|
#if defined(IMHEX_HAS_FONTCONFIG)
|
|
|
|
static bool enumerateFontConfig() {
|
|
|
|
if (!FcInit())
|
|
|
|
return false;
|
|
|
|
|
|
|
|
ON_SCOPE_EXIT { FcFini(); };
|
|
|
|
|
|
|
|
auto fonts = FcConfigGetFonts(nullptr, FcSetSystem);
|
|
|
|
if (fonts == nullptr)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (int i = 0; i < fonts->nfont; ++i) {
|
|
|
|
auto font = fonts->fonts[i];
|
|
|
|
FcChar8 *file, *fullName;
|
|
|
|
|
|
|
|
if (FcPatternGetString(font, FC_FILE, 0, &file) != FcResultMatch) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FcPatternGetString(font, FC_FULLNAME, 0, &fullName) != FcResultMatch
|
|
|
|
&& FcPatternGetString(font, FC_FAMILY, 0, &fullName) != FcResultMatch) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
registerFont(reinterpret_cast<const char *>(fullName), reinterpret_cast<const char *>(file));
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2024-02-24 22:46:52 +01:00
|
|
|
void enumerateFonts() {
|
2024-07-21 20:28:37 +02:00
|
|
|
#if defined(IMHEX_HAS_FONTCONFIG)
|
|
|
|
if (enumerateFontConfig())
|
|
|
|
return;
|
|
|
|
#endif
|
|
|
|
|
2024-02-24 22:46:52 +01:00
|
|
|
const std::array FontDirectories = {
|
|
|
|
"/usr/share/fonts",
|
|
|
|
"/usr/local/share/fonts",
|
|
|
|
"~/.fonts",
|
|
|
|
"~/.local/share/fonts"
|
|
|
|
};
|
|
|
|
|
|
|
|
for (const auto &directory : FontDirectories) {
|
|
|
|
if (!std::fs::exists(directory))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for (const auto &entry : std::fs::recursive_directory_iterator(directory)) {
|
|
|
|
if (!entry.exists())
|
|
|
|
continue;
|
|
|
|
if (!entry.is_regular_file())
|
|
|
|
continue;
|
|
|
|
if (entry.path().extension() != ".ttf" && entry.path().extension() != ".otf")
|
|
|
|
continue;
|
|
|
|
|
|
|
|
registerFont(entry.path().stem().c_str(), entry.path().c_str());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-05-03 21:39:31 +02:00
|
|
|
void Window::configureGLFW() {
|
2024-07-16 17:58:19 +02:00
|
|
|
#if defined(GLFW_SCALE_FRAMEBUFFER)
|
|
|
|
glfwWindowHint(GLFW_SCALE_FRAMEBUFFER, GLFW_FALSE);
|
|
|
|
#endif
|
|
|
|
|
2024-05-03 21:39:31 +02:00
|
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
|
|
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 0);
|
|
|
|
glfwWindowHint(GLFW_DECORATED, ImHexApi::System::isBorderlessWindowModeEnabled() ? GL_FALSE : GL_TRUE);
|
|
|
|
glfwWindowHint(GLFW_TRANSPARENT_FRAMEBUFFER, GLFW_TRUE);
|
|
|
|
|
|
|
|
#if defined(GLFW_WAYLAND_APP_ID)
|
|
|
|
glfwWindowHintString(GLFW_WAYLAND_APP_ID, "imhex");
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
void Window::initNative() {
|
2024-02-01 12:09:43 +01:00
|
|
|
log::impl::enableColorPrinting();
|
|
|
|
|
2023-01-07 10:32:01 +01:00
|
|
|
// Add plugin library folders to dll search path
|
2024-06-22 12:57:09 +02:00
|
|
|
for (const auto &path : paths::Libraries.read()) {
|
2023-01-07 10:32:01 +01:00
|
|
|
if (std::fs::exists(path))
|
2023-01-07 10:56:03 +01:00
|
|
|
setenv("LD_LIBRARY_PATH", hex::format("{};{}", hex::getEnvironmentVariable("LD_LIBRARY_PATH").value_or(""), path.string().c_str()).c_str(), true);
|
2023-01-07 10:32:01 +01:00
|
|
|
}
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Redirect stdout to log file if we're not running in a terminal
|
2022-01-24 20:53:17 +01:00
|
|
|
if (!isatty(STDOUT_FILENO)) {
|
2023-06-18 22:32:55 +02:00
|
|
|
log::impl::redirectToFile();
|
2021-08-22 20:24:42 +02:00
|
|
|
}
|
2024-02-24 22:46:52 +01:00
|
|
|
|
|
|
|
enumerateFonts();
|
2022-01-24 20:53:17 +01:00
|
|
|
}
|
2021-08-22 20:24:42 +02:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
void Window::setupNativeWindow() {
|
2022-07-02 16:22:38 +02:00
|
|
|
bool themeFollowSystem = ImHexApi::System::usesSystemThemeDetection();
|
2023-12-08 10:29:44 +01:00
|
|
|
EventOSThemeChanged::subscribe(this, [themeFollowSystem] {
|
2022-01-24 20:53:17 +01:00
|
|
|
if (!themeFollowSystem) return;
|
2021-09-16 22:23:51 +02:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
std::array<char, 128> buffer = { 0 };
|
|
|
|
std::string result;
|
2021-08-18 22:36:46 +02:00
|
|
|
|
2024-01-25 20:46:12 +01:00
|
|
|
// Ask dbus for the current theme. 1 for Dark, 2 for Light, 0 for default (Dark for ImHex)
|
|
|
|
// https://flatpak.github.io/xdg-desktop-portal/docs/doc-org.freedesktop.portal.Settings.html
|
2023-12-04 23:35:48 +01:00
|
|
|
FILE *pipe = popen("dbus-send --session --print-reply --dest=org.freedesktop.portal.Desktop /org/freedesktop/portal/desktop org.freedesktop.portal.Settings.Read string:'org.freedesktop.appearance' string:'color-scheme' 2>&1", "r");
|
2022-01-24 20:53:17 +01:00
|
|
|
if (pipe == nullptr) return;
|
2021-08-18 22:36:46 +02:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
while (fgets(buffer.data(), buffer.size(), pipe) != nullptr)
|
|
|
|
result += buffer.data();
|
2021-12-10 16:09:55 +01:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
auto exitCode = WEXITSTATUS(pclose(pipe));
|
|
|
|
if (exitCode != 0) return;
|
2021-12-10 16:09:55 +01:00
|
|
|
|
2024-01-25 20:46:12 +01:00
|
|
|
RequestChangeTheme::post(hex::containsIgnoreCase(result, "uint32 2") ? "Light" : "Dark");
|
2022-01-24 20:53:17 +01:00
|
|
|
});
|
2021-08-18 22:36:46 +02:00
|
|
|
|
2024-01-20 21:03:46 +01:00
|
|
|
// Register file drop callback
|
|
|
|
glfwSetDropCallback(m_window, [](GLFWwindow *, int count, const char **paths) {
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
|
|
EventFileDropped::post(reinterpret_cast<const char8_t *>(paths[i]));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2024-06-28 21:27:35 +02:00
|
|
|
glfwSetWindowRefreshCallback(m_window, [](GLFWwindow *window) {
|
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
|
|
|
win->fullFrame();
|
|
|
|
});
|
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
if (themeFollowSystem)
|
2023-12-08 10:29:44 +01:00
|
|
|
EventOSThemeChanged::post();
|
2022-01-24 20:53:17 +01:00
|
|
|
}
|
2021-08-18 22:36:46 +02:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
void Window::beginNativeWindowFrame() {
|
|
|
|
}
|
2021-08-18 23:12:27 +02:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
void Window::endNativeWindowFrame() {
|
|
|
|
}
|
2021-08-18 23:12:27 +02:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
}
|
|
|
|
|
2021-08-18 22:36:46 +02:00
|
|
|
#endif
|