2020-11-10 15:26:38 +01:00
|
|
|
#include "window.hpp"
|
|
|
|
|
2020-12-22 18:10:01 +01:00
|
|
|
#include <hex.hpp>
|
2021-08-29 22:15:18 +02:00
|
|
|
|
2022-02-01 18:09:40 +01:00
|
|
|
#include <hex/api/plugin_manager.hpp>
|
|
|
|
#include <hex/api/content_registry.hpp>
|
|
|
|
#include <hex/api/imhex_api.hpp>
|
2023-05-11 18:44:50 +02:00
|
|
|
#include <hex/api/layout_manager.hpp>
|
2023-11-18 14:50:43 +01:00
|
|
|
#include <hex/api/shortcut_manager.hpp>
|
2023-12-11 15:54:22 +01:00
|
|
|
#include <hex/api/workspace_manager.hpp>
|
2023-12-13 11:24:25 +01:00
|
|
|
#include <hex/api/tutorial_manager.hpp>
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2021-08-29 22:15:18 +02:00
|
|
|
#include <hex/helpers/utils.hpp>
|
2022-03-04 11:36:37 +01:00
|
|
|
#include <hex/helpers/fs.hpp>
|
2021-08-29 22:15:18 +02:00
|
|
|
#include <hex/helpers/logger.hpp>
|
2024-06-22 10:44:55 +02:00
|
|
|
#include <hex/helpers/default_paths.hpp>
|
2020-12-22 18:10:01 +01:00
|
|
|
|
2023-04-07 10:21:27 +02:00
|
|
|
#include <hex/ui/view.hpp>
|
|
|
|
#include <hex/ui/popup.hpp>
|
|
|
|
|
2021-03-06 13:09:20 +01:00
|
|
|
#include <chrono>
|
2021-08-17 13:41:44 +02:00
|
|
|
#include <csignal>
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2021-12-22 15:06:16 +01:00
|
|
|
#include <romfs/romfs.hpp>
|
|
|
|
|
2021-01-13 17:28:27 +01:00
|
|
|
#include <imgui.h>
|
|
|
|
#include <imgui_internal.h>
|
|
|
|
#include <imgui_impl_glfw.h>
|
|
|
|
#include <imgui_impl_opengl3.h>
|
2021-12-31 01:10:06 +01:00
|
|
|
#include <hex/ui/imgui_imhex_extensions.h>
|
2021-03-02 13:48:23 +01:00
|
|
|
#include <implot.h>
|
|
|
|
#include <implot_internal.h>
|
2024-12-17 23:51:45 +01:00
|
|
|
#include <implot3d.h>
|
|
|
|
#include <implot3d_internal.h>
|
2021-08-17 13:41:19 +02:00
|
|
|
#include <imnodes.h>
|
|
|
|
#include <imnodes_internal.h>
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2023-07-09 12:53:31 +02:00
|
|
|
#include <wolv/utils/string.hpp>
|
|
|
|
|
2020-11-10 15:26:38 +01:00
|
|
|
#include <GLFW/glfw3.h>
|
2023-12-19 23:21:20 +01:00
|
|
|
#include <hex/ui/toast.hpp>
|
2024-01-06 17:38:55 +01:00
|
|
|
#include <wolv/utils/guards.hpp>
|
2024-02-18 02:12:57 +01:00
|
|
|
#include <fmt/printf.h>
|
2025-01-04 15:35:19 +01:00
|
|
|
#include <fmt/chrono.h>
|
2021-08-29 14:18:45 +02:00
|
|
|
|
2020-11-10 15:26:38 +01:00
|
|
|
namespace hex {
|
|
|
|
|
2021-03-06 13:09:20 +01:00
|
|
|
using namespace std::literals::chrono_literals;
|
|
|
|
|
2022-02-15 23:07:48 +01:00
|
|
|
Window::Window() {
|
2024-04-10 21:04:57 +02:00
|
|
|
const static auto openEmergencyPopup = [this](const std::string &title){
|
|
|
|
TaskManager::doLater([this, title] {
|
2023-01-31 11:38:26 +01:00
|
|
|
for (const auto &provider : ImHexApi::Provider::getProviders())
|
|
|
|
ImHexApi::Provider::remove(provider, false);
|
|
|
|
|
|
|
|
ImGui::OpenPopup(title.c_str());
|
2024-04-10 21:04:57 +02:00
|
|
|
m_emergencyPopupOpen = true;
|
2023-01-31 11:38:26 +01:00
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Handle fatal error popups for errors detected during initialization
|
2021-04-18 20:24:42 +02:00
|
|
|
{
|
2022-02-01 18:09:40 +01:00
|
|
|
for (const auto &[argument, value] : ImHexApi::System::getInitArguments()) {
|
|
|
|
if (argument == "no-plugins") {
|
2023-01-31 11:38:26 +01:00
|
|
|
openEmergencyPopup("No Plugins");
|
2024-01-12 23:03:13 +01:00
|
|
|
} else if (argument == "duplicate-plugins") {
|
|
|
|
openEmergencyPopup("Duplicate Plugins loaded");
|
2021-04-18 20:24:42 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2021-04-17 15:46:26 +02:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Initialize the window
|
2020-11-10 15:26:38 +01:00
|
|
|
this->initGLFW();
|
|
|
|
this->initImGui();
|
2021-11-28 11:57:52 +01:00
|
|
|
this->setupNativeWindow();
|
2023-02-17 12:03:53 +01:00
|
|
|
this->registerEventHandlers();
|
|
|
|
|
2023-03-21 15:33:43 +01:00
|
|
|
ContentRegistry::Settings::impl::store();
|
2024-02-18 11:29:18 +01:00
|
|
|
ContentRegistry::Settings::impl::load();
|
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
EventWindowInitialized::post();
|
|
|
|
EventImHexStartupFinished::post();
|
2025-01-03 10:16:22 +01:00
|
|
|
RequestStartMigration::post();
|
2024-06-24 22:53:25 +02:00
|
|
|
|
|
|
|
TutorialManager::init();
|
2024-12-25 01:34:11 +01:00
|
|
|
|
|
|
|
#if defined(OS_MACOS)
|
|
|
|
ShortcutManager::enableMacOSMode();
|
|
|
|
#endif
|
2023-02-17 12:03:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
Window::~Window() {
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderDeleted::unsubscribe(this);
|
|
|
|
RequestCloseImHex::unsubscribe(this);
|
|
|
|
RequestUpdateWindowTitle::unsubscribe(this);
|
|
|
|
EventAbnormalTermination::unsubscribe(this);
|
|
|
|
RequestOpenPopup::unsubscribe(this);
|
2023-02-17 12:03:53 +01:00
|
|
|
|
2024-01-28 22:14:59 +01:00
|
|
|
EventWindowDeinitializing::post(m_window);
|
2023-12-11 15:54:22 +01:00
|
|
|
|
2024-05-18 20:32:34 +02:00
|
|
|
ContentRegistry::Settings::impl::store();
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
this->exitImGui();
|
|
|
|
this->exitGLFW();
|
|
|
|
}
|
2021-01-31 00:04:33 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
void Window::registerEventHandlers() {
|
2022-08-09 16:29:52 +02:00
|
|
|
// Initialize default theme
|
2023-12-08 10:29:44 +01:00
|
|
|
RequestChangeTheme::post("Dark");
|
2022-08-09 16:29:52 +02:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Handle the close window request by telling GLFW to shut down
|
2023-12-08 10:29:44 +01:00
|
|
|
RequestCloseImHex::subscribe(this, [this](bool noQuestions) {
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowShouldClose(m_window, GLFW_TRUE);
|
2021-08-21 13:55:21 +02:00
|
|
|
|
|
|
|
if (!noQuestions)
|
2023-12-19 13:10:25 +01:00
|
|
|
EventWindowClosing::post(m_window);
|
2021-02-01 19:03:45 +01:00
|
|
|
});
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Handle opening popups
|
2023-12-08 10:29:44 +01:00
|
|
|
RequestOpenPopup::subscribe(this, [this](auto name) {
|
2023-12-19 13:10:25 +01:00
|
|
|
std::scoped_lock lock(m_popupMutex);
|
2022-09-19 16:09:22 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_popupsToOpen.push_back(name);
|
2021-09-26 21:18:25 +02:00
|
|
|
});
|
2024-02-10 23:31:05 +01:00
|
|
|
|
2024-07-05 17:39:07 +02:00
|
|
|
EventDPIChanged::subscribe([this](float oldScaling, float newScaling) {
|
|
|
|
if (oldScaling == newScaling || oldScaling == 0 || newScaling == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
int width, height;
|
|
|
|
glfwGetWindowSize(m_window, &width, &height);
|
|
|
|
|
|
|
|
width = float(width) * newScaling / oldScaling;
|
|
|
|
height = float(height) * newScaling / oldScaling;
|
|
|
|
|
|
|
|
ImHexApi::System::impl::setMainWindowSize(width, height);
|
|
|
|
glfwSetWindowSize(m_window, width, height);
|
|
|
|
});
|
|
|
|
|
|
|
|
|
2024-02-10 23:31:05 +01:00
|
|
|
LayoutManager::registerLoadCallback([this](std::string_view line) {
|
|
|
|
int width = 0, height = 0;
|
|
|
|
sscanf(line.data(), "MainWindowSize=%d,%d", &width, &height);
|
|
|
|
|
|
|
|
if (width > 0 && height > 0) {
|
|
|
|
TaskManager::doLater([width, height, this]{
|
|
|
|
glfwSetWindowSize(m_window, width, height);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
2024-02-18 02:12:57 +01:00
|
|
|
void handleException() {
|
|
|
|
try {
|
|
|
|
throw;
|
|
|
|
} catch (const std::exception &e) {
|
|
|
|
log::fatal("Unhandled exception: {}", e.what());
|
2024-03-01 18:21:15 +01:00
|
|
|
EventCrashRecovered::post(e);
|
2024-02-18 02:12:57 +01:00
|
|
|
} catch (...) {
|
|
|
|
log::fatal("Unhandled exception: Unknown exception");
|
|
|
|
}
|
|
|
|
}
|
2023-10-04 12:00:32 +02:00
|
|
|
|
2024-02-18 02:12:57 +01:00
|
|
|
void errorRecoverLogCallback(void*, const char* fmt, ...) {
|
|
|
|
va_list args;
|
2023-09-07 20:33:49 +02:00
|
|
|
|
2024-02-18 02:12:57 +01:00
|
|
|
std::string message;
|
2024-03-01 18:21:15 +01:00
|
|
|
|
|
|
|
va_start(args, fmt);
|
2024-02-18 02:12:57 +01:00
|
|
|
message.resize(std::vsnprintf(nullptr, 0, fmt, args));
|
2024-03-01 18:21:15 +01:00
|
|
|
va_end(args);
|
2024-02-18 02:12:57 +01:00
|
|
|
|
2024-03-01 18:21:15 +01:00
|
|
|
va_start(args, fmt);
|
|
|
|
std::vsnprintf(message.data(), message.size(), fmt, args);
|
2024-02-18 02:12:57 +01:00
|
|
|
va_end(args);
|
|
|
|
|
2024-03-01 18:21:15 +01:00
|
|
|
message.resize(message.size() - 1);
|
|
|
|
|
2024-02-18 02:12:57 +01:00
|
|
|
log::error("{}", message);
|
|
|
|
}
|
2024-01-21 18:39:51 +01:00
|
|
|
|
2024-02-18 02:12:57 +01:00
|
|
|
void Window::fullFrame() {
|
2024-07-10 20:50:58 +02:00
|
|
|
[[maybe_unused]] static u32 crashWatchdog = 0;
|
2024-03-02 09:52:09 +01:00
|
|
|
|
2024-05-19 14:14:57 +02:00
|
|
|
if (auto g = ImGui::GetCurrentContext(); g == nullptr || g->WithinFrameScope) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2024-07-10 20:50:58 +02:00
|
|
|
#if !defined(DEBUG)
|
2024-02-18 02:12:57 +01:00
|
|
|
try {
|
2024-07-10 20:50:58 +02:00
|
|
|
#endif
|
|
|
|
|
|
|
|
// Render an entire frame
|
2024-02-18 02:12:57 +01:00
|
|
|
this->frameBegin();
|
|
|
|
this->frame();
|
|
|
|
this->frameEnd();
|
2024-03-02 09:52:09 +01:00
|
|
|
|
2024-07-10 20:50:58 +02:00
|
|
|
#if !defined(DEBUG)
|
2024-03-02 09:52:09 +01:00
|
|
|
// Feed the watchdog
|
|
|
|
crashWatchdog = 0;
|
2024-02-18 02:12:57 +01:00
|
|
|
} catch (...) {
|
2024-03-02 09:52:09 +01:00
|
|
|
// If an exception keeps being thrown, abort the application after 10 frames
|
|
|
|
// This is done to avoid the application getting stuck in an infinite loop of exceptions
|
|
|
|
crashWatchdog += 1;
|
|
|
|
if (crashWatchdog > 10) {
|
|
|
|
log::fatal("Crash watchdog triggered, aborting");
|
|
|
|
std::abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try to recover from the exception by bringing ImGui back into a working state
|
2024-02-23 17:48:23 +01:00
|
|
|
ImGui::EndFrame();
|
2024-03-01 20:55:03 +01:00
|
|
|
ImGui::UpdatePlatformWindows();
|
2024-03-02 09:52:09 +01:00
|
|
|
|
|
|
|
// Handle the exception
|
2024-02-18 02:12:57 +01:00
|
|
|
handleException();
|
|
|
|
}
|
2024-07-10 20:50:58 +02:00
|
|
|
#endif
|
2023-09-07 20:33:49 +02:00
|
|
|
}
|
|
|
|
|
2020-11-10 15:26:38 +01:00
|
|
|
void Window::loop() {
|
2024-06-27 17:11:24 +02:00
|
|
|
glfwShowWindow(m_window);
|
2023-12-19 13:10:25 +01:00
|
|
|
while (!glfwWindowShouldClose(m_window)) {
|
|
|
|
m_lastStartFrameTime = glfwGetTime();
|
2023-05-11 23:22:06 +02:00
|
|
|
|
2024-06-29 23:17:59 +02:00
|
|
|
{
|
|
|
|
int x = 0, y = 0;
|
|
|
|
int width = 0, height = 0;
|
|
|
|
glfwGetWindowPos(m_window, &x, &y);
|
|
|
|
glfwGetWindowSize(m_window, &width, &height);
|
|
|
|
|
|
|
|
ImHexApi::System::impl::setMainWindowPosition(x, y);
|
|
|
|
ImHexApi::System::impl::setMainWindowSize(width, height);
|
|
|
|
}
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
if (!glfwGetWindowAttrib(m_window, GLFW_VISIBLE) || glfwGetWindowAttrib(m_window, GLFW_ICONIFIED)) {
|
2023-02-17 12:03:53 +01:00
|
|
|
// If the application is minimized or not visible, don't render anything
|
2021-06-07 18:14:40 +02:00
|
|
|
glfwWaitEvents();
|
2021-12-16 23:48:52 +01:00
|
|
|
}
|
|
|
|
|
2024-02-18 02:12:57 +01:00
|
|
|
m_lastStartFrameTime = glfwGetTime();
|
|
|
|
|
|
|
|
static ImVec2 lastWindowSize = ImHexApi::System::getMainWindowSize();
|
|
|
|
if (ImHexApi::System::impl::isWindowResizable()) {
|
|
|
|
glfwSetWindowSizeLimits(m_window, 480_scaled, 360_scaled, GLFW_DONT_CARE, GLFW_DONT_CARE);
|
|
|
|
lastWindowSize = ImHexApi::System::getMainWindowSize();
|
|
|
|
} else {
|
|
|
|
glfwSetWindowSizeLimits(m_window, lastWindowSize.x, lastWindowSize.y, lastWindowSize.x, lastWindowSize.y);
|
|
|
|
}
|
|
|
|
|
2023-09-07 20:33:49 +02:00
|
|
|
this->fullFrame();
|
2023-08-11 22:03:30 +02:00
|
|
|
|
2023-12-20 13:42:42 +01:00
|
|
|
ImHexApi::System::impl::setLastFrameTime(glfwGetTime() - m_lastStartFrameTime);
|
|
|
|
|
2025-01-04 15:35:19 +01:00
|
|
|
{
|
|
|
|
while (true) {
|
|
|
|
glfwPollEvents();
|
2024-03-14 21:18:57 +01:00
|
|
|
|
2025-01-04 15:35:19 +01:00
|
|
|
if (ImHexApi::System::getTargetFPS() >= 200)
|
|
|
|
break;
|
2024-03-14 21:18:57 +01:00
|
|
|
|
2025-01-04 15:35:19 +01:00
|
|
|
{
|
|
|
|
std::unique_lock lock(m_sleepMutex);
|
|
|
|
m_sleepCondVar.wait_for(lock, std::chrono::microseconds(100));
|
|
|
|
if (m_sleepFlag.exchange(false))
|
|
|
|
break;
|
2024-02-28 20:16:15 +01:00
|
|
|
}
|
2023-08-11 22:03:30 +02:00
|
|
|
}
|
|
|
|
}
|
2023-12-11 11:42:33 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_lastFrameTime = glfwGetTime() - m_lastStartFrameTime;
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
2024-02-09 19:42:47 +01:00
|
|
|
|
|
|
|
// Hide the window as soon as the render loop exits to make the window
|
|
|
|
// disappear as soon as it's closed
|
|
|
|
glfwHideWindow(m_window);
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
void Window::frameBegin() {
|
2023-02-17 12:03:53 +01:00
|
|
|
// Start new ImGui Frame
|
2020-11-10 15:26:38 +01:00
|
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
|
|
ImGui_ImplGlfw_NewFrame();
|
|
|
|
ImGui::NewFrame();
|
|
|
|
|
2024-01-29 15:44:18 +01:00
|
|
|
EventFrameBegin::post();
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Handle all undocked floating windows
|
2022-01-24 20:53:17 +01:00
|
|
|
ImGuiViewport *viewport = ImGui::GetMainViewport();
|
2022-07-07 07:16:38 +02:00
|
|
|
ImGui::SetNextWindowPos(viewport->WorkPos);
|
2022-07-06 10:21:38 +02:00
|
|
|
ImGui::SetNextWindowSize(ImHexApi::System::getMainWindowSize() - ImVec2(0, ImGui::GetTextLineHeightWithSpacing()));
|
2020-11-10 15:26:38 +01:00
|
|
|
ImGui::SetNextWindowViewport(viewport->ID);
|
2023-05-11 23:56:51 +02:00
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 0.0F);
|
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowBorderSize, 0.0F);
|
2021-08-21 00:52:11 +02:00
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0, 0));
|
2020-11-30 00:03:12 +01:00
|
|
|
|
2024-04-10 21:04:57 +02:00
|
|
|
ImGuiWindowFlags windowFlags = ImGuiWindowFlags_NoDocking | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoNavFocus | ImGuiWindowFlags_NoBringToFrontOnFocus | ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoScrollWithMouse;
|
|
|
|
|
|
|
|
if (!m_emergencyPopupOpen)
|
|
|
|
windowFlags |= ImGuiWindowFlags_MenuBar;
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Render main dock space
|
2022-02-06 21:39:10 +01:00
|
|
|
if (ImGui::Begin("ImHexDockSpace", nullptr, windowFlags)) {
|
2021-08-21 00:52:11 +02:00
|
|
|
ImGui::PopStyleVar();
|
2023-11-08 11:53:26 +01:00
|
|
|
|
2021-12-10 16:09:55 +01:00
|
|
|
this->beginNativeWindowFrame();
|
2023-12-11 22:09:13 +01:00
|
|
|
} else {
|
|
|
|
ImGui::PopStyleVar();
|
2020-11-11 14:41:44 +01:00
|
|
|
}
|
2020-11-23 23:57:19 +01:00
|
|
|
ImGui::End();
|
2021-08-21 00:52:11 +02:00
|
|
|
ImGui::PopStyleVar(2);
|
2021-04-21 23:31:51 +02:00
|
|
|
|
2024-04-10 21:04:57 +02:00
|
|
|
// Plugin load error popups
|
|
|
|
// These are not translated because they should always be readable, no matter if any localization could be loaded or not
|
2022-02-01 23:57:48 +01:00
|
|
|
{
|
2024-06-27 17:09:20 +02:00
|
|
|
const static auto drawPluginFolderTable = [] {
|
2023-11-16 22:24:06 +01:00
|
|
|
ImGuiExt::UnderlinedText("Plugin folders");
|
2023-07-22 22:59:05 +02:00
|
|
|
if (ImGui::BeginTable("plugins", 2, ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_ScrollY | ImGuiTableFlags_SizingFixedFit, ImVec2(0, 100_scaled))) {
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::TableSetupScrollFreeze(0, 1);
|
|
|
|
ImGui::TableSetupColumn("Path", ImGuiTableColumnFlags_WidthStretch, 0.2);
|
|
|
|
ImGui::TableSetupColumn("Exists", ImGuiTableColumnFlags_WidthFixed, ImGui::GetTextLineHeight() * 3);
|
|
|
|
|
|
|
|
ImGui::TableHeadersRow();
|
|
|
|
|
2024-06-22 10:44:55 +02:00
|
|
|
for (const auto &path : paths::Plugins.all()) {
|
2022-06-09 16:58:40 +03:00
|
|
|
const auto filePath = path / "builtin.hexplug";
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::TableNextRow();
|
|
|
|
ImGui::TableNextColumn();
|
2023-03-12 18:43:05 +01:00
|
|
|
ImGui::TextUnformatted(wolv::util::toUTF8String(filePath).c_str());
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::TableNextColumn();
|
2023-12-23 21:09:41 +01:00
|
|
|
ImGui::TextUnformatted(wolv::io::fs::exists(filePath) ? "Yes" : "No");
|
2022-02-01 23:57:48 +01:00
|
|
|
}
|
|
|
|
ImGui::EndTable();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-04-10 21:04:57 +02:00
|
|
|
if (m_emergencyPopupOpen) {
|
|
|
|
const auto pos = ImHexApi::System::getMainWindowPosition();
|
|
|
|
const auto size = ImHexApi::System::getMainWindowSize();
|
|
|
|
ImGui::GetBackgroundDrawList()->AddRectFilled(pos, pos + size, ImGui::GetColorU32(ImGuiCol_WindowBg) | 0xFF000000);
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_ModalWindowDimBg, 0x00);
|
|
|
|
ON_SCOPE_EXIT { ImGui::PopStyleColor(); };
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// No plugins error popup
|
2024-04-10 21:04:57 +02:00
|
|
|
ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_Always, ImVec2(0.5F, 0.5F));
|
|
|
|
if (ImGui::BeginPopupModal("No Plugins", nullptr, ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoTitleBar)) {
|
|
|
|
ImGui::BringWindowToDisplayFront(ImGui::GetCurrentWindowRead());
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::TextUnformatted("No ImHex plugins loaded (including the built-in plugin)!");
|
|
|
|
ImGui::TextUnformatted("Make sure you installed ImHex correctly.");
|
|
|
|
ImGui::TextUnformatted("There should be at least a 'builtin.hexplug' file in your plugins folder.");
|
|
|
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
|
|
|
drawPluginFolderTable();
|
|
|
|
|
2023-07-22 22:59:05 +02:00
|
|
|
ImGui::NewLine();
|
2024-04-10 21:04:57 +02:00
|
|
|
if (ImGuiExt::DimmedButton("Close ImHex", ImVec2(ImGui::GetContentRegionAvail().x, 0)))
|
2023-07-22 22:59:05 +02:00
|
|
|
ImHexApi::System::closeImHex(true);
|
|
|
|
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::EndPopup();
|
|
|
|
}
|
|
|
|
|
2024-01-12 23:03:13 +01:00
|
|
|
// Duplicate plugins error popup
|
2024-04-10 21:04:57 +02:00
|
|
|
ImGui::SetNextWindowPos(ImGui::GetMainViewport()->GetCenter(), ImGuiCond_Always, ImVec2(0.5F, 0.5F));
|
|
|
|
if (ImGui::BeginPopupModal("Duplicate Plugins loaded", nullptr, ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoTitleBar)) {
|
|
|
|
ImGui::BringWindowToDisplayFront(ImGui::GetCurrentWindowRead());
|
2024-01-12 23:03:13 +01:00
|
|
|
ImGui::TextUnformatted("ImHex found and attempted to load multiple plugins with the same name!");
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::TextUnformatted("Make sure you installed ImHex correctly and, if needed,");
|
2024-01-12 23:03:13 +01:00
|
|
|
ImGui::TextUnformatted("cleaned up older installations correctly.");
|
|
|
|
ImGui::TextUnformatted("Each plugin should only ever be loaded once.");
|
2022-02-01 23:57:48 +01:00
|
|
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
|
|
|
drawPluginFolderTable();
|
|
|
|
|
2023-07-22 22:59:05 +02:00
|
|
|
ImGui::NewLine();
|
2024-04-10 21:04:57 +02:00
|
|
|
if (ImGuiExt::DimmedButton("Close ImHex", ImVec2(ImGui::GetContentRegionAvail().x, 0)))
|
2023-07-22 22:59:05 +02:00
|
|
|
ImHexApi::System::closeImHex(true);
|
|
|
|
|
2022-02-01 23:57:48 +01:00
|
|
|
ImGui::EndPopup();
|
|
|
|
}
|
2021-04-21 23:31:51 +02:00
|
|
|
}
|
2023-02-17 12:03:53 +01:00
|
|
|
|
|
|
|
// Open popups when plugins requested it
|
2024-12-06 23:38:57 +01:00
|
|
|
// We retry every frame until the popup actually opens
|
|
|
|
// It might not open the first time because another popup is already open
|
2022-09-19 16:09:22 +02:00
|
|
|
{
|
2023-12-19 13:10:25 +01:00
|
|
|
std::scoped_lock lock(m_popupMutex);
|
|
|
|
m_popupsToOpen.remove_if([](const auto &name) {
|
2022-09-19 16:09:22 +02:00
|
|
|
if (ImGui::IsPopupOpen(name.c_str()))
|
|
|
|
return true;
|
|
|
|
else
|
|
|
|
ImGui::OpenPopup(name.c_str());
|
|
|
|
|
|
|
|
return false;
|
|
|
|
});
|
|
|
|
}
|
2022-01-18 00:10:10 +01:00
|
|
|
|
2023-04-07 10:21:27 +02:00
|
|
|
// Draw popup stack
|
|
|
|
{
|
2023-05-10 22:29:17 +02:00
|
|
|
static bool positionSet = false;
|
|
|
|
static bool sizeSet = false;
|
2023-07-23 18:22:53 +02:00
|
|
|
static double popupDelay = -2.0;
|
2024-01-06 17:38:55 +01:00
|
|
|
static u32 displayFrameCount = 0;
|
2023-05-10 22:29:17 +02:00
|
|
|
|
|
|
|
static std::unique_ptr<impl::PopupBase> currPopup;
|
2023-11-21 14:38:01 +01:00
|
|
|
static Lang name("");
|
2023-04-09 23:24:48 +02:00
|
|
|
|
2024-01-06 17:38:55 +01:00
|
|
|
AT_FIRST_TIME {
|
|
|
|
EventImHexClosing::subscribe([] {
|
|
|
|
currPopup.reset();
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2023-05-10 22:29:17 +02:00
|
|
|
if (auto &popups = impl::PopupBase::getOpenPopups(); !popups.empty()) {
|
|
|
|
if (!ImGui::IsPopupOpen(ImGuiID(0), ImGuiPopupFlags_AnyPopupId)) {
|
2023-07-23 18:22:53 +02:00
|
|
|
if (popupDelay <= -1.0) {
|
2023-12-11 11:42:33 +01:00
|
|
|
popupDelay = 0.2;
|
2023-07-23 18:22:53 +02:00
|
|
|
} else {
|
2023-12-19 13:10:25 +01:00
|
|
|
popupDelay -= m_lastFrameTime;
|
2023-12-11 11:42:33 +01:00
|
|
|
if (popupDelay < 0 || popups.size() == 1) {
|
2023-07-23 18:22:53 +02:00
|
|
|
popupDelay = -2.0;
|
|
|
|
currPopup = std::move(popups.back());
|
2023-11-21 14:38:01 +01:00
|
|
|
name = Lang(currPopup->getUnlocalizedName());
|
2024-01-06 17:38:55 +01:00
|
|
|
displayFrameCount = 0;
|
2023-07-23 18:22:53 +02:00
|
|
|
|
|
|
|
ImGui::OpenPopup(name);
|
|
|
|
popups.pop_back();
|
|
|
|
}
|
|
|
|
}
|
2023-05-10 22:29:17 +02:00
|
|
|
}
|
|
|
|
}
|
2023-04-07 10:21:27 +02:00
|
|
|
|
2023-05-10 22:29:17 +02:00
|
|
|
if (currPopup != nullptr) {
|
2023-04-07 10:21:27 +02:00
|
|
|
bool open = true;
|
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
const auto &minSize = currPopup->getMinSize();
|
|
|
|
const auto &maxSize = currPopup->getMaxSize();
|
|
|
|
const bool hasConstraints = minSize.x != 0 && minSize.y != 0 && maxSize.x != 0 && maxSize.y != 0;
|
2023-04-09 15:28:48 +02:00
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
if (hasConstraints)
|
|
|
|
ImGui::SetNextWindowSizeConstraints(minSize, maxSize);
|
|
|
|
else
|
|
|
|
ImGui::SetNextWindowSize(ImVec2(0, 0), ImGuiCond_Appearing);
|
2023-04-08 00:58:53 +02:00
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
auto* closeButton = currPopup->hasCloseButton() ? &open : nullptr;
|
2023-04-09 15:28:48 +02:00
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
const auto flags = currPopup->getFlags() | (!hasConstraints ? (ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoResize) : ImGuiWindowFlags_None);
|
2023-04-08 00:58:53 +02:00
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
if (!positionSet) {
|
2023-11-23 09:33:47 +01:00
|
|
|
ImGui::SetNextWindowPos(ImHexApi::System::getMainWindowPosition() + (ImHexApi::System::getMainWindowSize() / 2.0F), ImGuiCond_Always, ImVec2(0.5F, 0.5F));
|
2023-11-23 09:19:51 +01:00
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
if (sizeSet)
|
|
|
|
positionSet = true;
|
|
|
|
}
|
2023-04-09 15:28:48 +02:00
|
|
|
|
2023-04-16 21:34:29 +02:00
|
|
|
const auto createPopup = [&](bool displaying) {
|
|
|
|
if (displaying) {
|
2024-01-06 17:38:55 +01:00
|
|
|
displayFrameCount += 1;
|
2023-04-08 00:58:53 +02:00
|
|
|
currPopup->drawContent();
|
2023-04-16 21:34:29 +02:00
|
|
|
|
|
|
|
if (ImGui::GetWindowSize().x > ImGui::GetStyle().FramePadding.x * 10)
|
|
|
|
sizeSet = true;
|
2023-04-08 00:58:53 +02:00
|
|
|
|
2023-10-05 08:55:54 +02:00
|
|
|
// Reset popup position if it's outside the main window when multi-viewport is not enabled
|
|
|
|
// If not done, the popup will be stuck outside the main window and cannot be accessed anymore
|
|
|
|
if ((ImGui::GetIO().ConfigFlags & ImGuiConfigFlags_ViewportsEnable) == ImGuiConfigFlags_None) {
|
|
|
|
const auto currWindowPos = ImGui::GetWindowPos();
|
|
|
|
const auto minWindowPos = ImHexApi::System::getMainWindowPosition() - ImGui::GetWindowSize();
|
|
|
|
const auto maxWindowPos = ImHexApi::System::getMainWindowPosition() + ImHexApi::System::getMainWindowSize();
|
|
|
|
if (currWindowPos.x > maxWindowPos.x || currWindowPos.y > maxWindowPos.y || currWindowPos.x < minWindowPos.x || currWindowPos.y < minWindowPos.y) {
|
|
|
|
positionSet = false;
|
|
|
|
GImGui->MovingWindow = nullptr;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-08 00:58:53 +02:00
|
|
|
ImGui::EndPopup();
|
|
|
|
}
|
2023-04-16 21:34:29 +02:00
|
|
|
};
|
|
|
|
|
2024-12-25 18:49:50 +01:00
|
|
|
std::string localizedName = name.get();
|
2023-04-16 21:34:29 +02:00
|
|
|
if (currPopup->isModal())
|
2024-12-25 18:49:50 +01:00
|
|
|
createPopup(ImGui::BeginPopupModal(localizedName.c_str(), closeButton, flags));
|
2023-04-16 21:34:29 +02:00
|
|
|
else
|
2024-12-25 18:49:50 +01:00
|
|
|
createPopup(ImGui::BeginPopup(localizedName.c_str(), flags));
|
2023-04-16 21:34:29 +02:00
|
|
|
|
2024-12-25 18:49:50 +01:00
|
|
|
if (!ImGui::IsPopupOpen(localizedName.c_str()) && displayFrameCount < 5) {
|
|
|
|
ImGui::OpenPopup(localizedName.c_str());
|
2024-01-06 17:38:55 +01:00
|
|
|
}
|
|
|
|
|
2024-05-09 08:49:31 -07:00
|
|
|
if (currPopup->shouldClose() || !open) {
|
2024-12-25 18:49:50 +01:00
|
|
|
log::debug("Closing popup '{}'", localizedName);
|
2023-04-16 21:34:29 +02:00
|
|
|
positionSet = sizeSet = false;
|
2023-04-07 10:21:27 +02:00
|
|
|
|
2023-05-10 22:29:17 +02:00
|
|
|
currPopup = nullptr;
|
2023-04-09 23:24:48 +02:00
|
|
|
}
|
2023-04-07 10:21:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-19 23:21:20 +01:00
|
|
|
// Draw Toasts
|
|
|
|
{
|
2023-12-26 00:22:47 +01:00
|
|
|
u32 index = 0;
|
|
|
|
for (const auto &toast : impl::ToastBase::getQueuedToasts() | std::views::take(4)) {
|
|
|
|
const auto toastHeight = 60_scaled;
|
2023-12-19 23:21:20 +01:00
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowRounding, 5_scaled);
|
2023-12-26 00:22:47 +01:00
|
|
|
ImGui::SetNextWindowSize(ImVec2(280_scaled, toastHeight));
|
|
|
|
ImGui::SetNextWindowPos((ImHexApi::System::getMainWindowPosition() + ImHexApi::System::getMainWindowSize()) - scaled({ 10, 10 }) - scaled({ 0, (10 + toastHeight) * index }), ImGuiCond_Always, ImVec2(1, 1));
|
|
|
|
if (ImGui::Begin(hex::format("##Toast_{}", index).c_str(), nullptr, ImGuiWindowFlags_NoCollapse | ImGuiWindowFlags_NoResize | ImGuiWindowFlags_NoScrollbar | ImGuiWindowFlags_NoScrollWithMouse | ImGuiWindowFlags_NoDocking | ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoFocusOnAppearing)) {
|
2023-12-19 23:21:20 +01:00
|
|
|
auto drawList = ImGui::GetWindowDrawList();
|
|
|
|
|
|
|
|
const auto min = ImGui::GetWindowPos();
|
|
|
|
const auto max = min + ImGui::GetWindowSize();
|
|
|
|
|
|
|
|
drawList->PushClipRect(min, min + scaled({ 5, 60 }));
|
2023-12-26 00:22:47 +01:00
|
|
|
drawList->AddRectFilled(min, max, toast->getColor(), 5_scaled);
|
2023-12-19 23:21:20 +01:00
|
|
|
drawList->PopClipRect();
|
|
|
|
|
|
|
|
ImGui::Indent();
|
2023-12-26 00:22:47 +01:00
|
|
|
toast->draw();
|
2023-12-19 23:21:20 +01:00
|
|
|
ImGui::Unindent();
|
2023-12-26 00:22:47 +01:00
|
|
|
|
|
|
|
if (ImGui::IsWindowHovered() || toast->getAppearTime() <= 0)
|
|
|
|
toast->setAppearTime(ImGui::GetTime());
|
2023-12-19 23:21:20 +01:00
|
|
|
}
|
|
|
|
ImGui::End();
|
|
|
|
ImGui::PopStyleVar();
|
|
|
|
|
2023-12-26 00:22:47 +01:00
|
|
|
index += 1;
|
2023-12-19 23:21:20 +01:00
|
|
|
}
|
|
|
|
|
2023-12-26 00:22:47 +01:00
|
|
|
std::erase_if(impl::ToastBase::getQueuedToasts(), [](const auto &toast){
|
|
|
|
return toast->getAppearTime() > 0 && (toast->getAppearTime() + impl::ToastBase::VisibilityTime) < ImGui::GetTime();
|
|
|
|
});
|
2023-12-19 23:21:20 +01:00
|
|
|
}
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Run all deferred calls
|
2022-08-17 16:15:36 +02:00
|
|
|
TaskManager::runDeferredCalls();
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
2021-06-07 18:14:40 +02:00
|
|
|
void Window::frame() {
|
2022-07-30 22:01:49 +02:00
|
|
|
auto &io = ImGui::GetIO();
|
2023-02-17 12:03:53 +01:00
|
|
|
|
2025-01-01 16:19:38 +01:00
|
|
|
ShortcutManager::resetLastActivatedMenu();
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Loop through all views and draw them
|
2023-03-21 15:33:43 +01:00
|
|
|
for (auto &[name, view] : ContentRegistry::Views::impl::getEntries()) {
|
2021-12-12 21:46:48 +01:00
|
|
|
ImGui::GetCurrentContext()->NextWindowData.ClearFlags();
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Draw always visible views
|
2023-11-21 13:47:50 +01:00
|
|
|
view->drawAlwaysVisibleContent();
|
2021-06-07 18:14:40 +02:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Skip views that shouldn't be processed currently
|
2021-06-07 18:14:40 +02:00
|
|
|
if (!view->shouldProcess())
|
|
|
|
continue;
|
|
|
|
|
2023-12-06 13:49:58 +01:00
|
|
|
const auto openViewCount = std::ranges::count_if(ContentRegistry::Views::impl::getEntries(), [](const auto &entry) {
|
2023-12-30 23:52:25 +01:00
|
|
|
const auto &[unlocalizedName, openView] = entry;
|
2023-12-06 13:49:58 +01:00
|
|
|
|
2023-12-30 23:52:25 +01:00
|
|
|
return openView->hasViewMenuItemEntry() && openView->shouldProcess();
|
2023-12-06 13:49:58 +01:00
|
|
|
});
|
|
|
|
|
|
|
|
ImGuiWindowClass windowClass = {};
|
|
|
|
|
|
|
|
windowClass.DockNodeFlagsOverrideSet |= ImGuiDockNodeFlags_NoCloseButton;
|
|
|
|
|
|
|
|
if (openViewCount <= 1 || LayoutManager::isLayoutLocked())
|
|
|
|
windowClass.DockNodeFlagsOverrideSet |= ImGuiDockNodeFlags_NoTabBar;
|
|
|
|
|
|
|
|
ImGui::SetNextWindowClass(&windowClass);
|
|
|
|
|
2024-04-27 20:19:45 +02:00
|
|
|
auto window = ImGui::FindWindowByName(view->getName().c_str());
|
|
|
|
if (window != nullptr && window->DockNode == nullptr)
|
|
|
|
ImGui::SetNextWindowBgAlpha(1.0F);
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Draw view
|
2023-11-21 13:47:50 +01:00
|
|
|
view->draw();
|
2023-12-13 23:03:39 +01:00
|
|
|
view->trackViewOpenState();
|
2021-12-12 00:41:44 +01:00
|
|
|
|
2023-11-17 14:46:21 +01:00
|
|
|
if (view->getWindowOpenState()) {
|
2022-01-11 20:29:06 +01:00
|
|
|
bool hasWindow = window != nullptr;
|
2022-02-01 22:09:44 +01:00
|
|
|
bool focused = false;
|
2021-12-23 15:11:38 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Get the currently focused view
|
2023-10-30 21:53:44 +01:00
|
|
|
if (hasWindow && (window->Flags & ImGuiWindowFlags_Popup) != ImGuiWindowFlags_Popup) {
|
|
|
|
auto windowName = View::toWindowName(name);
|
|
|
|
ImGui::Begin(windowName.c_str());
|
2021-12-31 11:01:22 +01:00
|
|
|
|
2023-10-30 21:53:44 +01:00
|
|
|
// Detect if the window is focused
|
2022-01-29 21:48:59 +01:00
|
|
|
focused = ImGui::IsWindowFocused(ImGuiFocusedFlags_ChildWindows | ImGuiFocusedFlags_NoPopupHierarchy);
|
2023-10-30 21:53:44 +01:00
|
|
|
|
|
|
|
// Dock the window if it's not already docked
|
|
|
|
if (view->didWindowJustOpen() && !ImGui::IsWindowDocked()) {
|
|
|
|
ImGui::DockBuilderDockWindow(windowName.c_str(), ImHexApi::System::getMainDockSpaceId());
|
2023-12-13 23:03:39 +01:00
|
|
|
EventViewOpened::post(view.get());
|
2023-10-30 21:53:44 +01:00
|
|
|
}
|
|
|
|
|
2024-09-15 06:18:50 -07:00
|
|
|
// Pass on currently pressed keys to the shortcut handler
|
|
|
|
for (const auto &key : m_pressedKeys) {
|
2024-12-15 21:44:43 +01:00
|
|
|
ShortcutManager::process(view.get(), io.ConfigMacOSXBehaviors ? io.KeySuper : io.KeyCtrl, io.KeyAlt, io.KeyShift, io.ConfigMacOSXBehaviors ? io.KeyCtrl : io.KeySuper, focused, key);
|
2024-09-15 06:18:50 -07:00
|
|
|
}
|
2021-12-23 15:11:38 +01:00
|
|
|
|
2024-09-15 06:18:50 -07:00
|
|
|
ImGui::End();
|
2021-12-23 15:11:38 +01:00
|
|
|
}
|
|
|
|
}
|
2021-06-07 18:14:40 +02:00
|
|
|
}
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Handle global shortcuts
|
2023-12-19 13:10:25 +01:00
|
|
|
for (const auto &key : m_pressedKeys) {
|
2024-12-15 21:44:43 +01:00
|
|
|
ShortcutManager::processGlobals(io.ConfigMacOSXBehaviors ? io.KeySuper : io.KeyCtrl, io.KeyAlt, io.KeyShift, io.ConfigMacOSXBehaviors ? io.KeyCtrl : io.KeySuper, key);
|
2022-07-30 22:01:49 +02:00
|
|
|
}
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_pressedKeys.clear();
|
2021-06-07 18:14:40 +02:00
|
|
|
}
|
|
|
|
|
2020-11-10 15:26:38 +01:00
|
|
|
void Window::frameEnd() {
|
2023-12-08 10:29:44 +01:00
|
|
|
EventFrameEnd::post();
|
2022-01-18 00:10:10 +01:00
|
|
|
|
2023-12-13 11:24:25 +01:00
|
|
|
TutorialManager::drawTutorial();
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Clean up all tasks that are done
|
2022-08-17 16:15:36 +02:00
|
|
|
TaskManager::collectGarbage();
|
|
|
|
|
2021-12-10 16:09:55 +01:00
|
|
|
this->endNativeWindowFrame();
|
2023-02-17 12:03:53 +01:00
|
|
|
|
2023-12-27 13:54:00 +01:00
|
|
|
// Finalize ImGui frame
|
2020-11-10 15:26:38 +01:00
|
|
|
ImGui::Render();
|
2020-11-17 13:58:50 +01:00
|
|
|
|
2024-02-27 18:55:09 +01:00
|
|
|
// Compare the previous frame buffer to the current one to determine if the window content has changed
|
2023-12-27 13:54:00 +01:00
|
|
|
// If not, there's no point in sending the draw data off to the GPU and swapping buffers
|
2024-02-27 18:55:09 +01:00
|
|
|
// NOTE: For anybody looking at this code and thinking "why not just hash the buffer and compare the hashes",
|
|
|
|
// the reason is that hashing the buffer is significantly slower than just comparing the buffers directly.
|
|
|
|
// The buffer might become quite large if there's a lot of vertices on the screen but it's still usually less than
|
|
|
|
// 10MB (out of which only the active portion needs to actually be compared) which is worth the ~60x speedup.
|
2023-12-27 13:54:00 +01:00
|
|
|
bool shouldRender = false;
|
|
|
|
{
|
2024-02-27 18:55:09 +01:00
|
|
|
static std::vector<u8> previousVtxData;
|
|
|
|
static size_t previousVtxDataSize = 0;
|
|
|
|
|
|
|
|
size_t offset = 0;
|
|
|
|
size_t vtxDataSize = 0;
|
2023-12-27 13:54:00 +01:00
|
|
|
|
|
|
|
for (const auto viewPort : ImGui::GetPlatformIO().Viewports) {
|
|
|
|
auto drawData = viewPort->DrawData;
|
|
|
|
for (int n = 0; n < drawData->CmdListsCount; n++) {
|
2024-02-27 18:55:09 +01:00
|
|
|
vtxDataSize += drawData->CmdLists[n]->VtxBuffer.size() * sizeof(ImDrawVert);
|
2023-12-27 13:54:00 +01:00
|
|
|
}
|
2024-02-27 18:55:09 +01:00
|
|
|
}
|
|
|
|
for (const auto viewPort : ImGui::GetPlatformIO().Viewports) {
|
|
|
|
auto drawData = viewPort->DrawData;
|
2023-12-27 13:54:00 +01:00
|
|
|
for (int n = 0; n < drawData->CmdListsCount; n++) {
|
2023-12-27 16:33:49 +01:00
|
|
|
const ImDrawList *cmdList = drawData->CmdLists[n];
|
2024-09-15 06:23:58 -07:00
|
|
|
std::string ownerName = cmdList->_OwnerName;
|
2024-02-27 18:55:09 +01:00
|
|
|
|
2024-09-15 06:23:58 -07:00
|
|
|
if (vtxDataSize == previousVtxDataSize && (!ownerName.contains("##Popup") || !ownerName.contains("##image"))) {
|
2024-02-27 18:55:09 +01:00
|
|
|
shouldRender = shouldRender || std::memcmp(previousVtxData.data() + offset, cmdList->VtxBuffer.Data, cmdList->VtxBuffer.size() * sizeof(ImDrawVert)) != 0;
|
|
|
|
} else {
|
|
|
|
shouldRender = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (previousVtxData.size() < offset + cmdList->VtxBuffer.size() * sizeof(ImDrawVert)) {
|
|
|
|
previousVtxData.resize(offset + cmdList->VtxBuffer.size() * sizeof(ImDrawVert));
|
|
|
|
}
|
|
|
|
|
|
|
|
std::memcpy(previousVtxData.data() + offset, cmdList->VtxBuffer.Data, cmdList->VtxBuffer.size() * sizeof(ImDrawVert));
|
|
|
|
offset += cmdList->VtxBuffer.size() * sizeof(ImDrawVert);
|
2023-12-27 13:54:00 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-02-27 18:55:09 +01:00
|
|
|
previousVtxDataSize = vtxDataSize;
|
2023-12-27 13:54:00 +01:00
|
|
|
}
|
|
|
|
|
2024-02-28 20:36:22 +01:00
|
|
|
GLFWwindow *backupContext = glfwGetCurrentContext();
|
2024-02-28 20:16:15 +01:00
|
|
|
ImGui::UpdatePlatformWindows();
|
2024-02-28 20:36:22 +01:00
|
|
|
ImGui::RenderPlatformWindowsDefault();
|
|
|
|
glfwMakeContextCurrent(backupContext);
|
2024-02-28 20:16:15 +01:00
|
|
|
|
2023-12-27 13:54:00 +01:00
|
|
|
if (shouldRender) {
|
2024-05-20 11:27:57 +02:00
|
|
|
auto* drawData = ImGui::GetDrawData();
|
|
|
|
|
|
|
|
// Avoid accidentally clearing the viewport when the application is minimized,
|
|
|
|
// otherwise the OS will display an empty frame during deminimization on macOS
|
|
|
|
if (drawData->DisplaySize.x != 0 && drawData->DisplaySize.y != 0) {
|
|
|
|
int displayWidth, displayHeight;
|
|
|
|
glfwGetFramebufferSize(m_window, &displayWidth, &displayHeight);
|
|
|
|
glViewport(0, 0, displayWidth, displayHeight);
|
|
|
|
glClearColor(0.00F, 0.00F, 0.00F, 0.00F);
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
|
|
|
|
|
|
|
|
glfwSwapBuffers(m_window);
|
|
|
|
}
|
2023-12-27 13:54:00 +01:00
|
|
|
}
|
|
|
|
|
2023-05-11 18:44:50 +02:00
|
|
|
// Process layout load requests
|
|
|
|
// NOTE: This needs to be done before a new frame is started, otherwise ImGui won't handle docking correctly
|
2023-12-12 00:16:21 +01:00
|
|
|
LayoutManager::process();
|
|
|
|
WorkspaceManager::process();
|
2024-12-28 21:37:45 +01:00
|
|
|
|
|
|
|
ImGui::GetIO().FontGlobalScale = 1.0F / ImHexApi::System::getBackingScaleFactor();
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
2021-03-01 08:56:49 +01:00
|
|
|
void Window::initGLFW() {
|
2024-01-28 22:14:59 +01:00
|
|
|
auto initialWindowProperties = ImHexApi::System::getInitialWindowProperties();
|
2022-01-24 20:53:17 +01:00
|
|
|
glfwSetErrorCallback([](int error, const char *desc) {
|
2024-05-17 20:22:55 +02:00
|
|
|
bool isWaylandError = error == GLFW_PLATFORM_ERROR;
|
|
|
|
#if defined(GLFW_FEATURE_UNAVAILABLE)
|
|
|
|
isWaylandError = isWaylandError || (error == GLFW_FEATURE_UNAVAILABLE);
|
|
|
|
#endif
|
|
|
|
isWaylandError = isWaylandError && std::string_view(desc).contains("Wayland");
|
|
|
|
|
|
|
|
if (isWaylandError) {
|
2023-07-12 14:33:09 +02:00
|
|
|
// Ignore error spam caused by Wayland not supporting moving or resizing
|
|
|
|
// windows or querying their position and size.
|
2024-05-17 20:22:55 +02:00
|
|
|
return;
|
2023-07-12 14:33:09 +02:00
|
|
|
}
|
|
|
|
|
2023-05-14 21:39:18 +02:00
|
|
|
try {
|
2023-07-12 14:33:09 +02:00
|
|
|
log::error("GLFW Error [0x{:05X}] : {}", error, desc);
|
2023-05-14 21:39:18 +02:00
|
|
|
} catch (const std::system_error &) {
|
|
|
|
// Catch and ignore system error that might be thrown when too many errors are being logged to a file
|
|
|
|
}
|
2020-11-11 14:41:44 +01:00
|
|
|
});
|
|
|
|
|
2022-01-13 14:34:27 +01:00
|
|
|
if (!glfwInit()) {
|
|
|
|
log::fatal("Failed to initialize GLFW!");
|
|
|
|
std::abort();
|
|
|
|
}
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2024-05-03 21:39:31 +02:00
|
|
|
configureGLFW();
|
2021-04-21 20:06:48 +02:00
|
|
|
glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
|
2021-11-28 11:57:52 +01:00
|
|
|
glfwWindowHint(GLFW_VISIBLE, GLFW_FALSE);
|
2023-10-04 12:00:32 +02:00
|
|
|
glfwWindowHint(GLFW_CLIENT_API, GLFW_OPENGL_API);
|
2020-11-30 00:03:12 +01:00
|
|
|
|
2024-01-28 22:14:59 +01:00
|
|
|
if (initialWindowProperties.has_value()) {
|
|
|
|
glfwWindowHint(GLFW_MAXIMIZED, initialWindowProperties->maximized);
|
2023-07-31 11:17:37 +02:00
|
|
|
}
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Create window
|
2023-12-19 13:10:25 +01:00
|
|
|
m_windowTitle = "ImHex";
|
|
|
|
m_window = glfwCreateWindow(1280_scaled, 720_scaled, m_windowTitle.c_str(), nullptr, nullptr);
|
2021-08-18 22:36:46 +02:00
|
|
|
|
2024-01-15 20:52:08 +01:00
|
|
|
ImHexApi::System::impl::setMainWindowHandle(m_window);
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowUserPointer(m_window, this);
|
2020-11-23 23:57:19 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
if (m_window == nullptr) {
|
2022-01-13 14:34:27 +01:00
|
|
|
log::fatal("Failed to create window!");
|
|
|
|
std::abort();
|
|
|
|
}
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2023-10-30 16:40:14 +01:00
|
|
|
// Force window to be fully opaque by default
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowOpacity(m_window, 1.0F);
|
2023-10-30 16:40:14 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwMakeContextCurrent(m_window);
|
2024-03-14 21:18:57 +01:00
|
|
|
|
|
|
|
// Disable VSync. Not like any graphics driver actually cares
|
|
|
|
glfwSwapInterval(0);
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Center window
|
2021-11-28 11:57:52 +01:00
|
|
|
GLFWmonitor *monitor = glfwGetPrimaryMonitor();
|
|
|
|
if (monitor != nullptr) {
|
|
|
|
const GLFWvidmode *mode = glfwGetVideoMode(monitor);
|
|
|
|
if (mode != nullptr) {
|
|
|
|
int monitorX, monitorY;
|
|
|
|
glfwGetMonitorPos(monitor, &monitorX, &monitorY);
|
|
|
|
|
|
|
|
int windowWidth, windowHeight;
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwGetWindowSize(m_window, &windowWidth, &windowHeight);
|
2021-11-28 11:57:52 +01:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowPos(m_window, monitorX + (mode->width - windowWidth) / 2, monitorY + (mode->height - windowHeight) / 2);
|
2021-11-28 11:57:52 +01:00
|
|
|
}
|
|
|
|
}
|
2021-11-28 01:17:48 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Set up initial window position
|
2021-08-18 22:36:46 +02:00
|
|
|
{
|
|
|
|
int x = 0, y = 0;
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwGetWindowPos(m_window, &x, &y);
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2024-01-28 22:14:59 +01:00
|
|
|
if (initialWindowProperties.has_value()) {
|
|
|
|
x = initialWindowProperties->x;
|
|
|
|
y = initialWindowProperties->y;
|
2023-07-31 11:17:37 +02:00
|
|
|
}
|
|
|
|
|
2022-02-01 18:09:40 +01:00
|
|
|
ImHexApi::System::impl::setMainWindowPosition(x, y);
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowPos(m_window, x, y);
|
2021-08-18 22:36:46 +02:00
|
|
|
}
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Set up initial window size
|
2021-08-18 22:36:46 +02:00
|
|
|
{
|
|
|
|
int width = 0, height = 0;
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwGetWindowSize(m_window, &width, &height);
|
2023-07-31 11:17:37 +02:00
|
|
|
|
2024-01-28 22:14:59 +01:00
|
|
|
if (initialWindowProperties.has_value()) {
|
|
|
|
width = initialWindowProperties->width;
|
|
|
|
height = initialWindowProperties->height;
|
2023-07-31 11:17:37 +02:00
|
|
|
}
|
|
|
|
|
2022-02-01 18:09:40 +01:00
|
|
|
ImHexApi::System::impl::setMainWindowSize(width, height);
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowSize(m_window, width, height);
|
2021-08-18 22:36:46 +02:00
|
|
|
}
|
2020-12-16 22:43:07 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Register window move callback
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowPosCallback(m_window, [](GLFWwindow *window, int x, int y) {
|
2022-02-01 18:09:40 +01:00
|
|
|
ImHexApi::System::impl::setMainWindowPosition(x, y);
|
2020-12-16 22:43:07 +01:00
|
|
|
|
2022-01-24 20:53:17 +01:00
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
2023-12-27 21:23:54 +01:00
|
|
|
win->m_unlockFrameRate = true;
|
2024-07-05 17:39:07 +02:00
|
|
|
win->fullFrame();
|
2021-08-18 22:36:46 +02:00
|
|
|
});
|
2020-12-16 22:43:07 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Register window resize callback
|
2024-05-30 21:49:01 +02:00
|
|
|
glfwSetWindowSizeCallback(m_window, [](GLFWwindow *window, [[maybe_unused]] int width, [[maybe_unused]] int height) {
|
2022-01-24 20:53:17 +01:00
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
2023-12-27 21:23:54 +01:00
|
|
|
win->m_unlockFrameRate = true;
|
2024-05-30 16:57:07 +02:00
|
|
|
|
|
|
|
#if !defined(OS_WINDOWS)
|
|
|
|
if (!glfwGetWindowAttrib(window, GLFW_ICONIFIED))
|
|
|
|
ImHexApi::System::impl::setMainWindowSize(width, height);
|
|
|
|
#endif
|
|
|
|
|
2024-05-20 11:27:57 +02:00
|
|
|
#if defined(OS_MACOS)
|
|
|
|
// Stop widgets registering hover effects while the window is being resized
|
|
|
|
if (macosIsWindowBeingResizedByUser(window)) {
|
|
|
|
ImGui::GetIO().MousePos = ImVec2();
|
|
|
|
}
|
2024-07-01 20:09:16 +02:00
|
|
|
#elif defined(OS_WEB)
|
|
|
|
win->fullFrame();
|
2024-05-19 14:14:57 +02:00
|
|
|
#endif
|
2023-02-16 08:53:23 +01:00
|
|
|
});
|
|
|
|
|
2023-12-27 21:23:54 +01:00
|
|
|
glfwSetCursorPosCallback(m_window, [](GLFWwindow *window, double, double) {
|
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
|
|
|
win->m_unlockFrameRate = true;
|
|
|
|
});
|
|
|
|
|
2025-01-04 15:35:19 +01:00
|
|
|
glfwSetMouseButtonCallback(m_window, [](GLFWwindow *window, int, int, int) {
|
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
|
|
|
win->m_unlockFrameRate = true;
|
|
|
|
});
|
|
|
|
|
2025-01-04 22:00:26 +01:00
|
|
|
glfwSetScrollCallback(m_window, [](GLFWwindow *window, double, double) {
|
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
|
|
|
win->m_unlockFrameRate = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
glfwSetWindowFocusCallback(m_window, [](GLFWwindow *window, int) {
|
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
|
|
|
win->m_unlockFrameRate = true;
|
|
|
|
});
|
|
|
|
|
2024-02-03 23:56:08 +01:00
|
|
|
glfwSetWindowFocusCallback(m_window, [](GLFWwindow *, int focused) {
|
|
|
|
EventWindowFocused::post(focused == GLFW_TRUE);
|
|
|
|
});
|
|
|
|
|
2024-12-25 01:34:11 +01:00
|
|
|
// Register key press callback
|
|
|
|
glfwSetInputMode(m_window, GLFW_LOCK_KEY_MODS, GLFW_TRUE);
|
|
|
|
glfwSetKeyCallback(m_window, [](GLFWwindow *window, int key, int scanCode, int action, int mods) {
|
|
|
|
std::ignore = mods;
|
|
|
|
|
2024-12-25 12:36:06 +01:00
|
|
|
#if !defined(OS_WEB)
|
|
|
|
// Handle A-Z keys using their ASCII value instead of the keycode
|
|
|
|
if (key >= GLFW_KEY_A && key <= GLFW_KEY_Z) {
|
|
|
|
std::string_view name = glfwGetKeyName(key, scanCode);
|
|
|
|
|
|
|
|
// If the key name is only one character long, use the ASCII value instead
|
|
|
|
// Otherwise the keyboard was set to a non-English layout and the key name
|
|
|
|
// is not the same as the ASCII value
|
|
|
|
if (!name.empty()) {
|
|
|
|
const std::uint8_t byte = name[0];
|
|
|
|
if (name.length() == 1 && byte <= 0x7F) {
|
|
|
|
key = std::toupper(byte);
|
|
|
|
}
|
2023-11-18 15:18:33 +01:00
|
|
|
}
|
|
|
|
}
|
2024-12-25 12:36:06 +01:00
|
|
|
#else
|
2024-12-25 13:03:55 +01:00
|
|
|
std::ignore = scanCode;
|
2024-12-25 12:36:06 +01:00
|
|
|
// Emscripten doesn't support glfwGetKeyName. Just pass the value through.
|
|
|
|
#endif
|
2023-11-18 15:18:33 +01:00
|
|
|
|
2024-12-25 01:34:11 +01:00
|
|
|
if (key == GLFW_KEY_UNKNOWN) return;
|
|
|
|
|
|
|
|
if (action == GLFW_PRESS || action == GLFW_REPEAT) {
|
|
|
|
if (key != GLFW_KEY_LEFT_CONTROL && key != GLFW_KEY_RIGHT_CONTROL &&
|
|
|
|
key != GLFW_KEY_LEFT_ALT && key != GLFW_KEY_RIGHT_ALT &&
|
|
|
|
key != GLFW_KEY_LEFT_SHIFT && key != GLFW_KEY_RIGHT_SHIFT &&
|
|
|
|
key != GLFW_KEY_LEFT_SUPER && key != GLFW_KEY_RIGHT_SUPER
|
|
|
|
) {
|
|
|
|
auto win = static_cast<Window *>(glfwGetWindowUserPointer(window));
|
|
|
|
win->m_unlockFrameRate = true;
|
|
|
|
|
|
|
|
if (!(mods & GLFW_MOD_NUM_LOCK)) {
|
|
|
|
if (key == GLFW_KEY_KP_0) key = GLFW_KEY_INSERT;
|
|
|
|
else if (key == GLFW_KEY_KP_1) key = GLFW_KEY_END;
|
|
|
|
else if (key == GLFW_KEY_KP_2) key = GLFW_KEY_DOWN;
|
|
|
|
else if (key == GLFW_KEY_KP_3) key = GLFW_KEY_PAGE_DOWN;
|
|
|
|
else if (key == GLFW_KEY_KP_4) key = GLFW_KEY_LEFT;
|
|
|
|
else if (key == GLFW_KEY_KP_6) key = GLFW_KEY_RIGHT;
|
|
|
|
else if (key == GLFW_KEY_KP_7) key = GLFW_KEY_HOME;
|
|
|
|
else if (key == GLFW_KEY_KP_8) key = GLFW_KEY_UP;
|
|
|
|
else if (key == GLFW_KEY_KP_9) key = GLFW_KEY_PAGE_UP;
|
2023-11-17 14:46:21 +01:00
|
|
|
}
|
2024-12-25 01:34:11 +01:00
|
|
|
|
|
|
|
win->m_pressedKeys.push_back(key);
|
2023-10-04 12:00:32 +02:00
|
|
|
}
|
2024-12-25 01:34:11 +01:00
|
|
|
}
|
|
|
|
});
|
2020-11-11 14:41:44 +01:00
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Register window close callback
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowCloseCallback(m_window, [](GLFWwindow *window) {
|
2023-12-08 10:29:44 +01:00
|
|
|
EventWindowClosing::post(window);
|
2023-02-17 12:03:53 +01:00
|
|
|
});
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowSizeLimits(m_window, 480_scaled, 360_scaled, GLFW_DONT_CARE, GLFW_DONT_CARE);
|
2025-01-04 15:35:19 +01:00
|
|
|
|
|
|
|
m_frameRateThread = std::jthread([this](const std::stop_token &stopToken) {
|
|
|
|
using Duration = std::chrono::duration<double, std::nano>;
|
|
|
|
Duration passedTime = {};
|
|
|
|
|
|
|
|
std::chrono::steady_clock::time_point startTime = {}, endTime = {};
|
|
|
|
Duration requestedFrameTime = {}, remainingUnlockedTime = {};
|
|
|
|
float targetFps = 0;
|
|
|
|
|
|
|
|
const auto nativeFps = [] -> float {
|
|
|
|
if (const auto monitor = glfwGetPrimaryMonitor(); monitor != nullptr) {
|
|
|
|
if (const auto videoMode = glfwGetVideoMode(monitor); videoMode != nullptr) {
|
|
|
|
return videoMode->refreshRate;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 60;
|
|
|
|
}();
|
|
|
|
|
|
|
|
while (!stopToken.stop_requested()) {
|
|
|
|
const auto iterationTime = endTime - startTime;
|
|
|
|
startTime = std::chrono::steady_clock::now();
|
|
|
|
|
|
|
|
targetFps = ImHexApi::System::getTargetFPS();
|
|
|
|
|
|
|
|
if (m_unlockFrameRate.exchange(false)) {
|
|
|
|
remainingUnlockedTime = std::chrono::seconds(2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the target frame rate is below 15, use the current monitor's refresh rate
|
|
|
|
if (targetFps < 15) {
|
|
|
|
targetFps = nativeFps;
|
|
|
|
}
|
|
|
|
|
|
|
|
passedTime += iterationTime;
|
|
|
|
if (remainingUnlockedTime > std::chrono::nanoseconds(0)) {
|
|
|
|
remainingUnlockedTime -= iterationTime;
|
|
|
|
} else {
|
|
|
|
targetFps = 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
requestedFrameTime = (Duration(1.0E9) / targetFps) / 1.3;
|
|
|
|
if (passedTime >= requestedFrameTime) {
|
|
|
|
std::scoped_lock lock(m_sleepMutex);
|
|
|
|
m_sleepFlag = true;
|
|
|
|
m_sleepCondVar.notify_all();
|
|
|
|
|
|
|
|
passedTime = {};
|
|
|
|
}
|
|
|
|
|
|
|
|
std::this_thread::sleep_for(std::chrono::microseconds(100));
|
|
|
|
|
|
|
|
endTime = std::chrono::steady_clock::now();
|
|
|
|
}
|
|
|
|
});
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
2023-10-04 12:00:32 +02:00
|
|
|
void Window::resize(i32 width, i32 height) {
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwSetWindowSize(m_window, width, height);
|
2023-10-04 12:00:32 +02:00
|
|
|
}
|
|
|
|
|
2020-11-10 15:26:38 +01:00
|
|
|
void Window::initImGui() {
|
|
|
|
IMGUI_CHECKVERSION();
|
2021-02-03 00:56:33 +01:00
|
|
|
|
2023-11-29 23:47:37 +01:00
|
|
|
auto fonts = ImHexApi::Fonts::getFontAtlas();
|
2022-02-01 18:09:40 +01:00
|
|
|
|
2023-12-20 16:31:31 +01:00
|
|
|
if (fonts == nullptr) {
|
|
|
|
fonts = IM_NEW(ImFontAtlas)();
|
|
|
|
|
|
|
|
fonts->AddFontDefault();
|
|
|
|
fonts->Build();
|
|
|
|
}
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Initialize ImGui and all other ImGui extensions
|
2024-12-17 23:51:45 +01:00
|
|
|
GImGui = ImGui::CreateContext(fonts);
|
|
|
|
GImPlot = ImPlot::CreateContext();
|
|
|
|
ImPlot3D::GImPlot3D = ImPlot3D::CreateContext();
|
|
|
|
GImNodes = ImNodes::CreateContext();
|
2021-02-03 00:56:33 +01:00
|
|
|
|
2022-02-01 22:09:44 +01:00
|
|
|
ImGuiIO &io = ImGui::GetIO();
|
2022-01-24 20:53:17 +01:00
|
|
|
ImGuiStyle &style = ImGui::GetStyle();
|
2020-11-23 15:51:40 +01:00
|
|
|
|
2022-12-29 19:26:00 +01:00
|
|
|
ImNodes::GetStyle().Flags = ImNodesStyleFlags_NodeOutline | ImNodesStyleFlags_GridLines;
|
|
|
|
|
2021-02-18 17:10:56 +01:00
|
|
|
io.ConfigFlags |= ImGuiConfigFlags_DockingEnable | ImGuiConfigFlags_NavEnableKeyboard;
|
2022-11-07 00:04:47 +01:00
|
|
|
io.ConfigWindowsMoveFromTitleBarOnly = true;
|
2023-02-16 16:29:41 +01:00
|
|
|
io.FontGlobalScale = 1.0F;
|
2022-03-22 08:20:14 +01:00
|
|
|
|
2022-06-03 10:35:05 +02:00
|
|
|
if (glfwGetPrimaryMonitor() != nullptr) {
|
2024-01-28 22:14:59 +01:00
|
|
|
if (ImHexApi::System::isMutliWindowModeEnabled())
|
2022-03-22 08:20:14 +01:00
|
|
|
io.ConfigFlags |= ImGuiConfigFlags_ViewportsEnable;
|
|
|
|
}
|
2021-02-18 17:10:56 +01:00
|
|
|
|
2022-01-11 16:02:05 +01:00
|
|
|
io.ConfigViewportsNoTaskBarIcon = false;
|
2020-11-23 15:51:40 +01:00
|
|
|
|
2021-08-17 13:39:46 +02:00
|
|
|
ImNodes::PushAttributeFlag(ImNodesAttributeFlags_EnableLinkDetachWithDragClick);
|
|
|
|
ImNodes::PushAttributeFlag(ImNodesAttributeFlags_EnableLinkCreationOnSnap);
|
|
|
|
|
2023-02-17 12:03:53 +01:00
|
|
|
// Allow ImNodes links to always be detached without holding down any button
|
2021-08-17 13:39:46 +02:00
|
|
|
{
|
2023-02-17 12:03:53 +01:00
|
|
|
static bool always = true;
|
2021-08-17 13:39:46 +02:00
|
|
|
ImNodes::GetIO().LinkDetachWithModifierClick.Modifier = &always;
|
|
|
|
}
|
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
io.UserData = &m_imguiCustomData;
|
2021-02-25 00:17:41 +01:00
|
|
|
|
2022-07-29 17:37:30 +02:00
|
|
|
auto scale = ImHexApi::System::getGlobalScale();
|
|
|
|
style.ScaleAllSizes(scale);
|
|
|
|
io.DisplayFramebufferScale = ImVec2(scale, scale);
|
2023-02-17 12:03:00 +01:00
|
|
|
io.Fonts->SetTexID(fonts->TexID);
|
2020-12-11 14:24:42 +01:00
|
|
|
|
2020-11-23 15:51:40 +01:00
|
|
|
style.WindowMenuButtonPosition = ImGuiDir_None;
|
2022-02-01 22:09:44 +01:00
|
|
|
style.IndentSpacing = 10.0F;
|
2023-03-17 19:58:08 +01:00
|
|
|
style.DisplaySafeAreaPadding = ImVec2(0.0F, 0.0F);
|
2020-11-10 15:26:38 +01:00
|
|
|
|
2024-07-01 23:53:14 +02:00
|
|
|
style.Colors[ImGuiCol_TabSelectedOverline] = ImVec4(0.0F, 0.0F, 0.0F, 0.0F);
|
|
|
|
style.Colors[ImGuiCol_TabDimmedSelectedOverline] = ImVec4(0.0F, 0.0F, 0.0F, 0.0F);
|
|
|
|
|
2020-11-15 15:49:21 +01:00
|
|
|
// Install custom settings handler
|
2023-02-17 12:03:53 +01:00
|
|
|
{
|
|
|
|
ImGuiSettingsHandler handler;
|
|
|
|
handler.TypeName = "ImHex";
|
|
|
|
handler.TypeHash = ImHashStr("ImHex");
|
|
|
|
|
|
|
|
handler.ReadOpenFn = [](ImGuiContext *ctx, ImGuiSettingsHandler *, const char *) -> void* { return ctx; };
|
|
|
|
|
2024-02-10 23:31:05 +01:00
|
|
|
handler.ReadLineFn = [](ImGuiContext *, ImGuiSettingsHandler *, void *, const char *line) {
|
|
|
|
LayoutManager::onLoad(line);
|
2023-02-17 12:03:53 +01:00
|
|
|
};
|
|
|
|
|
2024-02-10 23:31:05 +01:00
|
|
|
handler.WriteAllFn = [](ImGuiContext *, ImGuiSettingsHandler *handler, ImGuiTextBuffer *buffer) {
|
|
|
|
buffer->appendf("[%s][General]\n", handler->TypeName);
|
|
|
|
LayoutManager::onStore(buffer);
|
|
|
|
buffer->append("\n");
|
2023-02-17 12:03:53 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
handler.UserData = this;
|
2023-12-13 11:24:25 +01:00
|
|
|
|
|
|
|
auto context = ImGui::GetCurrentContext();
|
|
|
|
context->SettingsHandlers.push_back(handler);
|
|
|
|
context->TestEngineHookItems = true;
|
2023-02-17 12:03:53 +01:00
|
|
|
|
2023-07-07 09:20:33 +02:00
|
|
|
io.IniFilename = nullptr;
|
2023-02-17 12:03:53 +01:00
|
|
|
}
|
2021-03-01 08:56:49 +01:00
|
|
|
|
2023-10-04 12:00:32 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
ImGui_ImplGlfw_InitForOpenGL(m_window, true);
|
2021-04-21 20:06:48 +02:00
|
|
|
|
2022-08-02 13:12:12 +02:00
|
|
|
#if defined(OS_MACOS)
|
|
|
|
ImGui_ImplOpenGL3_Init("#version 150");
|
2023-10-04 12:00:32 +02:00
|
|
|
#elif defined(OS_WEB)
|
|
|
|
ImGui_ImplOpenGL3_Init();
|
2024-12-27 00:02:37 +01:00
|
|
|
ImGui_ImplGlfw_InstallEmscriptenCallbacks(m_window, "#canvas");
|
2022-08-02 13:12:12 +02:00
|
|
|
#else
|
|
|
|
ImGui_ImplOpenGL3_Init("#version 130");
|
|
|
|
#endif
|
2021-08-21 00:52:11 +02:00
|
|
|
|
|
|
|
for (const auto &plugin : PluginManager::getPlugins())
|
|
|
|
plugin.setImGuiContext(ImGui::GetCurrentContext());
|
2023-02-16 18:06:40 +01:00
|
|
|
|
2023-12-08 10:29:44 +01:00
|
|
|
RequestInitThemeHandlers::post();
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
2022-01-18 00:10:10 +01:00
|
|
|
void Window::exitGLFW() {
|
2023-12-19 13:10:25 +01:00
|
|
|
glfwDestroyWindow(m_window);
|
2020-11-10 15:26:38 +01:00
|
|
|
glfwTerminate();
|
2023-05-15 11:30:24 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
m_window = nullptr;
|
2020-11-10 15:26:38 +01:00
|
|
|
}
|
|
|
|
|
2022-01-18 00:10:10 +01:00
|
|
|
void Window::exitImGui() {
|
2020-11-10 15:26:38 +01:00
|
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
|
|
ImGui_ImplGlfw_Shutdown();
|
2024-03-21 21:28:06 +01:00
|
|
|
|
2024-12-17 23:51:45 +01:00
|
|
|
ImPlot3D::DestroyContext();
|
2021-03-02 13:48:23 +01:00
|
|
|
ImPlot::DestroyContext();
|
2020-11-10 15:26:38 +01:00
|
|
|
ImGui::DestroyContext();
|
|
|
|
}
|
|
|
|
|
2021-06-18 11:09:36 -07:00
|
|
|
}
|