#include "views/view_help.hpp" #include namespace hex { ViewHelp::ViewHelp() : View("hex.view.help.about.name"_lang) { } ViewHelp::~ViewHelp() { } static void drawTitle(const std::string &title) { ImGui::TextColored(ImVec4(0.6F, 0.6F, 1.0F, 1.0F), title.c_str()); } static void drawCodeSegment(const std::string &id, const std::string &code) { ImGui::PushStyleColor(ImGuiCol_ChildBg, ImVec4(0.2F, 0.2F, 0.2F, 0.3F)); ImGui::BeginChild(id.c_str(), ImVec2(-1, ImGui::CalcTextSize(code.c_str()).y)); ImGui::Text("%s", code.c_str()); ImGui::EndChild(); ImGui::NewLine(); ImGui::PopStyleColor(); }; void ViewHelp::drawAboutPopup() { if (ImGui::BeginPopupModal("hex.view.help.about.name"_lang, &this->m_aboutWindowOpen, ImGuiWindowFlags_AlwaysAutoResize)) { ImGui::Text("ImHex Hex Editor v%s by WerWolv -", IMHEX_VERSION); #if defined(GIT_BRANCH) && defined(GIT_COMMIT_HASH) ImGui::SameLine(); if (ImGui::Hyperlink(hex::format("%s@%s", GIT_BRANCH, GIT_COMMIT_HASH).c_str())) hex::openWebpage("https://github.com/WerWolv/ImHex/commit/" GIT_COMMIT_HASH); #endif ImGui::TextUnformatted("hex.view.help.about.translator"_lang); ImGui::TextUnformatted("hex.view.help.about.source"_lang); ImGui::SameLine(); if (ImGui::Hyperlink("WerWolv/ImHex")) hex::openWebpage("https://github.com/WerWolv/ImHex"); ImGui::NewLine(); ImGui::Text("hex.view.help.about.donations"_lang); ImGui::Separator(); constexpr const char* Links[] = { "https://werwolv.net/donate", "https://www.patreon.com/werwolv", "https://github.com/sponsors/WerWolv" }; ImGui::TextWrapped("hex.view.help.about.thanks"_lang); ImGui::NewLine(); for (auto &link : Links) { if (ImGui::Hyperlink(link)) hex::openWebpage(link); } ImGui::NewLine(); ImGui::Text("hex.view.help.about.libs"_lang); ImGui::Separator(); ImGui::PushStyleColor(ImGuiCol_ChildBg, ImVec4(0.2F, 0.2F, 0.2F, 0.3F)); ImGui::BulletText("ImGui by ocornut"); ImGui::BulletText("imgui_club by ocornut"); ImGui::BulletText("ImGui-Addons by gallickgunner"); ImGui::BulletText("ImGuiColorTextEdit by BalazsJako"); ImGui::BulletText("capstone by aquynh"); ImGui::BulletText("JSON for Modern C++ by nlohmann"); ImGui::NewLine(); ImGui::BulletText("GNU libmagic"); ImGui::BulletText("OpenSSL libcrypto"); ImGui::BulletText("GLFW3"); ImGui::BulletText("LLVM"); ImGui::BulletText("Python 3"); ImGui::BulletText("FreeType"); ImGui::PopStyleColor(); if (ImGui::IsKeyDown(ImGui::GetKeyIndex(ImGuiKey_Escape))) ImGui::CloseCurrentPopup(); ImGui::EndPopup(); } } void ViewHelp::drawPatternHelpPopup() { if (!this->m_patternHelpWindowOpen) return; ImGui::SetNextWindowSizeConstraints(ImVec2(450, 300), ImVec2(2000, 1000)); if (ImGui::Begin("Pattern Language Cheat Sheet", &this->m_patternHelpWindowOpen)) { ImGui::Text("ImHex Pattern Language Cheat Sheet"); ImGui::Separator(); ImGui::NewLine(); drawTitle("Preprocessor directives"); ImGui::TextWrapped( "Preprocessor directives can be used to alter the code before it's being parsed. Supported are " "#define to replace one string with another and #include to include a separate file"); drawCodeSegment("preprocessor", "#define HEADER_OFFSET 0x100\n" "#include \n" "#include \"mypattern.hexpat\"" ); drawTitle("Pragma directives"); ImGui::TextWrapped( "Pragma directives are used to give ImHex additional information about the code being read. Currently " "the following directives are supported."); drawCodeSegment("pragma", "// Allow this file to be loaded as pattern when a file\n" "// identified as application/x-executable gets loaded\n" "#pragma MIME application/x-executable\n" ); drawTitle("Built-in types"); ImGui::TextWrapped( "The following built-in types are available for use"); drawCodeSegment("built-in", "u8, s8\n" "u16, s16\n" "u32, s32\n" "u64, s64\n" "u128, s128\n" "float, double" ); drawTitle("Variables and Arrays"); ImGui::TextWrapped( "Normal variables as well as arrays are used to highlight and display values. " "It is possible to create arrays within structs and unions that use the value of a previously " "declared variable as size. In order to override the native / globally set endianess setting, every " "type in a variable declaration may be prefixed with be or le to turn only it into a big or little endian type."); drawCodeSegment("vars arrays", "u32 variable;\n" "s8 string[16];\n" "u8 customSizedArray[variable];\n" "be u32 bigEndianVariable;" ); drawTitle("Structs"); ImGui::TextWrapped( "To bundle multiple variables together, a struct can be used. To insert padding bytes which won't show " "up in the pattern data view or be highlighted, use the padding[size] syntax."); drawCodeSegment("struct", "struct Header {\n" " u32 magic;\n" " u8 version;\n" " padding[4];\n" " Flags flags;\n" "};" ); drawTitle("Unions"); ImGui::TextWrapped( "A union is used to make two or more variables occupy the same region of memory. " "The union will have the size of the biggest contained variable."); drawCodeSegment("union", "union Color {\n" " u32 rgba;\n" " Components components;\n" "};" ); drawTitle("Pointers"); ImGui::TextWrapped( "\"Another possible type of member in structs and unions are pointers. They are variables" "whose value is used as an offset from the start of the file to locate the actual offset. " "The leading type is treated as the data being pointed to and the trailing type as the size of the pointer."); drawCodeSegment("pointer", "Data *data : u16;" ); drawTitle("Bitfields"); ImGui::TextWrapped( "To decode values that are stored in fields that don't follow the typical 8 bit alignment, bitfields can be used. " "The size of these fields get specified in numbers of bits."); drawCodeSegment("bitfield", "bitfield Permission {\n" " r : 1;\n" " w : 1;\n" " x : 1;\n" "};" ); drawTitle("Enum"); ImGui::TextWrapped( "If a value can only be a few specific values with special meaning, an enum can be used. " "The underlying type has to be specified using a unsigned, built-in type after the name. " "Entries can be listed with or without a value. The values start counting at zero and increase by one " "for every next entry"); drawCodeSegment("enum", "enum OperatingSystem : u8 {\n" " Windows = 0x10,\n" " MacOSX,\n" " Linux = 'L'\n" "};" ); drawTitle("Using declarations"); ImGui::TextWrapped( "A using declaration can be used to create type aliases for already existing types. This can be " "a built-in type, a struct, enum or another alias type."); drawCodeSegment("using", "using magic_t = u32;" ); drawTitle("Comments"); ImGui::TextWrapped( "To create a comment the C // or /* */ syntax can be used. //-style comments end at the next new line " "and /*-style comments only end when at the next */."); drawCodeSegment("comment", "// This is a single line comment\n\n" "/* This is a\n" "multiline comment */" ); drawTitle("Variable placement"); ImGui::TextWrapped( "In order to highlight bytes and displaying their value in the pattern data window, " "a variable needs to be created and placed in memory. The following line of code creates" "a unsigned 32 bit variable named data and places it at offset 0x100." ); drawCodeSegment("var placement", "u32 data @ 0x100;"); } ImGui::End(); } void ViewHelp::drawMathEvaluatorHelp() { if (!this->m_mathHelpWindowOpen) return; ImGui::SetNextWindowSizeConstraints(ImVec2(450, 300), ImVec2(2000, 1000)); if (ImGui::Begin("Calculator Cheat Sheet", &this->m_mathHelpWindowOpen)) { ImGui::Text("ImHex Math Evaluator Cheat Sheet"); ImGui::Separator(); ImGui::NewLine(); ImGui::TextWrapped("ImGui has a simple math evaluator / calculator built-in. It works by parsing the " "expressions passed to it and displaying the result in the History table."); ImGui::NewLine(); drawTitle("Basic Operators"); ImGui::TextWrapped( "The following basic mathematical operators are supported"); drawCodeSegment("basicOperators", "+ : Addition\n" "- : Subtraction\n" "* : Multiplication\n" "/ : Division\n" "% : Modulus\n" "** : Exponentiation\n" "() : Brackets\n" "x = : Variable assignment"); drawTitle("Bitwise Operators"); ImGui::TextWrapped( "The following bitwise operators are supported"); drawCodeSegment("bitwiseOperators", "& : Bitwise AND\n" "| : Bitwise OR\n" "^ : Bitwise XOR\n" "~ : Bitwise NOT\n" "<< : Logical shift left\n" ">> : Logical shift right\n" "## : Bitwise concatenation"); drawTitle("Boolean Operators"); ImGui::TextWrapped( "The following bitwise operators are supported"); drawCodeSegment("booleanOperators", "&& : Boolean AND\n" "|| : Boolean OR\n" "^^ : Boolean XOR\n" "! : Boolean NOT\n" "== : Equality comparison\n" "!= : Inequality comparison\n" "> : Greater than comparison\n" "< : Less than comparison\n" ">= : Greater than or equals comparison\n" "<= : Less than or equals comparison"); drawTitle("Built-in Functions"); ImGui::TextWrapped( "The following functions are built into the evaluator"); drawCodeSegment("functions", "sin(x) : Sine function\n" "cos(x) : Cosine function\n" "tan(x) : Tangent function\n" "ceil(x) : Rounds value to next bigger integer\n" "floor(x) : Rounds value to next smaller integer\n" "sign(x) : Returns the value's sign\n" "abs(x) : Absolute value\n" "ln(x) : Natural Logarithm\n" "lb(x) : Logarithm with base 2\n" "log(x, b) : Logarithm with custom base. Defaults to base 10\n" "clear() : Clear History and variables\n" "read(a) : Read 1 byte from loaded data\n" "write(a, x) : Write 1 byte to loaded data"); } ImGui::End(); } void ViewHelp::drawContent() { if (!this->m_aboutWindowOpen && !this->m_mathHelpWindowOpen && !this->m_patternHelpWindowOpen) this->getWindowOpenState() = false; this->drawAboutPopup(); this->drawPatternHelpPopup(); this->drawMathEvaluatorHelp(); } void ViewHelp::drawMenu() { if (ImGui::BeginMenu("hex.menu.help"_lang)) { if (ImGui::MenuItem("hex.view.help.about.name"_lang, "")) { View::doLater([] { ImGui::OpenPopup("hex.view.help.about.name"_lang); }); this->m_aboutWindowOpen = true; this->getWindowOpenState() = true; } ImGui::Separator(); if (ImGui::MenuItem("hex.view.help.pattern_cheat_sheet"_lang, "")) { this->m_patternHelpWindowOpen = true; this->getWindowOpenState() = true; } if (ImGui::MenuItem("hex.view.help.calc_cheat_sheet"_lang, "")) { this->m_mathHelpWindowOpen = true; this->getWindowOpenState() = true; } ImGui::EndMenu(); } } }