#include "content/views/view_about.hpp" #include "hex/ui/popup.hpp" #include #include #include #include #include #include #include #include #include #include #include #include #include namespace hex::plugin::builtin { class PopupEE : public Popup { public: PopupEE() : Popup("Se" /* Not going to */ "cr" /* make it that easy */ "et") { } void drawContent() override { ImGuiIO& io = ImGui::GetIO(); ImVec2 size = scaled({ 320, 180 }); ImGui::InvisibleButton("canvas", size); ImVec2 p0 = ImGui::GetItemRectMin(); ImVec2 p1 = ImGui::GetItemRectMax(); ImDrawList* drawList = ImGui::GetWindowDrawList(); drawList->PushClipRect(p0, p1); ImVec4 mouseData; mouseData.x = (io.MousePos.x - p0.x) / size.x; mouseData.y = (io.MousePos.y - p0.y) / size.y; mouseData.z = io.MouseDownDuration[0]; mouseData.w = io.MouseDownDuration[1]; fx(drawList, p0, p1, size, mouseData, float(ImGui::GetTime())); } void fx(ImDrawList* drawList, ImVec2 startPos, ImVec2 endPos, ImVec2, ImVec4, float t) { const float CircleRadius = 5_scaled; const float Gap = 1_scaled; constexpr static auto func = [](i32 x, i32 y, float t) { return std::sin(t - std::sqrt(std::pow((x - 14), 2) + std::pow((y - 8), 2))); }; float x = startPos.x + CircleRadius + Gap; u32 ix = 0; while (x < endPos.x) { float y = startPos.y + CircleRadius + Gap; u32 iy = 0; while (y < endPos.y) { const float result = func(ix, iy, t); const float radius = CircleRadius * std::abs(result); const auto color = result < 0 ? ImColor(0xFF, 0, 0, 0xFF) : ImColor(0xFF, 0xFF, 0xFF, 0xFF); drawList->AddCircleFilled(ImVec2(x, y), radius, color); y += CircleRadius * 2 + Gap; iy += 1; } x += CircleRadius * 2 + Gap; ix += 1; } } }; ViewAbout::ViewAbout() : View::Modal("hex.builtin.view.help.about.name") { // Add "About" menu item to the help menu ContentRegistry::Interface::addMenuItem({ "hex.builtin.menu.help", "hex.builtin.view.help.about.name" }, ICON_VS_INFO, 1000, Shortcut::None, [this] { this->getWindowOpenState() = true; }); ContentRegistry::Interface::addMenuItemSeparator({ "hex.builtin.menu.help" }, 2000); ContentRegistry::Interface::addMenuItemSubMenu({ "hex.builtin.menu.help" }, 3000, [] { static std::string content; if (ImGui::InputTextWithHint("##search", "hex.builtin.view.help.documentation_search"_lang, content, ImGuiInputTextFlags_AutoSelectAll | ImGuiInputTextFlags_EscapeClearsAll | ImGuiInputTextFlags_EnterReturnsTrue)) { PopupDocsQuestion::open(content); content.clear(); ImGui::CloseCurrentPopup(); } }); ContentRegistry::Interface::addMenuItemSeparator({ "hex.builtin.menu.help" }, 4000); // Add documentation link to the help menu ContentRegistry::Interface::addMenuItem({ "hex.builtin.menu.help", "hex.builtin.view.help.documentation" }, ICON_VS_BOOK, 5000, Shortcut::None, [] { hex::openWebpage("https://docs.werwolv.net/imhex"); AchievementManager::unlockAchievement("hex.builtin.achievement.starting_out", "hex.builtin.achievement.starting_out.docs.name"); }); } void ViewAbout::drawAboutMainPage() { // Draw main about table if (ImGui::BeginTable("about_table", 2, ImGuiTableFlags_SizingFixedFit)) { ImGui::TableNextRow(); ImGui::TableNextColumn(); // Draw the ImHex icon if (!m_logoTexture.isValid()) m_logoTexture = ImGuiExt::Texture::fromImage(romfs::get("assets/common/logo.png").span(), ImGuiExt::Texture::Filter::Linear); ImGui::Image(m_logoTexture, scaled({ 100, 100 })); if (ImGui::IsItemClicked()) { m_clickCount += 1; } if (m_clickCount >= (2 * 3 + 4)) { this->getWindowOpenState() = false; PopupEE::open(); m_clickCount = 0; } ImGui::TableNextColumn(); ImGuiExt::BeginSubWindow("Build Information", ImVec2(450_scaled, 0), ImGuiChildFlags_AutoResizeX | ImGuiChildFlags_AutoResizeY); { this->drawBuildInformation(); } ImGuiExt::EndSubWindow(); ImGui::EndTable(); } // Draw donation links ImGuiExt::Header("hex.builtin.view.help.about.donations"_lang); if (ImGui::BeginChild("##ThanksWrapper", ImVec2(ImGui::GetContentRegionAvail().x, ImGui::GetTextLineHeightWithSpacing() * 3))) { ImGui::PushTextWrapPos(ImGui::GetContentRegionAvail().x * 0.8F); ImGuiExt::TextFormattedCentered("{}", static_cast("hex.builtin.view.help.about.thanks"_lang)); ImGui::PopTextWrapPos(); } ImGui::EndChild(); ImGui::NewLine(); struct DonationPage { ImGuiExt::Texture texture; const char *link; }; static std::array DonationPages = { DonationPage { ImGuiExt::Texture::fromImage(romfs::get("assets/common/donation/paypal.png").span(), ImGuiExt::Texture::Filter::Linear), "https://werwolv.net/donate" }, DonationPage { ImGuiExt::Texture::fromImage(romfs::get("assets/common/donation/github.png").span(), ImGuiExt::Texture::Filter::Linear), "https://github.com/sponsors/WerWolv" }, DonationPage { ImGuiExt::Texture::fromImage(romfs::get("assets/common/donation/patreon.png").span(), ImGuiExt::Texture::Filter::Linear), "https://patreon.com/werwolv" }, }; if (ImGui::BeginTable("DonationLinks", 5, ImGuiTableFlags_SizingStretchSame)) { ImGui::TableNextRow(); ImGui::TableNextColumn(); for (const auto &page : DonationPages) { ImGui::TableNextColumn(); const auto size = page.texture.getSize() / 1.5F; const auto startPos = ImGui::GetCursorScreenPos(); ImGui::Image(page.texture, page.texture.getSize() / 1.5F); if (ImGui::IsItemHovered()) { ImGui::GetForegroundDrawList()->AddShadowCircle(startPos + size / 2, size.x / 2, ImGui::GetColorU32(ImGuiCol_Button), 100.0F, ImVec2(), ImDrawFlags_ShadowCutOutShapeBackground); } if (ImGui::IsItemClicked()) { hex::openWebpage(page.link); } } ImGui::EndTable(); } ImGui::NewLine(); } void ViewAbout::drawBuildInformation() { if (ImGui::BeginTable("Information", 1, ImGuiTableFlags_RowBg | ImGuiTableFlags_BordersInner)) { ImGui::Indent(); ImGui::TableNextRow(); ImGui::TableNextColumn(); { // Draw basic information about ImHex and its version ImGuiExt::TextFormatted("ImHex Hex Editor v{} by WerWolv", ImHexApi::System::getImHexVersion()); ImGui::Indent(25_scaled); ImGuiExt::TextFormatted("Powered by Dear ImGui v{}", ImGui::GetVersion()); ImGui::Unindent(25_scaled); } ImGui::TableNextColumn(); { ImGuiExt::TextFormatted(" {} ", ICON_VS_SOURCE_CONTROL); ImGui::SameLine(0, 0); // Draw a clickable link to the current commit if (ImGuiExt::Hyperlink(hex::format("{0}@{1}", ImHexApi::System::getCommitBranch(), ImHexApi::System::getCommitHash()).c_str())) hex::openWebpage("https://github.com/WerWolv/ImHex/commit/" + ImHexApi::System::getCommitHash(true)); } ImGui::TableNextColumn(); { // Draw the build date and time ImGuiExt::TextFormatted("Compiled on {} at {}", __DATE__, __TIME__); } ImGui::TableNextColumn(); { // Draw the author of the current translation ImGui::TextUnformatted("hex.builtin.view.help.about.translator"_lang); } ImGui::TableNextColumn(); { // Draw information about the open-source nature of ImHex ImGui::TextUnformatted("hex.builtin.view.help.about.source"_lang); ImGui::SameLine(); // Draw a clickable link to the GitHub repository if (ImGuiExt::Hyperlink(ICON_VS_LOGO_GITHUB " " "WerWolv/ImHex")) hex::openWebpage("https://github.com/WerWolv/ImHex"); } ImGui::Unindent(); ImGui::EndTable(); } } static void drawContributorTable(const char *title, const auto &contributors) { ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2()); ImGuiExt::BeginSubWindow(title, ImVec2(ImGui::GetContentRegionAvail().x, 0), ImGuiChildFlags_AutoResizeX); ImGui::PopStyleVar(); { if (ImGui::BeginTable(title, 1, ImGuiTableFlags_RowBg | ImGuiTableFlags_Borders)) { for (const auto &contributor : contributors) { ImGui::TableNextRow(); if (contributor.mainContributor) { ImGui::TableSetBgColor(ImGuiTableBgTarget_RowBg0, ImGui::GetColorU32(ImGuiCol_PlotHistogram) & 0x1FFFFFFF); } ImGui::TableNextColumn(); if (ImGuiExt::Hyperlink(contributor.name)) hex::openWebpage(contributor.link); if (contributor.description[0] != '\0') { ImGui::Indent(); ImGui::TextUnformatted(contributor.description); ImGui::Unindent(); } } ImGui::EndTable(); } } ImGuiExt::EndSubWindow(); } void ViewAbout::drawContributorPage() { struct Contributor { const char *name; const char *description; const char *link; bool mainContributor; }; constexpr static std::array Contributors = { Contributor { "iTrooz", "A huge amount of help maintaining ImHex and the CI", "https://github.com/iTrooz", true }, Contributor { "jumanji144", "A ton of help with the Pattern Language, API and usage stats", "https://github.com/jumanji144", true }, Contributor { "AxCut", "A ton of great pattern language improvements and help with the issue tracker", "https://github.com/paxcut", false }, Contributor { "Mary", "Porting ImHex to macOS originally", "https://github.com/marysaka", false }, Contributor { "Roblabla", "Adding the MSI Windows installer", "https://github.com/roblabla", false }, Contributor { "jam1garner", "Adding support for Rust plugins", "https://github.com/jam1garner", false }, Contributor { "All other amazing contributors", "Being part of the community, opening issues, PRs and donating", "https://github.com/WerWolv/ImHex/graphs/contributors", false } }; constexpr static std::array Testers = { Contributor { "Nemoumbra", "Breaking my code literal seconds after I push it", "https://github.com/Nemoumbra", true }, Contributor { "Berylskid", "", "https://github.com/Berylskid", false }, Contributor { "Jan Polak", "", "https://github.com/polak-jan", false }, Contributor { "Ken-Kaneki", "", "https://github.com/loneicewolf", false }, Contributor { "Everybody who has reported issues", "Helping me find bugs and improve the software", "https://github.com/WerWolv/ImHex/issues", false } }; ImGuiExt::TextFormattedWrapped("These amazing people have contributed some incredible things to ImHex in the past.\nConsider opening a PR on the Git Repository to take your place among them!"); ImGui::NewLine(); drawContributorTable("Contributors", Contributors); ImGui::NewLine(); ImGuiExt::TextFormattedWrapped("All of these great people made ImHex work much much smoother.\nConsider joining our Tester team to help making ImHex better for everyone!"); ImGui::NewLine(); drawContributorTable("Testers", Testers); } void ViewAbout::drawLibraryCreditsPage() { struct Library { const char *name; const char *author; const char *link; }; constexpr static std::array ImGuiLibraries = { Library { "ImGui", "ocornut", "https://github.com/ocornut/imgui" }, Library { "ImPlot", "epezent", "https://github.com/epezent/implot" }, Library { "imnodes", "Nelarius", "https://github.com/Nelarius/imnodes" }, Library { "ImGuiColorTextEdit", "BalazsJako", "https://github.com/BalazsJako/ImGuiColorTextEdit" }, }; constexpr static std::array ExternalLibraries = { Library { "PatternLanguage", "WerWolv", "https://github.com/WerWolv/PatternLanguage" }, Library { "libwolv", "WerWolv", "https://github.com/WerWolv/libwolv" }, Library { "libromfs", "WerWolv", "https://github.com/WerWolv/libromfs" }, }; constexpr static std::array ThirdPartyLibraries = { Library { "json", "nlohmann", "https://github.com/nlohmann/json" }, Library { "fmt", "fmtlib", "https://github.com/fmtlib/fmt" }, Library { "nativefiledialog-extended", "btzy", "https://github.com/btzy/nativefiledialog-extended" }, Library { "xdgpp", "danyspin97", "https://sr.ht/~danyspin97/xdgpp" }, Library { "capstone", "aquynh", "https://github.com/aquynh/capstone" }, Library { "microtar", "rxi", "https://github.com/rxi/microtar" }, Library { "yara", "VirusTotal", "https://github.com/VirusTotal/yara" }, Library { "edlib", "Martinsos", "https://github.com/Martinsos/edlib" }, Library { "HashLibPlus", "ron4fun", "https://github.com/ron4fun/HashLibPlus" }, Library { "miniaudio", "mackron", "https://github.com/mackron/miniaudio" }, Library { "freetype", "freetype", "https://gitlab.freedesktop.org/freetype/freetype" }, Library { "mbedTLS", "ARMmbed", "https://github.com/ARMmbed/mbedtls" }, Library { "curl", "curl", "https://github.com/curl/curl" }, Library { "file", "file", "https://github.com/file/file" }, Library { "glfw", "glfw", "https://github.com/glfw/glfw" }, Library { "llvm", "llvm-project", "https://github.com/llvm/llvm-project" }, }; constexpr static auto drawTable = [](const char *category, const auto &libraries) { const auto width = ImGui::GetContentRegionAvail().x; ImGuiExt::BeginSubWindow(category); { for (const auto &library : libraries) { ImGui::PushStyleColor(ImGuiCol_ChildBg, ImGui::GetColorU32(ImGuiCol_TableHeaderBg)); ImGui::PushStyleVar(ImGuiStyleVar_ChildRounding, 50); ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, scaled({ 12, 3 })); if (ImGui::BeginChild(library.link, ImVec2(), ImGuiChildFlags_Border | ImGuiChildFlags_AutoResizeX | ImGuiChildFlags_AutoResizeY)) { if (ImGuiExt::Hyperlink(hex::format("{}/{}", library.author, library.name).c_str())) { hex::openWebpage(library.link); } ImGui::SetItemTooltip("%s", library.link); } ImGui::EndChild(); ImGui::SameLine(); if (ImGui::GetCursorPosX() > (width - 100_scaled)) ImGui::NewLine(); ImGui::PopStyleColor(); ImGui::PopStyleVar(2); } } ImGuiExt::EndSubWindow(); ImGui::NewLine(); }; ImGuiExt::TextFormattedWrapped("ImHex builds on top of the amazing work of a ton of talented library developers without which this project wouldn't stand."); ImGui::NewLine(); drawTable("ImGui", ImGuiLibraries); drawTable("External", ExternalLibraries); drawTable("Third Party", ThirdPartyLibraries); } void ViewAbout::drawLoadedPlugins() { const auto &plugins = PluginManager::getPlugins(); ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2()); ImGuiExt::BeginSubWindow("hex.builtin.view.help.about.plugins"_lang); ImGui::PopStyleVar(); { if (ImGui::BeginTable("plugins", 4, ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingFixedFit)) { ImGui::TableSetupScrollFreeze(0, 1); ImGui::TableSetupColumn("hex.builtin.view.help.about.plugins.plugin"_lang); ImGui::TableSetupColumn("hex.builtin.view.help.about.plugins.author"_lang); ImGui::TableSetupColumn("hex.builtin.view.help.about.plugins.desc"_lang, ImGuiTableColumnFlags_WidthStretch, 0.5); ImGui::TableSetupColumn("##loaded", ImGuiTableColumnFlags_WidthFixed, ImGui::GetTextLineHeight()); ImGui::TableHeadersRow(); for (const auto &plugin : plugins) { this->drawPluginRow(plugin); } ImGui::EndTable(); } } ImGuiExt::EndSubWindow(); } void ViewAbout::drawPluginRow(const hex::Plugin& plugin) { if (plugin.isLibraryPlugin()) return; auto features = plugin.getFeatures(); ImGui::TableNextRow(); ImGui::TableNextColumn(); bool open = false; ImGui::PushStyleColor(ImGuiCol_Text, ImGui::GetColorU32(ImGuiCol_Text)); if (features.empty()) ImGui::BulletText("%s", plugin.getPluginName().c_str()); else open = ImGui::TreeNode(plugin.getPluginName().c_str()); ImGui::PopStyleColor(); ImGui::TableNextColumn(); ImGui::TextUnformatted(plugin.getPluginAuthor().c_str()); ImGui::TableNextColumn(); ImGui::TextUnformatted(plugin.getPluginDescription().c_str()); ImGui::TableNextColumn(); ImGui::TextUnformatted(plugin.isLoaded() ? ICON_VS_CHECK : ICON_VS_CLOSE); if (open) { for (const auto &feature : plugin.getFeatures()) { ImGui::TableNextRow(); ImGui::TableNextColumn(); ImGuiExt::TextFormatted(" {}", feature.name.c_str()); ImGui::TableNextColumn(); ImGui::TableNextColumn(); ImGui::TableNextColumn(); ImGui::TextUnformatted(feature.enabled ? ICON_VS_CHECK : ICON_VS_CLOSE); } ImGui::TreePop(); } } void ViewAbout::drawPathsPage() { constexpr static std::array, size_t(fs::ImHexPath::END)> PathTypes = { { { "Patterns", fs::ImHexPath::Patterns }, { "Patterns Includes", fs::ImHexPath::PatternsInclude }, { "Magic", fs::ImHexPath::Magic }, { "Plugins", fs::ImHexPath::Plugins }, { "Yara Patterns", fs::ImHexPath::Yara }, { "Yara Advaned Analysis", fs::ImHexPath::YaraAdvancedAnalysis }, { "Config", fs::ImHexPath::Config }, { "Backups", fs::ImHexPath::Backups }, { "Resources", fs::ImHexPath::Resources }, { "Constants lists", fs::ImHexPath::Constants }, { "Custom encodings", fs::ImHexPath::Encodings }, { "Logs", fs::ImHexPath::Logs }, { "Recent files", fs::ImHexPath::Recent }, { "Scripts", fs::ImHexPath::Scripts }, { "Data inspector scripts", fs::ImHexPath::Inspectors }, { "Themes", fs::ImHexPath::Themes }, { "Native Libraries", fs::ImHexPath::Libraries }, { "Custom data processor nodes", fs::ImHexPath::Nodes }, { "Layouts", fs::ImHexPath::Layouts }, { "Workspaces", fs::ImHexPath::Workspaces }, } }; static_assert(PathTypes.back().first != nullptr, "All path items need to be populated!"); ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2()); ImGuiExt::BeginSubWindow("Paths", ImGui::GetContentRegionAvail()); { if (ImGui::BeginTable("##imhex_paths", 2, ImGuiTableFlags_ScrollY | ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingFixedFit)) { ImGui::TableSetupScrollFreeze(0, 1); ImGui::TableSetupColumn("Type"); ImGui::TableSetupColumn("Paths"); // Draw the table ImGui::TableHeadersRow(); for (const auto &[name, type] : PathTypes) { ImGui::TableNextRow(); ImGui::TableNextColumn(); ImGui::TextUnformatted(name); ImGui::TableNextColumn(); for (auto &path : fs::getDefaultPaths(type, true)){ // Draw hyperlink to paths that exist or red text if they don't if (wolv::io::fs::isDirectory(path)){ if (ImGuiExt::Hyperlink(wolv::util::toUTF8String(path).c_str())) { fs::openFolderExternal(path); } } else { ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_ToolbarRed), wolv::util::toUTF8String(path)); } } } ImGui::EndTable(); } } ImGuiExt::EndSubWindow(); ImGui::PopStyleVar(); } static void drawRegularLine(const std::string& line) { ImGui::Bullet(); ImGui::SameLine(); // Check if the line contains bold text auto boldStart = line.find("**"); if (boldStart == std::string::npos) { // Draw the line normally ImGui::TextUnformatted(line.c_str()); return; } // Find the end of the bold text auto boldEnd = line.find("**", boldStart + 2); // Draw the line with the bold text highlighted ImGui::TextUnformatted(line.substr(0, boldStart).c_str()); ImGui::SameLine(0, 0); ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_Highlight), "{}", line.substr(boldStart + 2, boldEnd - boldStart - 2).c_str()); ImGui::SameLine(0, 0); ImGui::TextUnformatted(line.substr(boldEnd + 2).c_str()); } struct ReleaseNotes { std::string title; std::vector notes; }; static ReleaseNotes parseReleaseNotes(const HttpRequest::Result& response) { ReleaseNotes notes; nlohmann::json json; if (!response.isSuccess()) { // An error occurred, display it notes.notes.push_back("## HTTP Error: " + std::to_string(response.getStatusCode())); return notes; } // A valid response was received, parse it try { json = nlohmann::json::parse(response.getData()); // Get the release title notes.title = json["name"].get(); // Get the release notes and split it into lines auto body = json["body"].get(); notes.notes = wolv::util::splitString(body, "\r\n"); } catch (std::exception &e) { notes.notes.push_back("## Error: " + std::string(e.what())); } return notes; } void ViewAbout::drawReleaseNotesPage() { static ReleaseNotes notes; // Set up the request to get the release notes the first time the page is opened AT_FIRST_TIME { static HttpRequest request("GET", GitHubApiURL + std::string("/releases/tags/v") + ImHexApi::System::getImHexVersion(false)); m_releaseNoteRequest = request.execute(); }; // Wait for the request to finish and parse the response if (m_releaseNoteRequest.valid()) { if (m_releaseNoteRequest.wait_for(std::chrono::seconds(0)) == std::future_status::ready) { notes = parseReleaseNotes(m_releaseNoteRequest.get()); } else { // Draw a spinner while the release notes are loading ImGuiExt::TextSpinner("hex.ui.common.loading"_lang); } } // Draw the release title if (!notes.title.empty()) { auto title = hex::format("v{}: {}", ImHexApi::System::getImHexVersion(false), notes.title); ImGuiExt::Header(title.c_str(), true); ImGui::Separator(); } // Draw the release notes and format them using parts of the GitHub Markdown syntax // This is not a full implementation of the syntax, but it's enough to make the release notes look good. for (const auto &line : notes.notes) { if (line.starts_with("## ")) { // Draw H2 Header ImGuiExt::Header(line.substr(3).c_str()); } else if (line.starts_with("### ")) { // Draw H3 Header ImGuiExt::Header(line.substr(4).c_str()); } else if (line.starts_with("- ")) { // Draw bullet point drawRegularLine(line.substr(2)); } else if (line.starts_with(" - ")) { // Draw further indented bullet point ImGui::Indent(); ImGui::Indent(); drawRegularLine(line.substr(6)); ImGui::Unindent(); ImGui::Unindent(); } } } struct Commit { std::string hash; std::string message; std::string description; std::string author; std::string date; std::string url; }; static std::vector parseCommits(const HttpRequest::Result& response) { nlohmann::json json; std::vector commits; if (!response.isSuccess()) { // An error occurred, display it commits.emplace_back( "hex.ui.common.error"_lang, "HTTP " + std::to_string(response.getStatusCode()), "", "", "" ); return { }; } // A valid response was received, parse it try { json = nlohmann::json::parse(response.getData()); for (auto &commit: json) { const auto message = commit["commit"]["message"].get(); // Split commit title and description. They're separated by two newlines. const auto messageEnd = message.find("\n\n"); auto commitTitle = messageEnd == std::string::npos ? message : message.substr(0, messageEnd); auto commitDescription = messageEnd == std::string::npos ? "" : message.substr(commitTitle.size() + 2); auto url = commit["html_url"].get(); auto sha = commit["sha"].get(); auto date = commit["commit"]["author"]["date"].get(); auto author = hex::format("{} <{}>", commit["commit"]["author"]["name"].get(), commit["commit"]["author"]["email"].get() ); // Move the commit data into the list of commits commits.emplace_back( std::move(sha), std::move(commitTitle), std::move(commitDescription), std::move(author), std::move(date), std::move(url) ); } } catch (std::exception &e) { commits.emplace_back( "hex.ui.common.error"_lang, e.what(), "", "", "" ); } return commits; } void ViewAbout::drawCommitHistoryPage() { static std::vector commits; // Set up the request to get the commit history the first time the page is opened AT_FIRST_TIME { static HttpRequest request("GET", GitHubApiURL + std::string("/commits?per_page=100")); m_commitHistoryRequest = request.execute(); }; // Wait for the request to finish and parse the response if (m_commitHistoryRequest.valid()) { if (m_commitHistoryRequest.wait_for(std::chrono::seconds(0)) == std::future_status::ready) { commits = parseCommits(m_commitHistoryRequest.get()); } else { // Draw a spinner while the commits are loading ImGuiExt::TextSpinner("hex.ui.common.loading"_lang); } } // Draw commits table if (commits.empty()) return; ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2()); ImGuiExt::BeginSubWindow("Commits", ImGui::GetContentRegionAvail()); ImGui::PopStyleVar(); this->drawCommitsTable(commits); ImGuiExt::EndSubWindow(); } void ViewAbout::drawCommitsTable(const auto& commits) { if (ImGui::BeginTable("##commits", 2, ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg | ImGuiTableFlags_SizingFixedFit | ImGuiTableFlags_ScrollY)) { // Draw commits for (const auto &commit: commits) { ImGui::PushID(commit.hash.c_str()); ImGui::TableNextRow(); this->drawCommitRow(commit); ImGui::PopID(); } ImGui::EndTable(); } } void ViewAbout::drawCommitRow(const auto &commit) { // Draw hover tooltip ImGui::TableNextColumn(); if (ImGui::Selectable("##commit", false, ImGuiSelectableFlags_SpanAllColumns)) { hex::openWebpage(commit.url); } if (ImGui::IsItemHovered()) { if (ImGui::BeginTooltip()) { // Draw author and commit date ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_Highlight), "{}", commit.author); ImGui::SameLine(); ImGuiExt::TextFormatted("@ {}", commit.date.c_str()); // Draw description if there is one if (!commit.description.empty()) { ImGui::Separator(); ImGuiExt::TextFormatted("{}", commit.description); } ImGui::EndTooltip(); } } // Draw commit hash ImGui::SameLine(0, 0); ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_Highlight), "{}", commit.hash.substr(0, 7)); // Draw the commit message ImGui::TableNextColumn(); const ImColor color = [&] { if (commit.hash == ImHexApi::System::getCommitHash(true)) return ImGui::GetStyleColorVec4(ImGuiCol_HeaderActive); else return ImGui::GetStyleColorVec4(ImGuiCol_Text); }(); ImGuiExt::TextFormattedColored(color, commit.message); } void ViewAbout::drawLicensePage() { const auto indentation = 50_scaled; ImGui::Indent(indentation); ImGuiExt::TextFormattedWrapped("{}", romfs::get("licenses/LICENSE").string()); ImGui::Unindent(indentation); } void ViewAbout::drawAboutPopup() { struct Tab { using Function = void (ViewAbout::*)(); const char *unlocalizedName; Function function; }; constexpr std::array Tabs = { Tab { "ImHex", &ViewAbout::drawAboutMainPage }, Tab { "hex.builtin.view.help.about.contributor", &ViewAbout::drawContributorPage }, Tab { "hex.builtin.view.help.about.libs", &ViewAbout::drawLibraryCreditsPage }, Tab { "hex.builtin.view.help.about.plugins", &ViewAbout::drawLoadedPlugins }, Tab { "hex.builtin.view.help.about.paths", &ViewAbout::drawPathsPage }, Tab { "hex.builtin.view.help.about.release_notes", &ViewAbout::drawReleaseNotesPage }, Tab { "hex.builtin.view.help.about.commits", &ViewAbout::drawCommitHistoryPage }, Tab { "hex.builtin.view.help.about.license", &ViewAbout::drawLicensePage }, }; // Allow the window to be closed by pressing ESC if (ImGui::IsKeyDown(ImGuiKey_Escape)) ImGui::CloseCurrentPopup(); if (ImGui::BeginTabBar("about_tab_bar")) { // Draw all tabs for (const auto &[unlocalizedName, function] : Tabs) { if (ImGui::BeginTabItem(Lang(unlocalizedName))) { ImGui::NewLine(); if (ImGui::BeginChild(1)) { (this->*function)(); } ImGui::EndChild(); ImGui::EndTabItem(); } } ImGui::EndTabBar(); } } void ViewAbout::drawContent() { this->drawAboutPopup(); } }