2023-11-21 14:38:01 +01:00
|
|
|
#include <hex/api/localization_manager.hpp>
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-11-14 01:40:43 +01:00
|
|
|
#include <imgui.h>
|
2023-11-12 01:22:01 +01:00
|
|
|
#include <hex/ui/imgui_imhex_extensions.h>
|
2024-12-14 20:36:09 +01:00
|
|
|
#include <fonts/vscode_icons.hpp>
|
2023-11-14 01:40:43 +01:00
|
|
|
|
2023-11-30 11:23:12 +01:00
|
|
|
#include <numeric>
|
2023-11-12 01:22:01 +01:00
|
|
|
|
|
|
|
namespace hex::plugin::builtin {
|
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
void drawEuclidianAlgorithm() {
|
|
|
|
static u64 a, b;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
static i64 gcdResult = 0;
|
|
|
|
static i64 lcmResult = 0;
|
|
|
|
static i64 p = 0, q = 0;
|
|
|
|
static bool overflow = false;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
constexpr static auto extendedGcd = []<typename T>(T a, T b) -> std::pair<T, T> {
|
|
|
|
T x = 1, y = 0;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
T xLast = 0, yLast = 1;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
while (b > 0) {
|
|
|
|
T quotient = a / b;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
std::tie(x, xLast) = std::tuple { xLast, x - quotient * xLast };
|
|
|
|
std::tie(y, yLast) = std::tuple { yLast, y - quotient * yLast };
|
|
|
|
std::tie(a, b) = std::tuple { b, a - quotient * b };
|
|
|
|
}
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
return { x, y };
|
|
|
|
};
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
ImGuiExt::TextFormattedWrapped("{}", "hex.builtin.tools.euclidean_algorithm.description"_lang);
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
ImGui::NewLine();
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
if (ImGuiExt::BeginBox()) {
|
|
|
|
bool hasChanged = false;
|
|
|
|
hasChanged = ImGui::InputScalar("A", ImGuiDataType_U64, &a) || hasChanged;
|
|
|
|
hasChanged = ImGui::InputScalar("B", ImGuiDataType_U64, &b) || hasChanged;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
// Update results when input changed
|
|
|
|
if (hasChanged) {
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
// Detect overflow
|
|
|
|
const u64 multiplicationResult = a * b;
|
|
|
|
if (a != 0 && multiplicationResult / a != b) {
|
|
|
|
gcdResult = 0;
|
|
|
|
lcmResult = 0;
|
|
|
|
p = 0;
|
|
|
|
q = 0;
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
overflow = true;
|
|
|
|
} else {
|
|
|
|
gcdResult = std::gcd<i128, i128>(a, b);
|
|
|
|
lcmResult = std::lcm<i128, i128>(a, b);
|
|
|
|
std::tie(p, q) = extendedGcd(a, b);
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
overflow = false;
|
|
|
|
}
|
2023-11-12 01:22:01 +01:00
|
|
|
}
|
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
ImGui::Separator();
|
|
|
|
|
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_Alpha, ImGui::GetStyle().DisabledAlpha);
|
|
|
|
ImGui::InputScalar("gcd(A, B)", ImGuiDataType_S64, &gcdResult, nullptr, nullptr, "%llu", ImGuiInputTextFlags_ReadOnly);
|
2023-11-12 01:22:01 +01:00
|
|
|
|
2023-12-01 13:53:51 +01:00
|
|
|
ImGui::Indent();
|
|
|
|
ImGuiExt::TextFormatted(ICON_VS_ARROW_RIGHT " a \u00D7 p + b \u00D7 q = ({0}) \u00D7 ({1}) + ({2}) \u00D7 ({3})", a, p, b, q);
|
|
|
|
ImGui::Unindent();
|
|
|
|
|
|
|
|
ImGui::InputScalar("lcm(A, B)", ImGuiDataType_S64, &lcmResult, nullptr, nullptr, "%llu", ImGuiInputTextFlags_ReadOnly);
|
|
|
|
ImGui::PopStyleVar();
|
|
|
|
|
|
|
|
ImGuiExt::EndBox();
|
2023-11-12 01:22:01 +01:00
|
|
|
}
|
2023-12-01 13:53:51 +01:00
|
|
|
|
|
|
|
if (overflow)
|
|
|
|
ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_ToolbarRed), "{}", "hex.builtin.tools.euclidean_algorithm.overflow"_lang);
|
|
|
|
else
|
|
|
|
ImGui::NewLine();
|
|
|
|
|
2023-11-12 01:22:01 +01:00
|
|
|
}
|
2023-12-01 13:53:51 +01:00
|
|
|
|
2023-11-12 01:22:01 +01:00
|
|
|
}
|