mirror of https://github.com/WerWolv/ImHex.git
patterns: Make updating favorites and pattern tooltips not lag out ImHex
This commit is contained in:
parent
c1babc2a55
commit
a449478e39
|
@ -1 +1 @@
|
||||||
Subproject commit b18b2e60e5c4cd6550c1d5e73fdc7e3b238dcd36
|
Subproject commit 57dc02b9d338650aaebc3627eeef6c2d7dd95738
|
|
@ -1,5 +1,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
|
#include <hex/api/task.hpp>
|
||||||
|
|
||||||
#include <pl/patterns/pattern.hpp>
|
#include <pl/patterns/pattern.hpp>
|
||||||
#include <pl/pattern_visitor.hpp>
|
#include <pl/pattern_visitor.hpp>
|
||||||
|
|
||||||
|
@ -68,8 +70,8 @@ namespace hex::plugin::builtin::ui {
|
||||||
|
|
||||||
bool isEditingPattern(const pl::ptrn::Pattern& pattern) const;
|
bool isEditingPattern(const pl::ptrn::Pattern& pattern) const;
|
||||||
void resetEditing();
|
void resetEditing();
|
||||||
bool matchesFilter(const std::vector<std::string> &filterPath, bool fullMatch);
|
bool matchesFilter(const std::vector<std::string> &filterPath, const std::vector<std::string> &patternPath, bool fullMatch);
|
||||||
void traversePatternTree(pl::ptrn::Pattern &pattern, const std::function<void(pl::ptrn::Pattern&)> &callback);
|
void traversePatternTree(pl::ptrn::Pattern &pattern, std::vector<std::string> &patternPath, const std::function<void(pl::ptrn::Pattern&)> &callback);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
std::map<const pl::ptrn::Pattern*, u64> m_displayEnd;
|
std::map<const pl::ptrn::Pattern*, u64> m_displayEnd;
|
||||||
|
@ -90,6 +92,7 @@ namespace hex::plugin::builtin::ui {
|
||||||
std::map<std::vector<std::string>, std::unique_ptr<pl::ptrn::Pattern>> m_favorites;
|
std::map<std::vector<std::string>, std::unique_ptr<pl::ptrn::Pattern>> m_favorites;
|
||||||
bool m_showFavoriteStars = false;
|
bool m_showFavoriteStars = false;
|
||||||
bool m_favoritesUpdated = false;
|
bool m_favoritesUpdated = false;
|
||||||
|
TaskHolder m_favoritesUpdateTask;
|
||||||
|
|
||||||
std::function<void(Region)> m_selectionCallback = [](Region) { };
|
std::function<void(Region)> m_selectionCallback = [](Region) { };
|
||||||
|
|
||||||
|
|
|
@ -379,6 +379,7 @@
|
||||||
"hex.builtin.pattern_drawer.tree_style.auto_expanded": "Auto Expanded Tree",
|
"hex.builtin.pattern_drawer.tree_style.auto_expanded": "Auto Expanded Tree",
|
||||||
"hex.builtin.pattern_drawer.tree_style.flattened": "Flattened",
|
"hex.builtin.pattern_drawer.tree_style.flattened": "Flattened",
|
||||||
"hex.builtin.pattern_drawer.type": "Type",
|
"hex.builtin.pattern_drawer.type": "Type",
|
||||||
|
"hex.builtin.pattern_drawer.updating": "Updating Favorites...",
|
||||||
"hex.builtin.pattern_drawer.value": "Value",
|
"hex.builtin.pattern_drawer.value": "Value",
|
||||||
"hex.builtin.pattern_drawer.var_name": "Name",
|
"hex.builtin.pattern_drawer.var_name": "Name",
|
||||||
"hex.builtin.pattern_drawer.visualizer.unknown": "Unknown visualizer",
|
"hex.builtin.pattern_drawer.visualizer.unknown": "Unknown visualizer",
|
||||||
|
|
|
@ -183,17 +183,17 @@ namespace hex::plugin::builtin::ui {
|
||||||
this->m_editingPatternOffset = 0x00;
|
this->m_editingPatternOffset = 0x00;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool PatternDrawer::matchesFilter(const std::vector<std::string> &filterPath, bool fullMatch) {
|
bool PatternDrawer::matchesFilter(const std::vector<std::string> &filterPath, const std::vector<std::string> &patternPath, bool fullMatch) {
|
||||||
if (fullMatch) {
|
if (fullMatch) {
|
||||||
if (this->m_currPatternPath.size() != filterPath.size())
|
if (patternPath.size() != filterPath.size())
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (this->m_currPatternPath.size() <= filterPath.size()) {
|
if (patternPath.size() <= filterPath.size()) {
|
||||||
for (ssize_t i = this->m_currPatternPath.size() - 1; i >= 0; i--) {
|
for (ssize_t i = patternPath.size() - 1; i >= 0; i--) {
|
||||||
const auto &filter = filterPath[i];
|
const auto &filter = filterPath[i];
|
||||||
|
|
||||||
if (this->m_currPatternPath[i] != filter && !filter.empty() && filter != "*") {
|
if (patternPath[i] != filter && !filter.empty() && filter != "*") {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -790,7 +790,7 @@ namespace hex::plugin::builtin::ui {
|
||||||
this->m_currPatternPath.push_back(pattern.getVariableName());
|
this->m_currPatternPath.push_back(pattern.getVariableName());
|
||||||
ON_SCOPE_EXIT { this->m_currPatternPath.pop_back(); };
|
ON_SCOPE_EXIT { this->m_currPatternPath.pop_back(); };
|
||||||
|
|
||||||
if (matchesFilter(this->m_filter, false))
|
if (matchesFilter(this->m_filter, this->m_currPatternPath, false))
|
||||||
pattern.accept(*this);
|
pattern.accept(*this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -994,14 +994,14 @@ namespace hex::plugin::builtin::ui {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
void PatternDrawer::traversePatternTree(pl::ptrn::Pattern &pattern, const std::function<void(pl::ptrn::Pattern&)> &callback) {
|
void PatternDrawer::traversePatternTree(pl::ptrn::Pattern &pattern, std::vector<std::string> &patternPath, const std::function<void(pl::ptrn::Pattern&)> &callback) {
|
||||||
this->m_currPatternPath.push_back(pattern.getVariableName());
|
patternPath.push_back(pattern.getVariableName());
|
||||||
ON_SCOPE_EXIT { this->m_currPatternPath.pop_back(); };
|
ON_SCOPE_EXIT { patternPath.pop_back(); };
|
||||||
|
|
||||||
callback(pattern);
|
callback(pattern);
|
||||||
if (auto iterable = dynamic_cast<pl::ptrn::IIterable*>(&pattern); iterable != nullptr) {
|
if (auto iterable = dynamic_cast<pl::ptrn::IIterable*>(&pattern); iterable != nullptr) {
|
||||||
iterable->forEachEntry(0, iterable->getEntryCount(), [&](u64, pl::ptrn::Pattern *entry) {
|
iterable->forEachEntry(0, iterable->getEntryCount(), [&](u64, pl::ptrn::Pattern *entry) {
|
||||||
traversePatternTree(*entry, callback);
|
traversePatternTree(*entry, patternPath, callback);
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1081,22 +1081,39 @@ namespace hex::plugin::builtin::ui {
|
||||||
if (!this->m_favoritesUpdated) {
|
if (!this->m_favoritesUpdated) {
|
||||||
this->m_favoritesUpdated = true;
|
this->m_favoritesUpdated = true;
|
||||||
|
|
||||||
for (auto &pattern : patterns) {
|
if (!this->m_favorites.empty() && !patterns.empty() && !this->m_favoritesUpdateTask.isRunning()) {
|
||||||
traversePatternTree(*pattern, [this](pl::ptrn::Pattern &pattern){
|
this->m_favoritesUpdateTask = TaskManager::createTask("hex.builtin.pattern_drawer.updating"_lang, TaskManager::NoProgress, [this, patterns](auto &task) {
|
||||||
for (auto &[path, favoritePattern] : this->m_favorites) {
|
size_t updatedFavorites = 0;
|
||||||
if (this->matchesFilter(path, true)) {
|
for (auto &pattern : patterns) {
|
||||||
favoritePattern = pattern.clone();
|
if (updatedFavorites == this->m_favorites.size())
|
||||||
break;
|
task.interrupt();
|
||||||
}
|
task.update();
|
||||||
|
|
||||||
|
std::vector<std::string> patternPath;
|
||||||
|
traversePatternTree(*pattern, patternPath, [&, this](pl::ptrn::Pattern &pattern) {
|
||||||
|
for (auto &[path, favoritePattern] : this->m_favorites) {
|
||||||
|
if (updatedFavorites == this->m_favorites.size())
|
||||||
|
task.interrupt();
|
||||||
|
task.update();
|
||||||
|
|
||||||
|
if (this->matchesFilter(patternPath, path, true)) {
|
||||||
|
favoritePattern = pattern.clone();
|
||||||
|
updatedFavorites += 1;
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::erase_if(this->m_favorites, [](const auto &entry) {
|
||||||
|
const auto &[path, favoritePattern] = entry;
|
||||||
|
|
||||||
|
return favoritePattern == nullptr;
|
||||||
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
std::erase_if(this->m_favorites, [](const auto &entry) {
|
|
||||||
const auto &[path, favoritePattern] = entry;
|
|
||||||
|
|
||||||
return favoritePattern == nullptr;
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (beginPatternTable(patterns, this->m_sortedPatterns, height)) {
|
if (beginPatternTable(patterns, this->m_sortedPatterns, height)) {
|
||||||
|
@ -1108,10 +1125,20 @@ namespace hex::plugin::builtin::ui {
|
||||||
ImGui::TableNextColumn();
|
ImGui::TableNextColumn();
|
||||||
ImGui::PushID(1);
|
ImGui::PushID(1);
|
||||||
if (ImGui::TreeNodeEx("hex.builtin.pattern_drawer.favorites"_lang, ImGuiTreeNodeFlags_SpanFullWidth)) {
|
if (ImGui::TreeNodeEx("hex.builtin.pattern_drawer.favorites"_lang, ImGuiTreeNodeFlags_SpanFullWidth)) {
|
||||||
for (auto &[path, pattern] : this->m_favorites) {
|
if (!this->m_favoritesUpdateTask.isRunning()) {
|
||||||
ImGui::PushID(pattern->getDisplayName().c_str());
|
for (auto &[path, pattern] : this->m_favorites) {
|
||||||
this->draw(*pattern);
|
if (pattern == nullptr)
|
||||||
ImGui::PopID();
|
continue;
|
||||||
|
|
||||||
|
ImGui::PushID(pattern->getDisplayName().c_str());
|
||||||
|
this->draw(*pattern);
|
||||||
|
ImGui::PopID();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
ImGui::TableNextRow();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::TableNextColumn();
|
||||||
|
ImGui::TextSpinner("hex.builtin.pattern_drawer.updating"_lang);
|
||||||
}
|
}
|
||||||
|
|
||||||
ImGui::TreePop();
|
ImGui::TreePop();
|
||||||
|
|
Loading…
Reference in New Issue