1
0
mirror of synced 2024-11-30 10:24:30 +01:00
ImHex/plugins/builtin/source/content/views/view_patches.cpp

120 lines
4.8 KiB
C++
Raw Normal View History

2021-12-07 22:47:41 +01:00
#include "content/views/view_patches.hpp"
2020-11-29 02:06:41 +01:00
#include <hex/providers/provider.hpp>
2020-11-29 02:06:41 +01:00
#include <hex/api/project_file_manager.hpp>
#include <nlohmann/json.hpp>
2020-11-29 02:06:41 +01:00
#include <string>
using namespace std::literals::string_literals;
2021-12-07 22:47:41 +01:00
namespace hex::plugin::builtin {
2020-11-29 02:06:41 +01:00
2021-12-07 22:47:41 +01:00
ViewPatches::ViewPatches() : View("hex.builtin.view.patches.name") {
2020-11-30 00:03:12 +01:00
ProjectFile::registerPerProviderHandler({
.basePath = "patches.json",
.required = false,
.load = [](prv::Provider *provider, const std::fs::path &basePath, Tar &tar) {
auto json = nlohmann::json::parse(tar.readString(basePath));
provider->getPatches() = json.at("patches").get<std::map<u64, u8>>();
return true;
},
.store = [](prv::Provider *provider, const std::fs::path &basePath, Tar &tar) {
nlohmann::json json;
json["patches"] = provider->getPatches();
tar.writeString(basePath, json.dump(4));
return true;
}
2020-11-30 00:03:12 +01:00
});
2020-11-29 02:06:41 +01:00
2022-09-28 15:01:43 +02:00
ImHexApi::HexEditor::addForegroundHighlightingProvider([](u64 offset, const u8* buffer, size_t, bool) -> std::optional<color_t> {
hex::unused(buffer);
2022-09-28 15:01:43 +02:00
if (!ImHexApi::Provider::isValid())
return std::nullopt;
auto provider = ImHexApi::Provider::get();
u8 byte = 0x00;
provider->read(offset, &byte, sizeof(u8), false);
const auto &patches = provider->getPatches();
if (patches.contains(offset) && patches.at(offset) != byte)
2022-09-28 15:01:43 +02:00
return ImGui::GetCustomColorU32(ImGuiCustomCol_ToolbarRed);
else
return std::nullopt;
});
2020-11-29 02:06:41 +01:00
}
void ViewPatches::drawContent() {
2021-12-07 22:47:41 +01:00
if (ImGui::Begin(View::toWindowName("hex.builtin.view.patches.name").c_str(), &this->getWindowOpenState(), ImGuiWindowFlags_NoCollapse)) {
auto provider = ImHexApi::Provider::get();
2020-11-29 02:06:41 +01:00
if (ImHexApi::Provider::isValid() && provider->isReadable()) {
2020-11-29 02:06:41 +01:00
if (ImGui::BeginTable("##patchesTable", 3, ImGuiTableFlags_Borders | ImGuiTableFlags_Resizable | ImGuiTableFlags_Sortable | ImGuiTableFlags_Reorderable | ImGuiTableFlags_RowBg | ImGuiTableFlags_ScrollY)) {
2020-11-29 02:06:41 +01:00
ImGui::TableSetupScrollFreeze(0, 1);
2021-12-07 22:47:41 +01:00
ImGui::TableSetupColumn("hex.builtin.view.patches.offset"_lang);
ImGui::TableSetupColumn("hex.builtin.view.patches.orig"_lang);
ImGui::TableSetupColumn("hex.builtin.view.patches.patch"_lang);
2020-11-29 02:06:41 +01:00
ImGui::TableHeadersRow();
auto &patches = provider->getPatches();
2022-02-01 22:09:44 +01:00
u32 index = 0;
2020-11-29 02:06:41 +01:00
ImGuiListClipper clipper;
clipper.Begin(patches.size());
while (clipper.Step()) {
auto iter = patches.begin();
for (auto i = 0; i < clipper.DisplayStart; i++)
iter++;
for (auto i = clipper.DisplayStart; i < clipper.DisplayEnd; i++) {
const auto &[address, patch] = *iter;
2020-11-29 02:06:41 +01:00
ImGui::TableNextRow();
ImGui::TableNextColumn();
2020-11-29 02:06:41 +01:00
if (ImGui::Selectable(("##patchLine" + std::to_string(index)).c_str(), false, ImGuiSelectableFlags_SpanAllColumns)) {
2022-02-08 18:38:54 +01:00
ImHexApi::HexEditor::setSelection(address, 1);
}
if (ImGui::IsMouseReleased(1) && ImGui::IsItemHovered()) {
ImGui::OpenPopup("PatchContextMenu");
this->m_selectedPatch = address;
}
ImGui::SameLine();
ImGui::TextFormatted("0x{0:08X}", address);
ImGui::TableNextColumn();
u8 previousValue = 0x00;
provider->readRaw(address, &previousValue, sizeof(u8));
ImGui::TextFormatted("0x{0:02X}", previousValue);
ImGui::TableNextColumn();
ImGui::TextFormatted("0x{0:02X}", patch);
index += 1;
iter++;
}
2020-11-29 02:06:41 +01:00
}
if (ImGui::BeginPopup("PatchContextMenu")) {
2021-12-07 22:47:41 +01:00
if (ImGui::MenuItem("hex.builtin.view.patches.remove"_lang)) {
2020-11-29 02:06:41 +01:00
patches.erase(this->m_selectedPatch);
}
ImGui::EndPopup();
}
ImGui::EndTable();
}
}
}
ImGui::End();
}
}