2021-12-07 22:47:41 +01:00
|
|
|
#include "content/views/view_diff.hpp"
|
2021-09-21 02:48:41 +02:00
|
|
|
|
2021-09-21 19:54:13 +02:00
|
|
|
#include <hex/api/imhex_api.hpp>
|
2021-09-21 02:48:41 +02:00
|
|
|
|
2021-09-21 19:54:13 +02:00
|
|
|
#include <hex/helpers/fmt.hpp>
|
2023-11-10 20:47:08 +01:00
|
|
|
#include <hex/providers/buffered_reader.hpp>
|
2024-01-28 22:14:59 +01:00
|
|
|
|
2024-12-14 20:36:09 +01:00
|
|
|
#include <fonts/vscode_icons.hpp>
|
2024-01-21 18:39:13 +01:00
|
|
|
#include <wolv/utils/guards.hpp>
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
namespace hex::plugin::diffing {
|
2021-09-21 02:48:41 +02:00
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
using DifferenceType = ContentRegistry::Diffing::DifferenceType;
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
ViewDiff::ViewDiff() : View::Window("hex.diffing.view.diff.name", ICON_VS_DIFF_SIDEBYSIDE) {
|
2023-08-29 12:14:12 +02:00
|
|
|
// Clear the selected diff providers when a provider is closed
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderClosed::subscribe(this, [this](prv::Provider *) {
|
2024-02-11 19:29:02 +01:00
|
|
|
this->reset();
|
|
|
|
});
|
|
|
|
EventDataChanged::subscribe(this, [this](prv::Provider *) {
|
|
|
|
m_analyzed = false;
|
2023-02-15 17:01:36 +01:00
|
|
|
});
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-08-29 12:14:12 +02:00
|
|
|
// Set the background highlight callbacks for the two hex editor columns
|
2023-12-19 13:10:25 +01:00
|
|
|
m_columns[0].hexEditor.setBackgroundHighlightCallback(this->createCompareFunction(1));
|
|
|
|
m_columns[1].hexEditor.setBackgroundHighlightCallback(this->createCompareFunction(0));
|
2021-09-21 19:54:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ViewDiff::~ViewDiff() {
|
2023-12-08 10:29:44 +01:00
|
|
|
EventProviderClosed::unsubscribe(this);
|
2024-02-11 19:29:02 +01:00
|
|
|
EventDataChanged::unsubscribe(this);
|
2021-09-21 19:54:13 +02:00
|
|
|
}
|
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
namespace {
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
bool drawDiffColumn(ViewDiff::Column &column, float height) {
|
2024-01-21 18:39:13 +01:00
|
|
|
if (height < 0)
|
|
|
|
return false;
|
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
bool scrolled = false;
|
|
|
|
ImGui::PushID(&column);
|
2024-01-21 18:39:13 +01:00
|
|
|
ON_SCOPE_EXIT { ImGui::PopID(); };
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-08-29 12:14:12 +02:00
|
|
|
// Draw the hex editor
|
2023-02-16 08:53:05 +01:00
|
|
|
float prevScroll = column.hexEditor.getScrollPosition();
|
|
|
|
column.hexEditor.draw(height);
|
|
|
|
float currScroll = column.hexEditor.getScrollPosition();
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-08-29 12:14:12 +02:00
|
|
|
// Check if the user scrolled the hex editor
|
2023-02-16 08:53:05 +01:00
|
|
|
if (prevScroll != currScroll) {
|
|
|
|
scrolled = true;
|
|
|
|
column.scrollLock = 5;
|
|
|
|
}
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
return scrolled;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool drawProviderSelector(ViewDiff::Column &column) {
|
|
|
|
bool shouldReanalyze = false;
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
ImGui::PushID(&column);
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2024-02-10 23:31:05 +01:00
|
|
|
auto providers = ImHexApi::Provider::getProviders();
|
2023-02-16 08:53:05 +01:00
|
|
|
auto &providerIndex = column.provider;
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-08-29 12:14:12 +02:00
|
|
|
// Get the name of the currently selected provider
|
2023-02-16 08:53:05 +01:00
|
|
|
std::string preview;
|
|
|
|
if (ImHexApi::Provider::isValid() && providerIndex >= 0)
|
|
|
|
preview = providers[providerIndex]->getName();
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-08-29 12:14:12 +02:00
|
|
|
// Draw combobox with all available providers
|
2023-02-16 08:53:05 +01:00
|
|
|
ImGui::SetNextItemWidth(ImGui::GetContentRegionAvail().x);
|
|
|
|
if (ImGui::BeginCombo("", preview.c_str())) {
|
|
|
|
|
|
|
|
for (size_t i = 0; i < providers.size(); i++) {
|
2023-04-13 15:03:14 +02:00
|
|
|
ImGui::PushID(i + 1);
|
2023-02-16 08:53:05 +01:00
|
|
|
if (ImGui::Selectable(providers[i]->getName().c_str())) {
|
|
|
|
providerIndex = i;
|
|
|
|
shouldReanalyze = true;
|
|
|
|
}
|
2023-04-13 15:03:14 +02:00
|
|
|
ImGui::PopID();
|
2021-09-21 19:54:13 +02:00
|
|
|
}
|
2023-02-16 08:53:05 +01:00
|
|
|
|
|
|
|
ImGui::EndCombo();
|
2021-09-21 19:54:13 +02:00
|
|
|
}
|
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
ImGui::PopID();
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-02-16 08:53:05 +01:00
|
|
|
return shouldReanalyze;
|
|
|
|
}
|
2021-09-21 19:54:13 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
void ViewDiff::analyze(prv::Provider *providerA, prv::Provider *providerB) {
|
2024-02-11 19:29:02 +01:00
|
|
|
auto commonSize = std::max(providerA->getActualSize(), providerB->getActualSize());
|
2024-07-27 16:29:06 +02:00
|
|
|
m_diffTask = TaskManager::createTask("hex.diffing.view.diff.task.diffing"_lang, commonSize, [this, providerA, providerB](Task &) {
|
2024-01-21 18:39:13 +01:00
|
|
|
auto differences = m_algorithm->analyze(providerA, providerB);
|
2023-08-29 12:14:12 +02:00
|
|
|
|
2024-02-11 20:44:21 +01:00
|
|
|
auto providers = ImHexApi::Provider::getProviders();
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
// Move the calculated differences over so they can be displayed
|
|
|
|
for (size_t i = 0; i < m_columns.size(); i++) {
|
|
|
|
auto &column = m_columns[i];
|
2024-02-11 20:44:21 +01:00
|
|
|
auto &provider = providers[column.provider];
|
2024-02-11 19:29:02 +01:00
|
|
|
|
|
|
|
column.differences = differences[i].overlapping({ provider->getBaseAddress(), provider->getBaseAddress() + provider->getActualSize() });
|
|
|
|
std::ranges::sort(
|
|
|
|
column.differences,
|
|
|
|
std::less(),
|
|
|
|
[](const auto &a) { return a.interval; }
|
|
|
|
);
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
column.diffTree = std::move(differences[i]);
|
2023-08-29 12:14:12 +02:00
|
|
|
}
|
2024-01-21 18:39:13 +01:00
|
|
|
m_analyzed = true;
|
|
|
|
});
|
2023-08-29 12:14:12 +02:00
|
|
|
}
|
|
|
|
|
2024-02-11 19:29:02 +01:00
|
|
|
void ViewDiff::reset() {
|
|
|
|
for (auto &column : m_columns) {
|
|
|
|
column.provider = -1;
|
|
|
|
column.hexEditor.setSelectionUnchecked(std::nullopt, std::nullopt);
|
|
|
|
column.diffTree.clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
std::function<std::optional<color_t>(u64, const u8*, size_t)> ViewDiff::createCompareFunction(size_t otherIndex) const {
|
|
|
|
const auto currIndex = otherIndex == 0 ? 1 : 0;
|
|
|
|
return [=, this](u64 address, const u8 *, size_t size) -> std::optional<color_t> {
|
|
|
|
if (!m_analyzed)
|
|
|
|
return std::nullopt;
|
2023-08-29 12:14:12 +02:00
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
const auto matches = m_columns[currIndex].diffTree.overlapping({ address, (address + size) - 1 });
|
|
|
|
if (matches.empty())
|
|
|
|
return std::nullopt;
|
2023-08-29 12:14:12 +02:00
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
const auto type = matches[0].value;
|
2023-08-29 12:14:12 +02:00
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
if (type == DifferenceType::Mismatch) {
|
|
|
|
return ImGuiExt::GetCustomColorU32(ImGuiCustomCol_DiffChanged);
|
|
|
|
} else if (type == DifferenceType::Insertion && currIndex == 0) {
|
|
|
|
return ImGuiExt::GetCustomColorU32(ImGuiCustomCol_DiffAdded);
|
|
|
|
} else if (type == DifferenceType::Deletion && currIndex == 1) {
|
|
|
|
return ImGuiExt::GetCustomColorU32(ImGuiCustomCol_DiffRemoved);
|
2023-08-29 12:14:12 +02:00
|
|
|
}
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
return std::nullopt;
|
|
|
|
};
|
2023-08-29 12:14:12 +02:00
|
|
|
}
|
|
|
|
|
2024-12-31 00:47:32 +01:00
|
|
|
static void drawByteString(const std::vector<u8> &bytes) {
|
|
|
|
for (u64 i = 0; i < bytes.size(); i += 1) {
|
|
|
|
if (i >= 16) {
|
|
|
|
ImGui::TextDisabled(ICON_VS_ELLIPSIS);
|
|
|
|
ImGui::SameLine(0, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
u8 byte = bytes[i];
|
|
|
|
ImGuiExt::TextFormattedDisabled("{0:02X} ", byte);
|
|
|
|
ImGui::SameLine(0, (i % 4 == 3) ? 4_scaled : 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-15 17:01:36 +01:00
|
|
|
void ViewDiff::drawContent() {
|
2023-12-19 13:10:25 +01:00
|
|
|
auto &[a, b] = m_columns;
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
a.hexEditor.enableSyncScrolling(false);
|
|
|
|
b.hexEditor.enableSyncScrolling(false);
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
if (a.scrollLock > 0) a.scrollLock--;
|
|
|
|
if (b.scrollLock > 0) b.scrollLock--;
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Change the hex editor providers if the user selected a new provider
|
|
|
|
{
|
|
|
|
const auto &providers = ImHexApi::Provider::getProviders();
|
|
|
|
if (a.provider >= 0 && size_t(a.provider) < providers.size())
|
|
|
|
a.hexEditor.setProvider(providers[a.provider]);
|
|
|
|
else
|
|
|
|
a.hexEditor.setProvider(nullptr);
|
|
|
|
|
|
|
|
if (b.provider >= 0 && size_t(b.provider) < providers.size())
|
|
|
|
b.hexEditor.setProvider(providers[b.provider]);
|
|
|
|
else
|
|
|
|
b.hexEditor.setProvider(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Analyze the providers if they are valid and the user selected a new provider
|
2024-01-21 18:39:13 +01:00
|
|
|
if (!m_analyzed && a.provider != -1 && b.provider != -1 && !m_diffTask.isRunning() && m_algorithm != nullptr) {
|
2023-11-21 13:47:50 +01:00
|
|
|
const auto &providers = ImHexApi::Provider::getProviders();
|
|
|
|
auto providerA = providers[a.provider];
|
|
|
|
auto providerB = providers[b.provider];
|
|
|
|
|
|
|
|
this->analyze(providerA, providerB);
|
|
|
|
}
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
if (auto &algorithms = ContentRegistry::Diffing::impl::getAlgorithms(); m_algorithm == nullptr && !algorithms.empty())
|
|
|
|
m_algorithm = algorithms.front().get();
|
|
|
|
|
2024-07-27 16:51:43 +02:00
|
|
|
static float height = 0;
|
|
|
|
static bool dragging = false;
|
|
|
|
|
|
|
|
const auto availableSize = ImGui::GetContentRegionAvail();
|
|
|
|
auto diffingColumnSize = availableSize;
|
|
|
|
diffingColumnSize.y *= 3.5 / 5.0;
|
|
|
|
diffingColumnSize.y -= ImGui::GetTextLineHeightWithSpacing();
|
|
|
|
diffingColumnSize.y += height;
|
|
|
|
|
|
|
|
if (availableSize.y > 1)
|
|
|
|
diffingColumnSize.y = std::clamp(diffingColumnSize.y, 1.0F, std::max(1.0F, availableSize.y - ImGui::GetTextLineHeightWithSpacing() * 3));
|
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
|
|
|
|
// Draw the two hex editor columns side by side
|
2024-07-27 16:51:43 +02:00
|
|
|
if (ImGui::BeginTable("##binary_diff", 2, ImGuiTableFlags_None, diffingColumnSize)) {
|
2024-01-21 18:39:13 +01:00
|
|
|
ImGui::TableSetupColumn("hex.diffing.view.diff.provider_a"_lang);
|
|
|
|
ImGui::TableSetupColumn("hex.diffing.view.diff.provider_b"_lang);
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::TableHeadersRow();
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-12-19 13:10:25 +01:00
|
|
|
ImGui::BeginDisabled(m_diffTask.isRunning());
|
2023-02-15 17:01:36 +01:00
|
|
|
{
|
2024-01-21 18:39:13 +01:00
|
|
|
// Draw settings button
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::TableNextColumn();
|
2024-01-21 18:39:13 +01:00
|
|
|
if (ImGuiExt::DimmedIconButton(ICON_VS_SETTINGS_GEAR, ImGui::GetStyleColorVec4(ImGuiCol_Text)))
|
2024-03-13 22:39:00 +01:00
|
|
|
RequestOpenPopup::post("##DiffingAlgorithmSettings");
|
2024-01-21 18:39:13 +01:00
|
|
|
|
|
|
|
ImGui::SameLine();
|
|
|
|
|
|
|
|
// Draw first provider selector
|
2023-12-19 13:10:25 +01:00
|
|
|
if (drawProviderSelector(a)) m_analyzed = false;
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw second provider selector
|
|
|
|
ImGui::TableNextColumn();
|
2023-12-19 13:10:25 +01:00
|
|
|
if (drawProviderSelector(b)) m_analyzed = false;
|
2023-02-15 17:01:36 +01:00
|
|
|
}
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::EndDisabled();
|
2021-09-21 02:48:41 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::TableNextRow();
|
2021-09-21 02:48:41 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw first hex editor column
|
|
|
|
ImGui::TableNextColumn();
|
2024-07-27 16:51:43 +02:00
|
|
|
bool scrollB = drawDiffColumn(a, diffingColumnSize.y);
|
2023-11-21 13:47:50 +01:00
|
|
|
|
|
|
|
// Draw second hex editor column
|
|
|
|
ImGui::TableNextColumn();
|
2024-07-27 16:51:43 +02:00
|
|
|
bool scrollA = drawDiffColumn(b, diffingColumnSize.y);
|
2023-11-21 13:47:50 +01:00
|
|
|
|
|
|
|
// Sync the scroll positions of the hex editors
|
|
|
|
{
|
|
|
|
if (scrollA && a.scrollLock == 0) {
|
|
|
|
a.hexEditor.setScrollPosition(b.hexEditor.getScrollPosition());
|
|
|
|
a.hexEditor.forceUpdateScrollPosition();
|
|
|
|
}
|
|
|
|
if (scrollB && b.scrollLock == 0) {
|
|
|
|
b.hexEditor.setScrollPosition(a.hexEditor.getScrollPosition());
|
|
|
|
b.hexEditor.forceUpdateScrollPosition();
|
|
|
|
}
|
2023-02-15 17:01:36 +01:00
|
|
|
}
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::EndTable();
|
|
|
|
}
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2024-07-27 16:51:43 +02:00
|
|
|
ImGui::Button("##table_drag_bar", ImVec2(ImGui::GetContentRegionAvail().x, 2_scaled));
|
|
|
|
if (ImGui::IsMouseDragging(ImGuiMouseButton_Left, 0)) {
|
|
|
|
if (ImGui::IsItemHovered())
|
|
|
|
dragging = true;
|
|
|
|
} else {
|
|
|
|
dragging = false;
|
|
|
|
}
|
|
|
|
if (ImGui::IsItemHovered()) {
|
|
|
|
ImGui::SetMouseCursor(ImGuiMouseCursor_ResizeNS);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (dragging) {
|
|
|
|
height += ImGui::GetMouseDragDelta(ImGuiMouseButton_Left, 0).y;
|
|
|
|
ImGui::ResetMouseDragDelta(ImGuiMouseButton_Left);
|
|
|
|
}
|
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw the differences table
|
2024-12-31 00:47:32 +01:00
|
|
|
if (ImGui::BeginTable("##differences", 4, ImGuiTableFlags_Borders | ImGuiTableFlags_ScrollY | ImGuiTableFlags_Reorderable | ImGuiTableFlags_SizingFixedFit)) {
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::TableSetupScrollFreeze(0, 1);
|
2024-12-31 00:47:32 +01:00
|
|
|
ImGui::TableSetupColumn("##Type", ImGuiTableColumnFlags_NoReorder);
|
|
|
|
ImGui::TableSetupColumn("hex.diffing.view.diff.provider_a"_lang);
|
|
|
|
ImGui::TableSetupColumn("hex.diffing.view.diff.provider_b"_lang);
|
|
|
|
ImGui::TableSetupColumn("hex.diffing.view.diff.changes"_lang);
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::TableHeadersRow();
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw the differences if the providers have been analyzed
|
2023-12-19 13:10:25 +01:00
|
|
|
if (m_analyzed) {
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGuiListClipper clipper;
|
2023-08-29 12:14:12 +02:00
|
|
|
|
2024-02-11 19:29:02 +01:00
|
|
|
auto &differencesA = m_columns[0].differences;
|
|
|
|
auto &differencesB = m_columns[1].differences;
|
|
|
|
clipper.Begin(int(std::min(differencesA.size(), differencesB.size())));
|
2024-01-21 18:39:13 +01:00
|
|
|
|
|
|
|
while (clipper.Step()) {
|
2023-11-21 13:47:50 +01:00
|
|
|
for (int i = clipper.DisplayStart; i < clipper.DisplayEnd; i++) {
|
|
|
|
ImGui::TableNextRow();
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::PushID(i);
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2024-02-11 19:29:02 +01:00
|
|
|
const auto &[regionA, typeA] = differencesA[i];
|
|
|
|
const auto &[regionB, typeB] = differencesB[i];
|
2022-03-27 00:01:28 +01:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw a clickable row for each difference that will select the difference in both hex editors
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2024-12-31 00:47:32 +01:00
|
|
|
// Draw difference type
|
|
|
|
ImGui::TableNextColumn();
|
|
|
|
switch (typeA) {
|
|
|
|
case DifferenceType::Mismatch:
|
|
|
|
ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_DiffChanged), ICON_VS_DIFF_MODIFIED);
|
|
|
|
ImGui::SetItemTooltip("hex.diffing.view.diff.modified"_lang);
|
|
|
|
break;
|
|
|
|
case DifferenceType::Insertion:
|
|
|
|
ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_DiffAdded), ICON_VS_DIFF_ADDED);
|
|
|
|
ImGui::SetItemTooltip("hex.diffing.view.diff.added"_lang);
|
|
|
|
break;
|
|
|
|
case DifferenceType::Deletion:
|
|
|
|
ImGuiExt::TextFormattedColored(ImGuiExt::GetCustomColorVec4(ImGuiCustomCol_DiffRemoved), ICON_VS_DIFF_REMOVED);
|
|
|
|
ImGui::SetItemTooltip("hex.diffing.view.diff.removed"_lang);
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw start address
|
|
|
|
ImGui::TableNextColumn();
|
2024-12-31 00:47:32 +01:00
|
|
|
if (ImGui::Selectable(hex::format("0x{:04X} - 0x{:04X}", regionA.start, regionA.end).c_str(), false, ImGuiSelectableFlags_SpanAllColumns)) {
|
2024-07-27 16:52:15 +02:00
|
|
|
const Region selectionA = { regionA.start, ((regionA.end - regionA.start) + 1) };
|
|
|
|
const Region selectionB = { regionB.start, ((regionB.end - regionB.start) + 1) };
|
|
|
|
|
|
|
|
a.hexEditor.setSelection(selectionA);
|
2023-11-21 13:47:50 +01:00
|
|
|
a.hexEditor.jumpToSelection();
|
2024-07-27 16:52:15 +02:00
|
|
|
b.hexEditor.setSelection(selectionB);
|
2023-11-21 13:47:50 +01:00
|
|
|
b.hexEditor.jumpToSelection();
|
2024-07-27 16:52:15 +02:00
|
|
|
|
|
|
|
const auto &providers = ImHexApi::Provider::getProviders();
|
|
|
|
auto openProvider = ImHexApi::Provider::get();
|
|
|
|
|
|
|
|
if (providers[a.provider] == openProvider)
|
|
|
|
ImHexApi::HexEditor::setSelection(selectionA);
|
|
|
|
else if (providers[b.provider] == openProvider)
|
|
|
|
ImHexApi::HexEditor::setSelection(selectionB);
|
2023-11-21 13:47:50 +01:00
|
|
|
}
|
2021-09-21 19:54:13 +02:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
// Draw end address
|
|
|
|
ImGui::TableNextColumn();
|
2024-12-31 00:47:32 +01:00
|
|
|
ImGui::TextUnformatted(hex::format("0x{:04X} - 0x{:04X}", regionB.start, regionB.end).c_str());
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2024-12-31 00:47:32 +01:00
|
|
|
const auto &providers = ImHexApi::Provider::getProviders();
|
|
|
|
std::vector<u8> data;
|
|
|
|
|
|
|
|
// Draw changes
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::TableNextColumn();
|
2024-12-31 00:47:32 +01:00
|
|
|
ImGui::Indent();
|
2024-01-21 18:39:13 +01:00
|
|
|
switch (typeA) {
|
|
|
|
case DifferenceType::Insertion:
|
2024-12-31 00:47:32 +01:00
|
|
|
data.resize(std::min<u64>(17, (regionA.end - regionA.start) + 1));
|
|
|
|
providers[a.provider]->read(regionA.start, data.data(), data.size());
|
|
|
|
drawByteString(data);
|
|
|
|
break;
|
|
|
|
case DifferenceType::Mismatch:
|
|
|
|
data.resize(std::min<u64>(17, (regionA.end - regionA.start) + 1));
|
|
|
|
providers[a.provider]->read(regionA.start, data.data(), data.size());
|
|
|
|
drawByteString(data);
|
|
|
|
|
|
|
|
ImGui::SameLine(0, 0);
|
|
|
|
ImGuiExt::TextFormatted(" {} ", ICON_VS_ARROW_RIGHT);
|
|
|
|
ImGui::SameLine(0, 0);
|
|
|
|
|
|
|
|
data.resize(std::min<u64>(17, (regionB.end - regionB.start) + 1));
|
|
|
|
providers[b.provider]->read(regionB.start, data.data(), data.size());
|
|
|
|
drawByteString(data);
|
2023-11-21 13:47:50 +01:00
|
|
|
break;
|
2024-01-21 18:39:13 +01:00
|
|
|
case DifferenceType::Deletion:
|
2024-12-31 00:47:32 +01:00
|
|
|
data.resize(std::min<u64>(17, (regionB.end - regionB.start) + 1));
|
|
|
|
providers[b.provider]->read(regionB.start, data.data(), data.size());
|
|
|
|
drawByteString(data);
|
2023-11-21 13:47:50 +01:00
|
|
|
break;
|
2024-01-21 18:39:13 +01:00
|
|
|
default:
|
2023-11-21 13:47:50 +01:00
|
|
|
break;
|
2021-09-21 19:54:13 +02:00
|
|
|
}
|
2024-12-31 00:47:32 +01:00
|
|
|
ImGui::Unindent();
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::PopID();
|
|
|
|
}
|
2024-01-21 18:39:13 +01:00
|
|
|
}
|
2021-09-21 19:54:13 +02:00
|
|
|
}
|
2023-02-15 17:01:36 +01:00
|
|
|
|
2023-11-21 13:47:50 +01:00
|
|
|
ImGui::EndTable();
|
2021-09-21 02:48:41 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-03-13 22:39:00 +01:00
|
|
|
void ViewDiff::drawAlwaysVisibleContent() {
|
|
|
|
ImGui::SetNextWindowSizeConstraints(ImVec2(0, 0), ImVec2(400_scaled, 600_scaled));
|
|
|
|
if (ImGui::BeginPopup("##DiffingAlgorithmSettings")) {
|
|
|
|
ImGuiExt::Header("hex.diffing.view.diff.algorithm"_lang, true);
|
|
|
|
ImGui::PushItemWidth(300_scaled);
|
|
|
|
if (ImGui::BeginCombo("##Algorithm", m_algorithm == nullptr ? "" : Lang(m_algorithm->getUnlocalizedName()))) {
|
|
|
|
for (const auto &algorithm : ContentRegistry::Diffing::impl::getAlgorithms()) {
|
|
|
|
ImGui::PushID(algorithm.get());
|
|
|
|
if (ImGui::Selectable(Lang(algorithm->getUnlocalizedName()))) {
|
|
|
|
m_algorithm = algorithm.get();
|
|
|
|
m_analyzed = false;
|
|
|
|
}
|
|
|
|
ImGui::PopID();
|
|
|
|
}
|
|
|
|
ImGui::EndCombo();
|
|
|
|
}
|
|
|
|
ImGui::PopItemWidth();
|
|
|
|
|
|
|
|
if (m_algorithm != nullptr) {
|
|
|
|
ImGuiExt::TextFormattedWrapped("{}", Lang(m_algorithm->getUnlocalizedDescription()));
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGuiExt::Header("hex.diffing.view.diff.settings"_lang);
|
|
|
|
if (m_algorithm != nullptr) {
|
|
|
|
auto drawList = ImGui::GetWindowDrawList();
|
|
|
|
auto prevIdx = drawList->_VtxCurrentIdx;
|
|
|
|
m_algorithm->drawSettings();
|
|
|
|
auto currIdx = drawList->_VtxCurrentIdx;
|
|
|
|
|
|
|
|
if (prevIdx == currIdx)
|
|
|
|
ImGuiExt::TextFormatted("hex.diffing.view.diff.settings.no_settings"_lang);
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::EndPopup();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-01-21 18:39:13 +01:00
|
|
|
}
|