b751f98e91
* ui/ux: Initial recreation of the hex editor view * ui/ux: Added back support for editing cells * ux: Make scrolling and selecting bytes feel nice again * ui/ux: Improved byte selecting, added footer * sys: Make math evaluator more generic to support integer only calculations * patterns: Moved value formatting into pattern language * ui/ux: Added Goto and Search popups, improved selection * ui: Added better tooltips for bookmarks and patterns * sys: Use worse hex search algorithm on macOS Sadly it still doesn't support `std::boyer_moore_horsepool_searcher` * ui: Added back missing events, menu items and shortcuts * fix: Bookmark highlighting being rendered off by one * fix: Various macOS build errors * fix: size_t is not u64 on macos * fix: std::fmod and std::pow not working with integer types on macos * fix: Missing semicolons * sys: Added proper integer pow function * ui: Added back support for custom encodings * fix: Editor not jumping to selection when selection gets changed * ui: Turn Hexii setting into a data visualizer * sys: Added back remaining shortcuts * sys: Remove old hex editor files * sys: Moved more legacy things away from the hex editor view, updated localization * fix: Hex editor scrolling behaving weirdly and inconsistently * sys: Cleaned up Hex editor code * sys: Added selection color setting, localized all new settings * fix: Search feature not working correctly * ui: Replace custom ImGui::Disabled function with native ImGui ones * ui: Fix bookmark tooltip rendering issues * fix: Another size_t not being 64 bit issue on MacOS
325 lines
13 KiB
C++
325 lines
13 KiB
C++
#include <hex/api/content_registry.hpp>
|
|
|
|
#include <imgui.h>
|
|
#include <implot.h>
|
|
|
|
#include <hex/ui/view.hpp>
|
|
#include <hex/helpers/project_file_handler.hpp>
|
|
#include <hex/helpers/file.hpp>
|
|
#include <hex/helpers/crypto.hpp>
|
|
|
|
#include <thread>
|
|
|
|
namespace hex::plugin::builtin {
|
|
|
|
static bool g_demoWindowOpen = false;
|
|
|
|
static void createFileMenu() {
|
|
ContentRegistry::Interface::registerMainMenuItem("hex.builtin.menu.file", 1000);
|
|
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.file", 1050, [&] {
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.open_file"_lang, "CTRL + O")) {
|
|
|
|
fs::openFileBrowser(fs::DialogMode::Open, {}, [](const auto &path) {
|
|
EventManager::post<RequestOpenFile>(path);
|
|
});
|
|
}
|
|
|
|
if (ImGui::BeginMenu("hex.builtin.menu.file.open_other"_lang)) {
|
|
|
|
for (const auto &unlocalizedProviderName : ContentRegistry::Provider::getEntries()) {
|
|
if (ImGui::MenuItem(LangEntry(unlocalizedProviderName))) {
|
|
EventManager::post<RequestCreateProvider>(unlocalizedProviderName, nullptr);
|
|
}
|
|
}
|
|
|
|
ImGui::EndMenu();
|
|
}
|
|
});
|
|
|
|
/* File open, quit imhex */
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.file", 1150, [&] {
|
|
bool providerValid = ImHexApi::Provider::isValid();
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.close"_lang, "", false, providerValid)) {
|
|
EventManager::post<EventFileUnloaded>();
|
|
ImHexApi::Provider::remove(ImHexApi::Provider::get());
|
|
}
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.quit"_lang, "", false)) {
|
|
ImHexApi::Common::closeImHex();
|
|
}
|
|
});
|
|
|
|
/* Project open / save */
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.file", 1250, [&] {
|
|
auto provider = ImHexApi::Provider::get();
|
|
bool providerValid = ImHexApi::Provider::isValid();
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.open_project"_lang, "")) {
|
|
fs::openFileBrowser(fs::DialogMode::Open, { {"Project File", "hexproj"}
|
|
},
|
|
[](const auto &path) {
|
|
ProjectFile::load(path);
|
|
});
|
|
}
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.save_project"_lang, "", false, providerValid && provider->isWritable())) {
|
|
if (ProjectFile::getProjectFilePath() == "") {
|
|
fs::openFileBrowser(fs::DialogMode::Save, { {"Project File", "hexproj"}
|
|
},
|
|
[](std::fs::path path) {
|
|
if (path.extension() != ".hexproj") {
|
|
path.replace_extension(".hexproj");
|
|
}
|
|
|
|
ProjectFile::store(path);
|
|
});
|
|
} else
|
|
ProjectFile::store();
|
|
}
|
|
});
|
|
|
|
/* Import / Export */
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.file", 1300, [&] {
|
|
auto provider = ImHexApi::Provider::get();
|
|
bool providerValid = ImHexApi::Provider::isValid();
|
|
|
|
/* Import */
|
|
if (ImGui::BeginMenu("hex.builtin.menu.file.import"_lang)) {
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.import.base64"_lang)) {
|
|
|
|
fs::openFileBrowser(fs::DialogMode::Open, {}, [](const auto &path) {
|
|
fs::File inputFile(path, fs::File::Mode::Read);
|
|
if (!inputFile.isValid()) {
|
|
View::showErrorPopup("hex.builtin.menu.file.import.base64.popup.open_error"_lang);
|
|
return;
|
|
}
|
|
|
|
auto base64 = inputFile.readBytes();
|
|
|
|
if (!base64.empty()) {
|
|
auto data = crypt::decode64(base64);
|
|
|
|
if (data.empty())
|
|
View::showErrorPopup("hex.builtin.menu.file.import.base64.popup.import_error"_lang);
|
|
else {
|
|
fs::openFileBrowser(fs::DialogMode::Save, {}, [&data](const std::fs::path &path) {
|
|
fs::File outputFile(path, fs::File::Mode::Create);
|
|
|
|
if (!outputFile.isValid())
|
|
View::showErrorPopup("hex.builtin.menu.file.import.base64.popup.import_error"_lang);
|
|
|
|
outputFile.write(data);
|
|
});
|
|
}
|
|
} else {
|
|
View::showErrorPopup("hex.builtin.popup.file_open_error"_lang);
|
|
}
|
|
});
|
|
}
|
|
|
|
ImGui::Separator();
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.import.ips"_lang, nullptr, false)) {
|
|
|
|
fs::openFileBrowser(fs::DialogMode::Open, {}, [](const auto &path) {
|
|
std::thread([path] {
|
|
auto task = ImHexApi::Tasks::createTask("hex.common.processing", 0);
|
|
|
|
auto patchData = fs::File(path, fs::File::Mode::Read).readBytes();
|
|
auto patch = hex::loadIPSPatch(patchData);
|
|
|
|
task.setMaxValue(patch.size());
|
|
|
|
auto provider = ImHexApi::Provider::get();
|
|
|
|
u64 progress = 0;
|
|
for (auto &[address, value] : patch) {
|
|
provider->addPatch(address, &value, 1);
|
|
progress++;
|
|
task.update(progress);
|
|
}
|
|
|
|
provider->createUndoPoint();
|
|
}).detach();
|
|
});
|
|
}
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.import.ips32"_lang, nullptr, false)) {
|
|
fs::openFileBrowser(fs::DialogMode::Open, {}, [](const auto &path) {
|
|
std::thread([path] {
|
|
auto task = ImHexApi::Tasks::createTask("hex.common.processing", 0);
|
|
|
|
auto patchData = fs::File(path, fs::File::Mode::Read).readBytes();
|
|
auto patch = hex::loadIPS32Patch(patchData);
|
|
|
|
task.setMaxValue(patch.size());
|
|
|
|
auto provider = ImHexApi::Provider::get();
|
|
|
|
u64 progress = 0;
|
|
for (auto &[address, value] : patch) {
|
|
provider->addPatch(address, &value, 1);
|
|
progress++;
|
|
task.update(progress);
|
|
}
|
|
|
|
provider->createUndoPoint();
|
|
}).detach();
|
|
});
|
|
}
|
|
|
|
ImGui::EndMenu();
|
|
}
|
|
|
|
|
|
/* Export */
|
|
if (ImGui::BeginMenu("hex.builtin.menu.file.export"_lang, providerValid && provider->isWritable())) {
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.export.ips"_lang, nullptr, false)) {
|
|
Patches patches = provider->getPatches();
|
|
if (!patches.contains(0x00454F45) && patches.contains(0x00454F46)) {
|
|
u8 value = 0;
|
|
provider->read(0x00454F45, &value, sizeof(u8));
|
|
patches[0x00454F45] = value;
|
|
}
|
|
|
|
std::thread([patches] {
|
|
auto task = ImHexApi::Tasks::createTask("hex.common.processing", 0);
|
|
|
|
auto data = generateIPSPatch(patches);
|
|
|
|
ImHexApi::Tasks::doLater([data] {
|
|
fs::openFileBrowser(fs::DialogMode::Save, {}, [&data](const auto &path) {
|
|
auto file = fs::File(path, fs::File::Mode::Create);
|
|
if (!file.isValid()) {
|
|
View::showErrorPopup("hex.builtin.menu.file.export.base64.popup.export_error"_lang);
|
|
return;
|
|
}
|
|
|
|
file.write(data);
|
|
});
|
|
});
|
|
}).detach();
|
|
}
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.file.export.ips32"_lang, nullptr, false)) {
|
|
Patches patches = provider->getPatches();
|
|
if (!patches.contains(0x00454F45) && patches.contains(0x45454F46)) {
|
|
u8 value = 0;
|
|
provider->read(0x45454F45, &value, sizeof(u8));
|
|
patches[0x45454F45] = value;
|
|
}
|
|
|
|
std::thread([patches] {
|
|
auto task = ImHexApi::Tasks::createTask("hex.common.processing", 0);
|
|
|
|
auto data = generateIPS32Patch(patches);
|
|
|
|
ImHexApi::Tasks::doLater([data] {
|
|
fs::openFileBrowser(fs::DialogMode::Save, {}, [&data](const auto &path) {
|
|
auto file = fs::File(path, fs::File::Mode::Create);
|
|
if (!file.isValid()) {
|
|
View::showErrorPopup("hex.builtin.menu.file.export.popup.create"_lang);
|
|
return;
|
|
}
|
|
|
|
file.write(data);
|
|
});
|
|
});
|
|
}).detach();
|
|
}
|
|
|
|
ImGui::EndMenu();
|
|
}
|
|
});
|
|
}
|
|
|
|
static void createEditMenu() {
|
|
ContentRegistry::Interface::registerMainMenuItem("hex.builtin.menu.edit", 2000);
|
|
|
|
/* Provider Undo / Redo */
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.edit", 1000, [&] {
|
|
auto provider = ImHexApi::Provider::get();
|
|
bool providerValid = ImHexApi::Provider::isValid();
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.edit.undo"_lang, "CTRL + Z", false, providerValid && provider->canUndo()))
|
|
provider->undo();
|
|
if (ImGui::MenuItem("hex.builtin.menu.edit.redo"_lang, "CTRL + Y", false, providerValid && provider->canRedo()))
|
|
provider->redo();
|
|
});
|
|
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.edit", 1050, [&] {
|
|
auto provider = ImHexApi::Provider::get();
|
|
bool providerValid = ImHexApi::Provider::isValid();
|
|
auto selection = ImHexApi::HexEditor::getSelection();
|
|
|
|
if (ImGui::MenuItem("hex.builtin.menu.edit.bookmark"_lang, nullptr, false, selection.has_value() && providerValid)) {
|
|
auto base = provider->getBaseAddress();
|
|
|
|
ImHexApi::Bookmarks::add(base + selection->getStartAddress(), selection->getEndAddress(), {}, {});
|
|
}
|
|
});
|
|
|
|
}
|
|
|
|
static void createViewMenu() {
|
|
ContentRegistry::Interface::registerMainMenuItem("hex.builtin.menu.view", 3000);
|
|
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.view", 1000, [] {
|
|
for (auto &[name, view] : ContentRegistry::Views::getEntries()) {
|
|
if (view->hasViewMenuItemEntry())
|
|
ImGui::MenuItem(LangEntry(view->getUnlocalizedName()), "", &view->getWindowOpenState());
|
|
}
|
|
});
|
|
|
|
#if defined(DEBUG)
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.view", 2000, [] {
|
|
ImGui::MenuItem("hex.builtin.menu.view.demo"_lang, "", &g_demoWindowOpen);
|
|
});
|
|
#endif
|
|
}
|
|
|
|
static void createLayoutMenu() {
|
|
ContentRegistry::Interface::registerMainMenuItem("hex.builtin.menu.layout", 4000);
|
|
|
|
ContentRegistry::Interface::addMenuItem("hex.builtin.menu.layout", 1000, [] {
|
|
for (auto &[layoutName, func] : ContentRegistry::Interface::getLayouts()) {
|
|
if (ImGui::MenuItem(LangEntry(layoutName), "", false, ImHexApi::Provider::isValid())) {
|
|
auto dock = ImHexApi::System::getMainDockSpaceId();
|
|
|
|
for (auto &[viewName, view] : ContentRegistry::Views::getEntries()) {
|
|
view->getWindowOpenState() = false;
|
|
}
|
|
|
|
ImGui::DockBuilderRemoveNode(dock);
|
|
ImGui::DockBuilderAddNode(dock);
|
|
func(dock);
|
|
ImGui::DockBuilderFinish(dock);
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
static void createHelpMenu() {
|
|
ContentRegistry::Interface::registerMainMenuItem("hex.builtin.menu.help", 5000);
|
|
|
|
}
|
|
|
|
|
|
void registerMainMenuEntries() {
|
|
createFileMenu();
|
|
createEditMenu();
|
|
createViewMenu();
|
|
createLayoutMenu();
|
|
createHelpMenu();
|
|
|
|
(void)EventManager::subscribe<EventFrameEnd>([] {
|
|
if (g_demoWindowOpen) {
|
|
ImGui::ShowDemoWindow(&g_demoWindowOpen);
|
|
ImPlot::ShowDemoWindow(&g_demoWindowOpen);
|
|
}
|
|
});
|
|
}
|
|
|
|
} |