2021-01-22 18:01:39 +01:00
|
|
|
|
#include <hex/plugin.hpp>
|
|
|
|
|
|
2021-08-28 00:45:59 +02:00
|
|
|
|
#include <hex/helpers/net.hpp>
|
|
|
|
|
|
2021-01-22 18:01:39 +01:00
|
|
|
|
#include <regex>
|
2021-08-28 00:45:59 +02:00
|
|
|
|
#include <chrono>
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
#include <llvm/Demangle/Demangle.h>
|
|
|
|
|
#include "math_evaluator.hpp"
|
|
|
|
|
|
|
|
|
|
namespace hex::plugin::builtin {
|
|
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
2021-08-28 16:02:53 +02:00
|
|
|
|
using namespace std::literals::string_literals;
|
2021-08-28 00:45:59 +02:00
|
|
|
|
using namespace std::literals::chrono_literals;
|
|
|
|
|
|
2021-08-28 18:15:20 +02:00
|
|
|
|
int updateStringSizeCallback(ImGuiInputTextCallbackData *data) {
|
|
|
|
|
auto &mathInput = *static_cast<std::string*>(data->UserData);
|
|
|
|
|
|
|
|
|
|
mathInput.resize(data->BufTextLen);
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-22 18:01:39 +01:00
|
|
|
|
void drawDemangler() {
|
|
|
|
|
static std::vector<char> mangledBuffer(0xF'FFFF, 0x00);
|
|
|
|
|
static std::string demangledName;
|
|
|
|
|
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::InputText("hex.builtin.tools.demangler.mangled"_lang, mangledBuffer.data(), 0xF'FFFF)) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
demangledName = llvm::demangle(mangledBuffer.data());
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-11 23:09:45 +01:00
|
|
|
|
ImGui::InputText("hex.builtin.tools.demangler.demangled"_lang, demangledName.data(), demangledName.size(), ImGuiInputTextFlags_ReadOnly);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void drawASCIITable() {
|
|
|
|
|
static bool asciiTableShowOctal = false;
|
|
|
|
|
|
|
|
|
|
ImGui::BeginTable("##asciitable", 4);
|
|
|
|
|
ImGui::TableSetupColumn("");
|
|
|
|
|
ImGui::TableSetupColumn("");
|
|
|
|
|
ImGui::TableSetupColumn("");
|
|
|
|
|
ImGui::TableSetupColumn("");
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
|
|
|
|
|
for (u8 tablePart = 0; tablePart < 4; tablePart++) {
|
|
|
|
|
ImGui::BeginTable("##asciitablepart", asciiTableShowOctal ? 4 : 3, ImGuiTableFlags_BordersInnerV | ImGuiTableFlags_BordersOuter | ImGuiTableFlags_RowBg);
|
|
|
|
|
ImGui::TableSetupColumn("dec");
|
|
|
|
|
if (asciiTableShowOctal)
|
|
|
|
|
ImGui::TableSetupColumn("oct");
|
|
|
|
|
ImGui::TableSetupColumn("hex");
|
|
|
|
|
ImGui::TableSetupColumn("char");
|
|
|
|
|
|
|
|
|
|
ImGui::TableHeadersRow();
|
|
|
|
|
|
|
|
|
|
u32 rowCount = 0;
|
|
|
|
|
for (u8 i = 0; i < 0x80 / 4; i++) {
|
|
|
|
|
ImGui::TableNextRow(ImGuiTableRowFlags_Headers);
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::Text("%02d", i + 32 * tablePart);
|
|
|
|
|
|
|
|
|
|
if (asciiTableShowOctal) {
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::Text("0o%02o", i + 32 * tablePart);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::Text("0x%02x", i + 32 * tablePart);
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::Text("%s", makePrintable(i + 32 * tablePart).c_str());
|
|
|
|
|
|
|
|
|
|
ImGui::TableSetBgColor(ImGuiTableBgTarget_RowBg0, ((rowCount % 2) == 0) ? 0xFF101010 : 0xFF303030);
|
|
|
|
|
|
|
|
|
|
rowCount++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
}
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
|
2021-02-11 23:09:45 +01:00
|
|
|
|
ImGui::Checkbox("hex.builtin.tools.ascii_table.octal"_lang, &asciiTableShowOctal);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void drawRegexReplacer() {
|
|
|
|
|
static std::vector<char> regexInput(0xF'FFFF, 0x00);;
|
|
|
|
|
static std::vector<char> regexPattern(0xF'FFFF, 0x00);;
|
|
|
|
|
static std::vector<char> replacePattern(0xF'FFFF, 0x00);;
|
|
|
|
|
static std::string regexOutput(0xF'FFFF, 0x00);;
|
|
|
|
|
|
|
|
|
|
bool shouldInvalidate;
|
|
|
|
|
|
2021-02-11 23:09:45 +01:00
|
|
|
|
shouldInvalidate = ImGui::InputText("hex.builtin.tools.regex_replacer.pattern"_lang, regexPattern.data(), regexPattern.size());
|
|
|
|
|
shouldInvalidate = ImGui::InputText("hex.builtin.tools.regex_replacer.replace"_lang, replacePattern.data(), replacePattern.size()) || shouldInvalidate;
|
|
|
|
|
shouldInvalidate = ImGui::InputTextMultiline("hex.builtin.tools.regex_replacer.input"_lang, regexInput.data(), regexInput.size()) || shouldInvalidate;
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
if (shouldInvalidate) {
|
|
|
|
|
try {
|
|
|
|
|
regexOutput = std::regex_replace(regexInput.data(), std::regex(regexPattern.data()), replacePattern.data());
|
|
|
|
|
} catch (std::regex_error&) {}
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-11 23:09:45 +01:00
|
|
|
|
ImGui::InputTextMultiline("hex.builtin.tools.regex_replacer.input"_lang, regexOutput.data(), regexOutput.size(), ImVec2(0, 0), ImGuiInputTextFlags_ReadOnly);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void drawColorPicker() {
|
|
|
|
|
static std::array<float, 4> pickedColor = { 0 };
|
|
|
|
|
|
|
|
|
|
ImGui::SetNextItemWidth(300.0F);
|
2021-02-11 23:09:45 +01:00
|
|
|
|
ImGui::ColorPicker4("hex.builtin.tools.color"_lang, pickedColor.data(),
|
2021-01-22 18:01:39 +01:00
|
|
|
|
ImGuiColorEditFlags_Uint8 | ImGuiColorEditFlags_AlphaBar | ImGuiColorEditFlags_DisplayRGB | ImGuiColorEditFlags_DisplayHex);
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void drawMathEvaluator() {
|
|
|
|
|
static std::vector<long double> mathHistory;
|
|
|
|
|
static std::string lastMathError;
|
2021-08-28 18:15:20 +02:00
|
|
|
|
static auto mathInput = []{
|
|
|
|
|
std::string s;
|
|
|
|
|
s.reserve(0xFFFF);
|
|
|
|
|
std::memset(s.data(), 0x00, s.capacity());
|
|
|
|
|
|
|
|
|
|
return s;
|
|
|
|
|
}();
|
|
|
|
|
bool evaluate = false;
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
static MathEvaluator mathEvaluator = [&]{
|
|
|
|
|
MathEvaluator evaluator;
|
|
|
|
|
|
|
|
|
|
evaluator.registerStandardVariables();
|
|
|
|
|
evaluator.registerStandardFunctions();
|
|
|
|
|
|
|
|
|
|
evaluator.setFunction("clear", [&](auto args) -> std::optional<long double> {
|
|
|
|
|
mathHistory.clear();
|
|
|
|
|
lastMathError.clear();
|
|
|
|
|
mathEvaluator.getVariables().clear();
|
|
|
|
|
mathEvaluator.registerStandardVariables();
|
2021-08-28 18:15:20 +02:00
|
|
|
|
mathInput.clear();
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
return { };
|
|
|
|
|
}, 0, 0);
|
|
|
|
|
|
|
|
|
|
evaluator.setFunction("read", [](auto args) -> std::optional<long double> {
|
|
|
|
|
u8 value = 0;
|
|
|
|
|
|
|
|
|
|
auto provider = SharedData::currentProvider;
|
|
|
|
|
if (provider == nullptr || !provider->isReadable() || args[0] >= provider->getActualSize())
|
|
|
|
|
return { };
|
|
|
|
|
|
|
|
|
|
provider->read(args[0], &value, sizeof(u8));
|
|
|
|
|
|
|
|
|
|
return value;
|
|
|
|
|
}, 1, 1);
|
|
|
|
|
|
|
|
|
|
evaluator.setFunction("write", [](auto args) -> std::optional<long double> {
|
|
|
|
|
auto provider = SharedData::currentProvider;
|
|
|
|
|
if (provider == nullptr || !provider->isWritable() || args[0] >= provider->getActualSize())
|
|
|
|
|
return { };
|
|
|
|
|
|
|
|
|
|
if (args[1] > 0xFF)
|
|
|
|
|
return { };
|
|
|
|
|
|
|
|
|
|
u8 value = args[1];
|
|
|
|
|
provider->write(args[0], &value, sizeof(u8));
|
|
|
|
|
|
|
|
|
|
return { };
|
|
|
|
|
}, 2, 2);
|
|
|
|
|
|
|
|
|
|
return std::move(evaluator);
|
|
|
|
|
}();
|
|
|
|
|
|
|
|
|
|
enum class MathDisplayType { Standard, Scientific, Engineering, Programmer } mathDisplayType;
|
|
|
|
|
if (ImGui::BeginTabBar("##mathFormatTabBar")) {
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::BeginTabItem("hex.builtin.tools.format.standard"_lang)) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
mathDisplayType = MathDisplayType::Standard;
|
|
|
|
|
ImGui::EndTabItem();
|
|
|
|
|
}
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::BeginTabItem("hex.builtin.tools.format.scientific"_lang)) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
mathDisplayType = MathDisplayType::Scientific;
|
|
|
|
|
ImGui::EndTabItem();
|
|
|
|
|
}
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::BeginTabItem("hex.builtin.tools.format.engineering"_lang)) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
mathDisplayType = MathDisplayType::Engineering;
|
|
|
|
|
ImGui::EndTabItem();
|
|
|
|
|
}
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::BeginTabItem("hex.builtin.tools.format.programmer"_lang)) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
mathDisplayType = MathDisplayType::Programmer;
|
|
|
|
|
ImGui::EndTabItem();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::EndTabBar();
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-28 18:15:20 +02:00
|
|
|
|
if (ImGui::BeginTable("##mathWrapper", 3)) {
|
|
|
|
|
ImGui::TableSetupColumn("##keypad", ImGuiTableColumnFlags_WidthFixed | ImGuiTableColumnFlags_NoResize);
|
|
|
|
|
ImGui::TableSetupColumn("##results", ImGuiTableColumnFlags_WidthStretch, 0.666);
|
|
|
|
|
ImGui::TableSetupColumn("##variables", ImGuiTableColumnFlags_WidthStretch, 0.666);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
|
ImGui::TableNextColumn();
|
2021-08-28 18:15:20 +02:00
|
|
|
|
|
|
|
|
|
auto buttonSize = ImVec2(3, 2) * ImGui::GetTextLineHeightWithSpacing();
|
|
|
|
|
|
|
|
|
|
if (ImGui::Button("Ans", buttonSize)) mathInput += "ans"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("Pi", buttonSize)) mathInput += "pi"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("e", buttonSize)) mathInput += "e"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("CE", buttonSize)) mathInput.clear(); ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button(ICON_FA_BACKSPACE, buttonSize)) mathInput.clear(); ImGui::SameLine();
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
switch (mathDisplayType) {
|
|
|
|
|
case MathDisplayType::Standard:
|
|
|
|
|
case MathDisplayType::Scientific:
|
|
|
|
|
case MathDisplayType::Engineering:
|
|
|
|
|
if (ImGui::Button("x²", buttonSize)) mathInput += "** 2"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("1/x", buttonSize)) mathInput += "1/"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("|x|", buttonSize)) mathInput += "abs"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("exp", buttonSize)) mathInput += "e ** "; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("%", buttonSize)) mathInput += "%"; ImGui::SameLine();
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Programmer:
|
|
|
|
|
if (ImGui::Button("<<", buttonSize)) mathInput += "<<"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button(">>", buttonSize)) mathInput += ">>"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("&", buttonSize)) mathInput += "&"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("|", buttonSize)) mathInput += "|"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("^", buttonSize)) mathInput += "^"; ImGui::SameLine();
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
if (ImGui::Button("sqrt", buttonSize)) mathInput += "sqrt"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("(", buttonSize)) mathInput += "("; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button(")", buttonSize)) mathInput += ")"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("sign", buttonSize)) mathInput += "sign"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("÷", buttonSize)) mathInput += "/"; ImGui::SameLine();
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
if (ImGui::Button("xª", buttonSize)) mathInput += "**"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("7", buttonSize)) mathInput += "7"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("8", buttonSize)) mathInput += "8"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("9", buttonSize)) mathInput += "9"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("×", buttonSize)) mathInput += "*"; ImGui::SameLine();
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
if (ImGui::Button("log", buttonSize)) mathInput += "log"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("4", buttonSize)) mathInput += "4"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("5", buttonSize)) mathInput += "5"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("6", buttonSize)) mathInput += "6"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("-", buttonSize)) mathInput += "-"; ImGui::SameLine();
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
if (ImGui::Button("ln", buttonSize)) mathInput += "ln"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("1", buttonSize)) mathInput += "1"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("2", buttonSize)) mathInput += "2"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("3", buttonSize)) mathInput += "3"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("+", buttonSize)) mathInput += "+"; ImGui::SameLine();
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
if (ImGui::Button("lb", buttonSize)) mathInput += "lb"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("x=", buttonSize)) mathInput += "="; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button("0", buttonSize)) mathInput += "0"; ImGui::SameLine();
|
|
|
|
|
if (ImGui::Button(".", buttonSize)) mathInput += "."; ImGui::SameLine();
|
|
|
|
|
|
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_Button, ImGui::GetCustomColorVec4(ImGuiCustomCol_DescButtonHovered));
|
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_ButtonActive, ImGui::GetCustomColorVec4(ImGuiCustomCol_DescButton));
|
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_ButtonHovered, ImGui::GetCustomColorVec4(ImGuiCustomCol_DescButtonActive));
|
|
|
|
|
if (ImGui::Button("=", buttonSize)) evaluate = true; ImGui::SameLine();
|
|
|
|
|
ImGui::PopStyleColor(3);
|
|
|
|
|
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
|
|
|
|
|
if (ImGui::BeginTable("##mathHistory", 1, ImGuiTableFlags_ScrollY | ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg, ImVec2(0, 300))) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
ImGui::TableSetupScrollFreeze(0, 1);
|
2021-02-11 23:09:45 +01:00
|
|
|
|
ImGui::TableSetupColumn("hex.builtin.tools.history"_lang);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
ImGuiListClipper clipper;
|
|
|
|
|
clipper.Begin(mathHistory.size());
|
|
|
|
|
|
|
|
|
|
ImGui::TableHeadersRow();
|
|
|
|
|
while (clipper.Step()) {
|
|
|
|
|
for (u64 i = clipper.DisplayStart; i < clipper.DisplayEnd; i++) {
|
|
|
|
|
if (i == 0)
|
|
|
|
|
ImGui::PushStyleColor(ImGuiCol_Text, ImU32(ImColor(0xA5, 0x45, 0x45)));
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
|
|
|
|
|
switch (mathDisplayType) {
|
|
|
|
|
case MathDisplayType::Standard:
|
|
|
|
|
ImGui::Text("%.3Lf", mathHistory[(mathHistory.size() - 1) - i]);
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Scientific:
|
|
|
|
|
ImGui::Text("%.6Le", mathHistory[(mathHistory.size() - 1) - i]);
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Engineering:
|
|
|
|
|
ImGui::Text("%s", hex::toEngineeringString(mathHistory[(mathHistory.size() - 1) - i]).c_str());
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Programmer:
|
|
|
|
|
ImGui::Text("0x%llX (%llu)",
|
|
|
|
|
u64(mathHistory[(mathHistory.size() - 1) - i]),
|
|
|
|
|
u64(mathHistory[(mathHistory.size() - 1) - i]));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (i == 0)
|
|
|
|
|
ImGui::PopStyleColor();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clipper.End();
|
|
|
|
|
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
2021-08-28 18:15:20 +02:00
|
|
|
|
if (ImGui::BeginTable("##mathVariables", 2, ImGuiTableFlags_ScrollY | ImGuiTableFlags_Borders | ImGuiTableFlags_RowBg, ImVec2(0, 300))) {
|
2021-01-22 18:01:39 +01:00
|
|
|
|
ImGui::TableSetupScrollFreeze(0, 1);
|
2021-02-11 23:09:45 +01:00
|
|
|
|
ImGui::TableSetupColumn("hex.builtin.tools.name"_lang);
|
|
|
|
|
ImGui::TableSetupColumn("hex.builtin.tools.value"_lang);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
|
|
|
|
|
ImGui::TableHeadersRow();
|
|
|
|
|
for (const auto &[name, value] : mathEvaluator.getVariables()) {
|
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::TextUnformatted(name.c_str());
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
switch (mathDisplayType) {
|
|
|
|
|
case MathDisplayType::Standard:
|
|
|
|
|
ImGui::Text("%.3Lf", value);
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Scientific:
|
|
|
|
|
ImGui::Text("%.6Le", value);
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Engineering:
|
|
|
|
|
ImGui::Text("%s", hex::toEngineeringString(value).c_str());
|
|
|
|
|
break;
|
|
|
|
|
case MathDisplayType::Programmer:
|
|
|
|
|
ImGui::Text("0x%llX (%llu)", u64(value), u64(value));
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
}
|
2021-08-28 18:15:20 +02:00
|
|
|
|
|
|
|
|
|
ImGui::PushItemWidth(ImGui::GetContentRegionAvailWidth());
|
|
|
|
|
if (ImGui::InputText("##input", mathInput.data(), mathInput.capacity(), ImGuiInputTextFlags_EnterReturnsTrue | ImGuiInputTextFlags_AutoSelectAll | ImGuiInputTextFlags_CallbackEdit, updateStringSizeCallback, &mathInput)) {
|
|
|
|
|
ImGui::SetKeyboardFocusHere();
|
|
|
|
|
evaluate = true;
|
|
|
|
|
}
|
|
|
|
|
ImGui::PopItemWidth();
|
|
|
|
|
|
|
|
|
|
if (!lastMathError.empty())
|
|
|
|
|
ImGui::TextColored(ImColor(0xA00040FF), "%s", hex::format("hex.builtin.tools.error"_lang, lastMathError).c_str());
|
|
|
|
|
else
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
|
|
|
|
|
if (evaluate) {
|
|
|
|
|
std::optional<long double> result;
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
result = mathEvaluator.evaluate(mathInput);
|
|
|
|
|
} catch (std::invalid_argument &e) {
|
|
|
|
|
lastMathError = e.what();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (result.has_value()) {
|
|
|
|
|
mathHistory.push_back(result.value());
|
|
|
|
|
mathInput.clear();
|
|
|
|
|
lastMathError.clear();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-22 18:01:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-05 00:17:56 +01:00
|
|
|
|
void drawBaseConverter() {
|
|
|
|
|
static char buffer[4][0xFFF] = { { '0' }, { '0' }, { '0' }, { '0' } };
|
|
|
|
|
|
|
|
|
|
static auto CharFilter = [](ImGuiInputTextCallbackData *data) -> int {
|
|
|
|
|
switch (*static_cast<u32*>(data->UserData)) {
|
|
|
|
|
case 16: return std::isxdigit(data->EventChar);
|
|
|
|
|
case 10: return std::isdigit(data->EventChar);
|
|
|
|
|
case 8: return std::isdigit(data->EventChar) && data->EventChar < '8';
|
|
|
|
|
case 2: return data->EventChar == '0' || data->EventChar == '1';
|
|
|
|
|
default: return false;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static auto ConvertBases = [](u8 base) {
|
|
|
|
|
u64 number;
|
|
|
|
|
|
|
|
|
|
errno = 0;
|
|
|
|
|
switch (base) {
|
|
|
|
|
case 16: number = std::strtoull(buffer[1], nullptr, base); break;
|
|
|
|
|
case 10: number = std::strtoull(buffer[0], nullptr, base); break;
|
|
|
|
|
case 8: number = std::strtoull(buffer[2], nullptr, base); break;
|
|
|
|
|
case 2: number = std::strtoull(buffer[3], nullptr, base); break;
|
|
|
|
|
default: return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
auto base10String = std::to_string(number);
|
2021-03-03 19:58:22 +01:00
|
|
|
|
auto base16String = hex::format("0x{0:X}", number);
|
|
|
|
|
auto base8String = hex::format("{0:#o}", number);
|
2021-02-05 00:17:56 +01:00
|
|
|
|
auto base2String = hex::toBinaryString(number);
|
|
|
|
|
|
|
|
|
|
std::strncpy(buffer[0], base10String.c_str(), sizeof(buffer[0]));
|
|
|
|
|
std::strncpy(buffer[1], base16String.c_str(), sizeof(buffer[1]));
|
|
|
|
|
std::strncpy(buffer[2], base8String.c_str(), sizeof(buffer[2]));
|
|
|
|
|
std::strncpy(buffer[3], base2String.c_str(), sizeof(buffer[3]));
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
u8 base = 10;
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::InputText("hex.builtin.tools.base_converter.dec"_lang, buffer[0], 20 + 1, ImGuiInputTextFlags_CallbackCharFilter, CharFilter, &base))
|
2021-02-05 00:17:56 +01:00
|
|
|
|
ConvertBases(base);
|
|
|
|
|
|
|
|
|
|
base = 16;
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::InputText("hex.builtin.tools.base_converter.hex"_lang, buffer[1], 16 + 1, ImGuiInputTextFlags_CallbackCharFilter, CharFilter, &base))
|
2021-02-05 00:17:56 +01:00
|
|
|
|
ConvertBases(base);
|
|
|
|
|
|
|
|
|
|
base = 8;
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::InputText("hex.builtin.tools.base_converter.oct"_lang, buffer[2], 22 + 1, ImGuiInputTextFlags_CallbackCharFilter, CharFilter, &base))
|
2021-02-05 00:17:56 +01:00
|
|
|
|
ConvertBases(base);
|
|
|
|
|
|
|
|
|
|
base = 2;
|
2021-02-11 23:09:45 +01:00
|
|
|
|
if (ImGui::InputText("hex.builtin.tools.base_converter.bin"_lang, buffer[3], 64 + 1, ImGuiInputTextFlags_CallbackCharFilter, CharFilter, &base))
|
2021-02-05 00:17:56 +01:00
|
|
|
|
ConvertBases(base);
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-22 18:01:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
2021-07-27 22:46:37 +02:00
|
|
|
|
void drawPermissionsCalculator() {
|
|
|
|
|
static bool setuid, setgid, sticky;
|
|
|
|
|
static bool r[3], w[3], x[3];
|
|
|
|
|
|
|
|
|
|
ImGui::TextUnformatted("hex.builtin.tools.permissions.perm_bits"_lang);
|
|
|
|
|
ImGui::Separator();
|
|
|
|
|
|
|
|
|
|
if (ImGui::BeginTable("Permissions", 4, ImGuiTableFlags_Borders)) {
|
|
|
|
|
ImGui::TableSetupScrollFreeze(0, 1);
|
|
|
|
|
ImGui::TableSetupColumn("Special", ImGuiTableColumnFlags_NoSort);
|
|
|
|
|
ImGui::TableSetupColumn("User", ImGuiTableColumnFlags_NoSort);
|
|
|
|
|
ImGui::TableSetupColumn("Group", ImGuiTableColumnFlags_NoSort);
|
|
|
|
|
ImGui::TableSetupColumn("Other", ImGuiTableColumnFlags_NoSort);
|
|
|
|
|
|
|
|
|
|
ImGui::TableHeadersRow();
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
|
|
|
|
|
ImGui::Checkbox("setuid", &setuid);
|
|
|
|
|
ImGui::Checkbox("setgid", &setgid);
|
|
|
|
|
ImGui::Checkbox("Sticky bit", &sticky);
|
|
|
|
|
|
|
|
|
|
for (u8 i = 0; i < 3; i++) {
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
|
|
|
|
|
ImGui::PushID(i);
|
|
|
|
|
ImGui::Checkbox("Read", &r[i]);
|
|
|
|
|
ImGui::Checkbox("Write", &w[i]);
|
|
|
|
|
ImGui::Checkbox("Execute", &x[i]);
|
|
|
|
|
ImGui::PopID();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
ImGui::TextUnformatted("hex.builtin.tools.permissions.absolute"_lang);
|
|
|
|
|
ImGui::Separator();
|
|
|
|
|
|
|
|
|
|
auto result = hex::format("{}{}{}{}",
|
|
|
|
|
(setuid << 2) | (setgid << 1) | (sticky << 0),
|
|
|
|
|
(r[0] << 2) | (w[0] << 1) | (x[0] << 0),
|
|
|
|
|
(r[1] << 2) | (w[1] << 1) | (x[1] << 0),
|
|
|
|
|
(r[2] << 2) | (w[2] << 1) | (x[2] << 0));
|
|
|
|
|
ImGui::InputText("##permissions_absolute", result.data(), result.size(), ImGuiInputTextFlags_ReadOnly);
|
|
|
|
|
|
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
|
|
|
|
|
static const auto WarningColor = ImColor(0.92F, 0.25F, 0.2F, 1.0F);
|
|
|
|
|
if (setuid && !x[0])
|
2021-07-28 13:23:50 +02:00
|
|
|
|
ImGui::TextColored(WarningColor, "%s", static_cast<const char*>("hex.builtin.tools.permissions.setuid_error"_lang));
|
2021-07-27 22:46:37 +02:00
|
|
|
|
if (setgid && !x[1])
|
2021-07-28 13:23:50 +02:00
|
|
|
|
ImGui::TextColored(WarningColor, "%s", static_cast<const char*>("hex.builtin.tools.permissions.setgid_error"_lang));
|
2021-07-27 22:46:37 +02:00
|
|
|
|
if (sticky && !x[2])
|
2021-07-28 13:23:50 +02:00
|
|
|
|
ImGui::TextColored(WarningColor, "%s", static_cast<const char*>("hex.builtin.tools.permissions.sticky_error"_lang));
|
2021-07-27 22:46:37 +02:00
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-28 00:45:59 +02:00
|
|
|
|
void drawFileUploader() {
|
|
|
|
|
struct UploadedFile {
|
|
|
|
|
std::string fileName, link, size;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
static hex::Net net;
|
|
|
|
|
static std::future<Response<std::string>> uploadProcess;
|
|
|
|
|
static std::filesystem::path currFile;
|
|
|
|
|
static std::vector<UploadedFile> links;
|
|
|
|
|
|
|
|
|
|
bool uploading = uploadProcess.valid() && uploadProcess.wait_for(0s) != std::future_status::ready;
|
|
|
|
|
|
|
|
|
|
ImGui::Header("hex.builtin.tools.file_uploader.control"_lang, true);
|
|
|
|
|
if (!uploading) {
|
|
|
|
|
if (ImGui::Button("hex.builtin.tools.file_uploader.upload"_lang)) {
|
|
|
|
|
hex::openFileBrowser("hex.builtin.tools.file_uploader.done"_lang, DialogMode::Open, { }, [&](auto path){
|
|
|
|
|
uploadProcess = net.uploadFile("https://api.anonfiles.com/upload", path);
|
|
|
|
|
currFile = path;
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (ImGui::Button("hex.common.cancel"_lang)) {
|
|
|
|
|
net.cancel();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
|
|
|
|
|
ImGui::ProgressBar(net.getProgress(), ImVec2(0, 0), uploading ? nullptr : "Done!");
|
|
|
|
|
|
|
|
|
|
ImGui::Header("hex.builtin.tools.file_uploader.recent"_lang);
|
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
if (ImGui::BeginTable("##links", 3, ImGuiTableFlags_ScrollY | ImGuiTableFlags_Borders | ImGuiTableFlags_Resizable | ImGuiTableFlags_RowBg, ImVec2(0, 200))) {
|
2021-08-28 00:45:59 +02:00
|
|
|
|
ImGui::TableSetupScrollFreeze(0, 1);
|
|
|
|
|
ImGui::TableSetupColumn("hex.common.file"_lang);
|
|
|
|
|
ImGui::TableSetupColumn("hex.common.link"_lang);
|
|
|
|
|
ImGui::TableSetupColumn("hex.common.size"_lang);
|
|
|
|
|
ImGui::TableHeadersRow();
|
|
|
|
|
|
|
|
|
|
ImGuiListClipper clipper;
|
|
|
|
|
clipper.Begin(links.size());
|
|
|
|
|
|
|
|
|
|
while (clipper.Step()) {
|
|
|
|
|
for (s32 i = clipper.DisplayEnd - 1; i >= clipper.DisplayStart; i--) {
|
|
|
|
|
auto &[fileName, link, size] = links[i];
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextRow();
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::TextUnformatted(fileName.c_str());
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
if (ImGui::Hyperlink(link.c_str())) {
|
|
|
|
|
if (ImGui::GetMergedKeyModFlags() == ImGuiKeyModFlags_Ctrl)
|
|
|
|
|
hex::openWebpage(link);
|
|
|
|
|
else
|
|
|
|
|
ImGui::SetClipboardText(link.c_str());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
ImGui::InfoTooltip("hex.builtin.tools.file_uploader.tooltip"_lang);
|
|
|
|
|
|
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
|
ImGui::TextUnformatted(size.c_str());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
clipper.End();
|
|
|
|
|
|
|
|
|
|
ImGui::EndTable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (uploadProcess.valid() && uploadProcess.wait_for(0s) == std::future_status::ready) {
|
|
|
|
|
auto response = uploadProcess.get();
|
|
|
|
|
if (response.code == 200) {
|
|
|
|
|
try {
|
|
|
|
|
auto json = nlohmann::json::parse(response.body);
|
|
|
|
|
links.push_back({
|
|
|
|
|
currFile.filename().string(),
|
|
|
|
|
json["data"]["file"]["url"]["short"],
|
|
|
|
|
json["data"]["file"]["metadata"]["size"]["readable"]
|
|
|
|
|
});
|
|
|
|
|
} catch (...) {
|
|
|
|
|
View::showErrorPopup("hex.builtin.tools.file_uploader.invalid_response"_lang);
|
|
|
|
|
}
|
|
|
|
|
} else if (response.code == 0) {
|
|
|
|
|
// Canceled by user, no action needed
|
|
|
|
|
} else View::showErrorPopup(hex::format("hex.builtin.tools.file_uploader.error"_lang, response.code));
|
|
|
|
|
|
|
|
|
|
uploadProcess = { };
|
|
|
|
|
currFile.clear();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-28 16:02:53 +02:00
|
|
|
|
void drawWikiExplainer() {
|
|
|
|
|
static hex::Net net;
|
2021-08-28 18:15:47 +02:00
|
|
|
|
|
2021-08-28 16:02:53 +02:00
|
|
|
|
static std::string resultTitle, resultExtract;
|
|
|
|
|
static std::future<Response<std::string>> searchProcess;
|
|
|
|
|
static bool extendedSearch = false;
|
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
static auto searchString = []{
|
|
|
|
|
std::string s;
|
|
|
|
|
s.reserve(0xFFFF);
|
|
|
|
|
std::memset(s.data(), 0x00, s.capacity());
|
|
|
|
|
|
|
|
|
|
return s;
|
|
|
|
|
}();
|
|
|
|
|
|
2021-08-28 16:02:53 +02:00
|
|
|
|
constexpr static auto WikipediaApiUrl = "https://en.wikipedia.org/w/api.php?format=json&action=query&prop=extracts&explaintext&redirects=10&formatversion=2";
|
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
ImGui::Header("hex.builtin.tools.wiki_explain.control"_lang, true);
|
2021-08-28 16:02:53 +02:00
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
bool startSearch;
|
2021-08-28 16:02:53 +02:00
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
startSearch = ImGui::InputText("##search", searchString.data(), searchString.capacity(), ImGuiInputTextFlags_EnterReturnsTrue | ImGuiInputTextFlags_CallbackEdit, updateStringSizeCallback, &searchString);
|
2021-08-28 16:02:53 +02:00
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
ImGui::BeginDisabled(searchProcess.valid() && searchProcess.wait_for(0s) != std::future_status::ready || searchString.empty());
|
|
|
|
|
startSearch = ImGui::Button("hex.builtin.tools.wiki_explain.search"_lang) || startSearch;
|
2021-08-28 16:02:53 +02:00
|
|
|
|
ImGui::EndDisabled();
|
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
if (startSearch && !searchString.empty()) {
|
2021-08-28 16:02:53 +02:00
|
|
|
|
searchProcess = net.getString(WikipediaApiUrl + "&exintro"s + "&titles="s + net.encode(searchString));
|
|
|
|
|
}
|
|
|
|
|
|
2021-08-28 18:15:47 +02:00
|
|
|
|
ImGui::Header("hex.builtin.tools.wiki_explain.results"_lang);
|
2021-08-28 16:02:53 +02:00
|
|
|
|
|
|
|
|
|
if (ImGui::BeginChild("##summary", ImVec2(0, 300), true)) {
|
|
|
|
|
if (!resultTitle.empty() && !resultExtract.empty()) {
|
|
|
|
|
ImGui::HeaderColored(resultTitle.c_str(), ImGui::GetCustomColorVec4(ImGuiCustomCol_Highlight),true);
|
|
|
|
|
ImGui::TextWrapped("%s", resultExtract.c_str());
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
ImGui::EndChild();
|
|
|
|
|
|
|
|
|
|
if (searchProcess.valid() && searchProcess.wait_for(0s) == std::future_status::ready) {
|
|
|
|
|
try {
|
|
|
|
|
auto response = searchProcess.get();
|
|
|
|
|
if (response.code != 200) throw std::runtime_error("Invalid response");
|
|
|
|
|
|
|
|
|
|
auto json = nlohmann::json::parse(response.body);
|
|
|
|
|
|
|
|
|
|
resultTitle = json["query"]["pages"][0]["title"];
|
|
|
|
|
resultExtract = json["query"]["pages"][0]["extract"];
|
|
|
|
|
|
|
|
|
|
if (!extendedSearch && resultExtract.ends_with(':')) {
|
|
|
|
|
extendedSearch = true;
|
|
|
|
|
searchProcess = net.getString(WikipediaApiUrl + "&titles="s + net.encode(searchString));
|
|
|
|
|
resultTitle.clear();
|
|
|
|
|
resultExtract.clear();
|
|
|
|
|
} else {
|
|
|
|
|
extendedSearch = false;
|
|
|
|
|
searchString.clear();
|
|
|
|
|
}
|
|
|
|
|
} catch (...) {
|
|
|
|
|
resultTitle.clear();
|
|
|
|
|
resultExtract.clear();
|
|
|
|
|
extendedSearch = false;
|
|
|
|
|
searchProcess = { };
|
|
|
|
|
|
|
|
|
|
resultTitle = "???";
|
2021-08-28 18:15:47 +02:00
|
|
|
|
resultExtract = "hex.builtin.tools.wiki_explain.invalid_response"_lang.get();
|
2021-08-28 16:02:53 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-22 18:01:39 +01:00
|
|
|
|
void registerToolEntries() {
|
2021-02-13 15:15:32 +01:00
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.demangler", drawDemangler);
|
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.ascii_table", drawASCIITable);
|
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.regex_replacer", drawRegexReplacer);
|
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.color", drawColorPicker);
|
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.calc", drawMathEvaluator);
|
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.base_converter", drawBaseConverter);
|
2021-07-27 22:46:37 +02:00
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.permissions", drawPermissionsCalculator);
|
2021-08-28 00:45:59 +02:00
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.file_uploader", drawFileUploader);
|
2021-08-28 16:02:53 +02:00
|
|
|
|
ContentRegistry::Tools::add("hex.builtin.tools.wiki_explain", drawWikiExplainer);
|
2021-01-22 18:01:39 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
}
|