mirror of https://github.com/WerWolv/ImHex.git
sys: Drastically improve highlighting performance
This commit is contained in:
parent
ba68f463e5
commit
618eead341
|
@ -104,7 +104,7 @@ namespace hex {
|
|||
EVENT_DEF(EventFileLoaded, fs::path);
|
||||
EVENT_DEF(EventFileUnloaded);
|
||||
EVENT_DEF(EventDataChanged);
|
||||
EVENT_DEF(EventPatternChanged, std::vector<pl::PatternData *> &);
|
||||
EVENT_DEF(EventHighlightingChanged);
|
||||
EVENT_DEF(EventWindowClosing, GLFWwindow *);
|
||||
EVENT_DEF(EventRegionSelected, Region);
|
||||
EVENT_DEF(EventProjectFileStore);
|
||||
|
|
|
@ -39,11 +39,15 @@ namespace hex {
|
|||
id, Highlighting {region, color, tooltip}
|
||||
});
|
||||
|
||||
EventManager::post<EventHighlightingChanged>();
|
||||
|
||||
return id;
|
||||
}
|
||||
|
||||
void removeHighlight(u32 id) {
|
||||
s_highlights.erase(id);
|
||||
|
||||
EventManager::post<EventHighlightingChanged>();
|
||||
}
|
||||
|
||||
std::map<u32, Highlighting> &getHighlights() {
|
||||
|
|
|
@ -14,12 +14,22 @@ namespace hex::prv {
|
|||
class Provider;
|
||||
}
|
||||
|
||||
|
||||
namespace hex::plugin::builtin {
|
||||
|
||||
|
||||
using SearchFunction = std::vector<std::pair<u64, u64>> (*)(prv::Provider *&provider, std::string string);
|
||||
|
||||
struct HighlightBlock {
|
||||
struct Highlight {
|
||||
color_t color;
|
||||
std::vector<std::string> tooltips;
|
||||
};
|
||||
|
||||
constexpr static size_t Size = 0x2000;
|
||||
|
||||
u64 base;
|
||||
std::array<Highlight, Size> highlight;
|
||||
};
|
||||
|
||||
class ViewHexEditor : public View {
|
||||
public:
|
||||
ViewHexEditor();
|
||||
|
@ -54,6 +64,8 @@ namespace hex::plugin::builtin {
|
|||
hex::EncodingFile m_currEncodingFile;
|
||||
u8 m_highlightAlpha = 0x80;
|
||||
|
||||
std::list<HighlightBlock> m_highlights;
|
||||
|
||||
bool m_processingImportExport = false;
|
||||
bool m_advancedDecodingEnabled = false;
|
||||
|
||||
|
|
|
@ -59,45 +59,72 @@ namespace hex::plugin::builtin {
|
|||
};
|
||||
|
||||
this->m_memoryEditor.HighlightFn = [](const ImU8 *data, size_t off, bool next) -> bool {
|
||||
auto _this = (ViewHexEditor *)(data);
|
||||
|
||||
std::optional<u32> currColor, prevColor;
|
||||
bool firstByte = off == 0x00;
|
||||
|
||||
auto _this = (ViewHexEditor *)(data);
|
||||
auto provider = ImHexApi::Provider::get();
|
||||
|
||||
off += provider->getBaseAddress() + provider->getCurrentPageAddress();
|
||||
|
||||
u32 alpha = static_cast<u32>(_this->m_highlightAlpha) << 24;
|
||||
|
||||
for (const auto &[id, highlight] : ImHexApi::HexEditor::getHighlights()) {
|
||||
auto ®ion = highlight.getRegion();
|
||||
auto &color = highlight.getColor();
|
||||
std::optional<color_t> currColor, prevColor;
|
||||
|
||||
if (off >= region.address && off < (region.address + region.size))
|
||||
currColor = (color & 0x00FFFFFF) | alpha;
|
||||
if ((off - 1) >= region.address && (off - 1) < (region.address + region.size))
|
||||
prevColor = (color & 0x00FFFFFF) | alpha;
|
||||
for (auto &highlightBlock : _this->m_highlights) {
|
||||
if (off >= highlightBlock.base && off < (highlightBlock.base + HighlightBlock::Size)) {
|
||||
currColor = highlightBlock.highlight[off - highlightBlock.base].color;
|
||||
}
|
||||
if ((off - 1) >= highlightBlock.base && (off - 1) < (highlightBlock.base + HighlightBlock::Size)) {
|
||||
prevColor = highlightBlock.highlight[(off - 1) - highlightBlock.base].color;
|
||||
}
|
||||
|
||||
if (currColor && prevColor) break;
|
||||
|
||||
if (firstByte)
|
||||
prevColor = 0x00;
|
||||
}
|
||||
|
||||
{
|
||||
auto patterns = provider->getPatternLanguageRuntime().getPatterns();
|
||||
for (const auto &pattern : patterns) {
|
||||
auto child = pattern->getPattern(off);
|
||||
if (child != nullptr) {
|
||||
auto color = (child->getColor() & 0x00FFFFFF) | alpha;
|
||||
currColor = currColor.has_value() ? ImAlphaBlendColors(color, currColor.value()) : color;
|
||||
break;
|
||||
if (!currColor || !prevColor) {
|
||||
if (_this->m_highlights.size() > 0x10)
|
||||
_this->m_highlights.pop_front();
|
||||
|
||||
auto blockStartOffset = off - (off % HighlightBlock::Size);
|
||||
HighlightBlock newBlock = {
|
||||
blockStartOffset,
|
||||
{}
|
||||
};
|
||||
|
||||
for (u32 i = 0; i < HighlightBlock::Size; i++) {
|
||||
std::optional<color_t> highlightColor;
|
||||
std::string highlightTooltip;
|
||||
|
||||
for (const auto &[id, highlight] : ImHexApi::HexEditor::getHighlights()) {
|
||||
auto ®ion = highlight.getRegion();
|
||||
auto &color = highlight.getColor();
|
||||
auto &tooltip = highlight.getTooltip();
|
||||
|
||||
if (blockStartOffset + i >= region.address && blockStartOffset + i < (region.address + region.size)) {
|
||||
highlightColor = (color & 0x00FFFFFF) | alpha;
|
||||
highlightTooltip = tooltip;
|
||||
}
|
||||
}
|
||||
|
||||
auto patterns = provider->getPatternLanguageRuntime().getPatterns();
|
||||
for (const auto &pattern : patterns) {
|
||||
auto child = pattern->getPattern(blockStartOffset + i);
|
||||
if (child != nullptr) {
|
||||
auto color = (child->getColor() & 0x00FFFFFF) | alpha;
|
||||
highlightColor = highlightColor.has_value() ? ImAlphaBlendColors(color, highlightColor.value()) : color;
|
||||
break;
|
||||
}
|
||||
}
|
||||
|
||||
auto &currHighlight = newBlock.highlight[i];
|
||||
currHighlight.color = highlightColor.value_or(0x00);
|
||||
if (!highlightTooltip.empty())
|
||||
currHighlight.tooltips.push_back(highlightTooltip);
|
||||
}
|
||||
|
||||
for (const auto &pattern : patterns) {
|
||||
auto child = pattern->getPattern(off - 1);
|
||||
if (child != nullptr) {
|
||||
auto color = (child->getColor() & 0x00FFFFFF) | alpha;
|
||||
prevColor = prevColor.has_value() ? ImAlphaBlendColors(color, currColor.value()) : color;
|
||||
break;
|
||||
}
|
||||
}
|
||||
_this->m_highlights.push_back(std::move(newBlock));
|
||||
}
|
||||
|
||||
if (next && prevColor != currColor) {
|
||||
|
@ -114,24 +141,26 @@ namespace hex::plugin::builtin {
|
|||
};
|
||||
|
||||
this->m_memoryEditor.HoverFn = [](const ImU8 *data, size_t off) {
|
||||
auto _this = (ViewHexEditor *)(data);
|
||||
|
||||
bool tooltipShown = false;
|
||||
|
||||
auto provider = ImHexApi::Provider::get();
|
||||
off += provider->getBaseAddress() + provider->getCurrentPageAddress();
|
||||
|
||||
for (const auto &[id, highlight] : ImHexApi::HexEditor::getHighlights()) {
|
||||
auto ®ion = highlight.getRegion();
|
||||
auto &color = highlight.getColor();
|
||||
auto &tooltip = highlight.getTooltip();
|
||||
|
||||
if (off >= region.address && off < (region.address + region.size)) {
|
||||
if (!tooltipShown && !tooltip.empty()) {
|
||||
for (auto &highlightBlock : _this->m_highlights) {
|
||||
if (off >= highlightBlock.base && off < (highlightBlock.base + HighlightBlock::Size)) {
|
||||
auto &highlight = highlightBlock.highlight[off - highlightBlock.base];
|
||||
if (!tooltipShown && !highlight.tooltips.empty()) {
|
||||
ImGui::BeginTooltip();
|
||||
tooltipShown = true;
|
||||
}
|
||||
ImGui::ColorButton(tooltip.c_str(), ImColor(color).Value);
|
||||
ImGui::SameLine(0, 10);
|
||||
ImGui::TextUnformatted(tooltip.c_str());
|
||||
|
||||
for (const auto &tooltip : highlight.tooltips) {
|
||||
ImGui::ColorButton(tooltip.c_str(), ImColor(highlight.color).Value);
|
||||
ImGui::SameLine(0, 10);
|
||||
ImGui::TextUnformatted(tooltip.c_str());
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
@ -175,6 +204,7 @@ namespace hex::plugin::builtin {
|
|||
EventManager::unsubscribe<EventWindowClosing>(this);
|
||||
EventManager::unsubscribe<RequestOpenWindow>(this);
|
||||
EventManager::unsubscribe<EventSettingsChanged>(this);
|
||||
EventManager::unsubscribe<EventHighlightingChanged>(this);
|
||||
}
|
||||
|
||||
void ViewHexEditor::drawContent() {
|
||||
|
@ -384,11 +414,7 @@ namespace hex::plugin::builtin {
|
|||
|
||||
EventManager::post<EventFileLoaded>(path);
|
||||
EventManager::post<EventDataChanged>();
|
||||
|
||||
{
|
||||
std::vector<pl::PatternData *> patterns;
|
||||
EventManager::post<EventPatternChanged>(patterns);
|
||||
}
|
||||
EventManager::post<EventHighlightingChanged>();
|
||||
}
|
||||
|
||||
void ViewHexEditor::copyBytes() const {
|
||||
|
@ -908,6 +934,10 @@ namespace hex::plugin::builtin {
|
|||
|
||||
region = Region { address, size };
|
||||
});
|
||||
|
||||
EventManager::subscribe<EventHighlightingChanged>(this, [this] {
|
||||
this->m_highlights.clear();
|
||||
});
|
||||
}
|
||||
|
||||
void ViewHexEditor::registerShortcuts() {
|
||||
|
|
|
@ -7,7 +7,7 @@ namespace hex::plugin::builtin {
|
|||
|
||||
ViewPatternData::ViewPatternData() : View("hex.builtin.view.pattern_data.name") {
|
||||
|
||||
EventManager::subscribe<EventPatternChanged>(this, [this](auto &) {
|
||||
EventManager::subscribe<EventHighlightingChanged>(this, [this]() {
|
||||
if (!ImHexApi::Provider::isValid()) return;
|
||||
|
||||
this->m_sortedPatterns[ImHexApi::Provider::get()].clear();
|
||||
|
@ -15,7 +15,7 @@ namespace hex::plugin::builtin {
|
|||
}
|
||||
|
||||
ViewPatternData::~ViewPatternData() {
|
||||
EventManager::unsubscribe<EventPatternChanged>(this);
|
||||
EventManager::unsubscribe<EventHighlightingChanged>(this);
|
||||
}
|
||||
|
||||
static bool beginPatternDataTable(prv::Provider *&provider, const std::vector<pl::PatternData *> &patterns, std::vector<pl::PatternData *> &sortedPatterns) {
|
||||
|
|
|
@ -624,10 +624,7 @@ namespace hex::plugin::builtin {
|
|||
this->m_console.clear();
|
||||
this->clearPatternData();
|
||||
|
||||
{
|
||||
std::vector<pl::PatternData *> patterns;
|
||||
EventManager::post<EventPatternChanged>(patterns);
|
||||
}
|
||||
EventManager::post<EventHighlightingChanged>();
|
||||
|
||||
std::thread([this, code] {
|
||||
std::map<std::string, pl::Token::Literal> envVars;
|
||||
|
@ -663,7 +660,7 @@ namespace hex::plugin::builtin {
|
|||
}
|
||||
|
||||
if (result) {
|
||||
EventManager::post<EventPatternChanged>(runtime.getPatterns());
|
||||
EventManager::post<EventHighlightingChanged>();
|
||||
}
|
||||
|
||||
this->m_runningEvaluators--;
|
||||
|
|
Loading…
Reference in New Issue