1
0
mirror of synced 2024-11-16 03:53:22 +01:00
ImHex/plugins/builtin/source/content/views/view_strings.cpp

258 lines
12 KiB
C++
Raw Normal View History

2021-12-07 22:47:41 +01:00
#include "content/views/view_strings.hpp"
2020-11-15 01:42:43 +01:00
#include <hex/providers/provider.hpp>
#include <hex/helpers/fmt.hpp>
2020-11-15 01:42:43 +01:00
#include <cstring>
2021-02-22 10:16:58 +01:00
#include <thread>
#include <regex>
2020-11-15 01:42:43 +01:00
#include <llvm/Demangle/Demangle.h>
#include <hex/ui/imgui_imhex_extensions.h>
2022-03-25 21:08:38 +01:00
#include <imgui_internal.h>
using namespace std::literals::string_literals;
2021-12-07 22:47:41 +01:00
namespace hex::plugin::builtin {
2020-11-15 01:42:43 +01:00
2021-12-07 22:47:41 +01:00
ViewStrings::ViewStrings() : View("hex.builtin.view.strings.name") {
EventManager::subscribe<EventDataChanged>(this, [this]() {
this->m_foundStrings.clear();
2020-11-15 01:42:43 +01:00
});
this->m_filter.reserve(0xFFFF);
std::memset(this->m_filter.data(), 0x00, this->m_filter.capacity());
EventManager::subscribe<EventFileUnloaded>(this, [this] {
this->m_foundStrings.clear();
});
2020-11-15 01:42:43 +01:00
}
ViewStrings::~ViewStrings() {
EventManager::unsubscribe<EventDataChanged>(this);
EventManager::unsubscribe<EventFileUnloaded>(this);
2020-11-15 01:42:43 +01:00
}
std::string readString(const FoundString &foundString) {
std::string string(foundString.size + 1, '\0');
ImHexApi::Provider::get()->read(foundString.offset, string.data(), foundString.size);
return string;
}
void ViewStrings::createStringContextMenu(const FoundString &foundString) {
if (ImGui::TableGetColumnFlags(2) == ImGuiTableColumnFlags_IsHovered && ImGui::IsMouseReleased(1) && ImGui::IsItemHovered()) {
ImGui::OpenPopup("StringContextMenu");
this->m_selectedString = readString(foundString);
}
if (ImGui::BeginPopup("StringContextMenu")) {
2021-12-07 22:47:41 +01:00
if (ImGui::MenuItem("hex.builtin.view.strings.copy"_lang)) {
ImGui::SetClipboardText(this->m_selectedString.c_str());
}
ImGui::Separator();
2021-12-07 22:47:41 +01:00
if (ImGui::MenuItem("hex.builtin.view.strings.demangle"_lang)) {
this->m_demangledName = llvm::demangle(this->m_selectedString);
if (!this->m_demangledName.empty())
ImHexApi::Tasks::doLater([] { ImGui::OpenPopup("hex.builtin.view.strings.demangle.name"_lang); });
}
ImGui::EndPopup();
}
}
2021-02-22 10:16:58 +01:00
void ViewStrings::searchStrings() {
this->m_foundStrings.clear();
this->m_filterIndices.clear();
2021-02-22 10:16:58 +01:00
this->m_searching = true;
2021-02-22 10:16:58 +01:00
std::thread([this] {
auto provider = ImHexApi::Provider::get();
2022-02-01 22:09:44 +01:00
auto task = ImHexApi::Tasks::createTask("hex.builtin.view.strings.searching", provider->getActualSize());
2020-11-15 01:42:43 +01:00
std::vector<u8> buffer(1024, 0x00);
u32 foundCharacters = 0;
for (u64 offset = 0; offset < provider->getActualSize(); offset += buffer.size()) {
task.update(offset);
size_t readSize = std::min(u64(buffer.size()), provider->getActualSize() - offset);
provider->read(offset + provider->getBaseAddress(), buffer.data(), readSize);
2020-11-15 01:42:43 +01:00
for (u32 i = 0; i < readSize; i++) {
if (buffer[i] >= ' ' && buffer[i] <= '~' && offset < provider->getActualSize() - 1)
2020-11-15 01:42:43 +01:00
foundCharacters++;
else {
if (foundCharacters >= static_cast<u32>(this->m_minimumLength)) {
FoundString foundString = {
offset + i - foundCharacters + provider->getBaseAddress(),
foundCharacters
};
2020-11-15 01:42:43 +01:00
this->m_foundStrings.push_back(foundString);
this->m_filterIndices.push_back(this->m_foundStrings.size() - 1);
2020-11-15 01:42:43 +01:00
}
foundCharacters = 0;
}
}
}
2021-02-22 10:16:58 +01:00
this->m_searching = false;
}).detach();
}
void ViewStrings::drawContent() {
auto provider = ImHexApi::Provider::get();
2021-02-22 10:16:58 +01:00
2021-12-07 22:47:41 +01:00
if (ImGui::Begin(View::toWindowName("hex.builtin.view.strings.name").c_str(), &this->getWindowOpenState(), ImGuiWindowFlags_NoCollapse)) {
if (ImHexApi::Provider::isValid() && provider->isReadable()) {
ImGui::Disabled([this] {
2021-12-07 22:47:41 +01:00
if (ImGui::InputInt("hex.builtin.view.strings.min_length"_lang, &this->m_minimumLength, 1, 0))
2021-02-22 10:16:58 +01:00
this->m_foundStrings.clear();
2022-03-25 21:08:38 +01:00
if (this->m_minimumLength < 1)
this->m_minimumLength = 1;
ImGui::Checkbox("Regex", &this->m_regex);
2022-03-25 21:08:38 +01:00
bool filterError = this->m_regex && !this->m_pattern_parsed;
if (filterError)
ImGui::PushStyleColor(ImGuiCol_FrameBg, ImAlphaBlendColors(ImGui::GetColorU32(ImGuiCol_FrameBg), ImColor(1.0f, 0.2f, 0.2f, 0.5f)));
ImGui::InputText(
2022-03-25 21:08:38 +01:00
"hex.builtin.view.strings.filter"_lang, this->m_filter.data(), this->m_filter.size(), ImGuiInputTextFlags_CallbackEdit | ImGuiInputTextFlags_CallbackResize, [](ImGuiInputTextCallbackData *data) {
auto &view = *static_cast<ViewStrings *>(data->UserData);
2022-03-25 21:08:38 +01:00
if (data->EventFlag == ImGuiInputTextFlags_CallbackResize) {
view.m_filter.resize(data->BufSize);
data->Buf = view.m_filter.data();
}
2022-03-25 21:08:38 +01:00
if (data->EventFlag == ImGuiInputTextFlags_CallbackEdit) {
view.m_filterIndices.clear();
std::regex pattern;
if (view.m_regex) {
try {
pattern = std::regex(data->Buf);
view.m_pattern_parsed = true;
} catch (std::regex_error &e) {
view.m_pattern_parsed = false;
}
}
for (u64 i = 0; i < view.m_foundStrings.size(); i++) {
if (view.m_regex) {
if (view.m_pattern_parsed && std::regex_search(readString(view.m_foundStrings[i]), pattern))
view.m_filterIndices.push_back(i);
} else if (readString(view.m_foundStrings[i]).find(data->Buf) != std::string::npos) {
view.m_filterIndices.push_back(i);
}
}
}
2022-03-25 21:08:38 +01:00
return 0;
},
this);
2022-03-25 21:08:38 +01:00
if (filterError)
ImGui::PopStyleColor();
2021-12-07 22:47:41 +01:00
if (ImGui::Button("hex.builtin.view.strings.extract"_lang))
2021-02-22 10:16:58 +01:00
this->searchStrings();
},
2022-02-01 22:09:44 +01:00
this->m_searching);
2021-02-22 10:16:58 +01:00
if (this->m_searching) {
ImGui::SameLine();
2021-12-07 22:47:41 +01:00
ImGui::TextSpinner("hex.builtin.view.strings.searching"_lang);
} else if (!this->m_foundStrings.empty()) {
ImGui::SameLine();
ImGui::TextFormatted("hex.builtin.view.strings.results"_lang, this->m_filterIndices.size());
}
2020-11-15 03:51:59 +01:00
ImGui::Separator();
ImGui::NewLine();
if (ImGui::BeginTable("##strings", 3, ImGuiTableFlags_Borders | ImGuiTableFlags_Resizable | ImGuiTableFlags_Sortable | ImGuiTableFlags_Reorderable | ImGuiTableFlags_RowBg | ImGuiTableFlags_ScrollY)) {
ImGui::TableSetupScrollFreeze(0, 1);
2021-12-07 22:47:41 +01:00
ImGui::TableSetupColumn("hex.builtin.view.strings.offset"_lang, 0, -1, ImGui::GetID("offset"));
ImGui::TableSetupColumn("hex.builtin.view.strings.size"_lang, 0, -1, ImGui::GetID("size"));
ImGui::TableSetupColumn("hex.builtin.view.strings.string"_lang, 0, -1, ImGui::GetID("string"));
2020-11-15 03:51:59 +01:00
auto sortSpecs = ImGui::TableGetSortSpecs();
if (sortSpecs->SpecsDirty) {
std::sort(this->m_foundStrings.begin(), this->m_foundStrings.end(), [&sortSpecs](FoundString &left, FoundString &right) -> bool {
if (sortSpecs->Specs->ColumnUserID == ImGui::GetID("offset")) {
if (sortSpecs->Specs->SortDirection == ImGuiSortDirection_Ascending)
return left.offset > right.offset;
else
return left.offset < right.offset;
} else if (sortSpecs->Specs->ColumnUserID == ImGui::GetID("size")) {
if (sortSpecs->Specs->SortDirection == ImGuiSortDirection_Ascending)
return left.size > right.size;
else
return left.size < right.size;
} else if (sortSpecs->Specs->ColumnUserID == ImGui::GetID("string")) {
if (sortSpecs->Specs->SortDirection == ImGuiSortDirection_Ascending)
return readString(left) > readString(right);
else
return readString(left) < readString(right);
}
return false;
});
2020-11-15 03:51:59 +01:00
sortSpecs->SpecsDirty = false;
}
2020-11-15 01:42:43 +01:00
2020-11-15 03:51:59 +01:00
ImGui::TableHeadersRow();
ImGuiListClipper clipper;
clipper.Begin(this->m_filterIndices.size());
2020-11-20 11:56:37 +01:00
while (clipper.Step()) {
for (int i = clipper.DisplayStart; i < clipper.DisplayEnd; i++) {
auto &foundString = this->m_foundStrings[this->m_filterIndices[i]];
2022-02-01 22:09:44 +01:00
auto string = readString(foundString);
2020-11-20 11:56:37 +01:00
2020-11-23 13:08:24 +01:00
ImGui::TableNextRow();
2020-11-20 11:56:37 +01:00
ImGui::TableNextColumn();
if (ImGui::Selectable(("##StringLine"s + std::to_string(i)).c_str(), false, ImGuiSelectableFlags_SpanAllColumns)) {
2022-02-08 18:38:54 +01:00
ImHexApi::HexEditor::setSelection(foundString.offset, foundString.size);
}
ImGui::PushID(i + 1);
createStringContextMenu(foundString);
ImGui::PopID();
ImGui::SameLine();
ImGui::TextFormatted("0x{0:08X} : 0x{1:08X}", foundString.offset, foundString.offset + foundString.size);
2020-11-20 11:56:37 +01:00
ImGui::TableNextColumn();
ImGui::TextFormatted("0x{0:04X}", foundString.size);
2020-11-20 11:56:37 +01:00
ImGui::TableNextColumn();
ImGui::TextUnformatted(string.c_str());
2020-11-20 11:56:37 +01:00
}
2020-11-15 03:51:59 +01:00
}
clipper.End();
2020-11-15 01:42:43 +01:00
2020-11-15 03:51:59 +01:00
ImGui::EndTable();
2020-11-15 01:42:43 +01:00
}
}
}
ImGui::End();
2021-12-07 22:47:41 +01:00
if (ImGui::BeginPopup("hex.builtin.view.strings.demangle.title"_lang)) {
if (ImGui::BeginChild("##scrolling", ImVec2(500, 150))) {
2021-12-07 22:47:41 +01:00
ImGui::TextUnformatted("hex.builtin.view.strings.demangle.title"_lang);
ImGui::Separator();
2022-01-15 14:14:53 +01:00
ImGui::TextFormattedWrapped("{}", this->m_demangledName.c_str());
ImGui::NewLine();
2021-12-07 22:47:41 +01:00
if (ImGui::Button("hex.builtin.view.strings.demangle.copy"_lang))
ImGui::SetClipboardText(this->m_demangledName.c_str());
}
ImGui::EndChild();
ImGui::EndPopup();
}
2020-11-15 01:42:43 +01:00
}
}