#include #include #include #include #include namespace hex::plugin::builtin { template constexpr ImGuiDataType getImGuiDataType() { if constexpr (std::same_as) return ImGuiDataType_U8; else if constexpr (std::same_as) return ImGuiDataType_U16; else if constexpr (std::same_as) return ImGuiDataType_U32; else if constexpr (std::same_as) return ImGuiDataType_U64; else if constexpr (std::same_as) return ImGuiDataType_S8; else if constexpr (std::same_as) return ImGuiDataType_S16; else if constexpr (std::same_as) return ImGuiDataType_S32; else if constexpr (std::same_as) return ImGuiDataType_S64; else if constexpr (std::same_as) return ImGuiDataType_Float; else if constexpr (std::same_as) return ImGuiDataType_Double; else static_assert(hex::always_false::value, "Invalid data type!"); } template class DataVisualizerHexadecimal : public hex::ContentRegistry::HexEditor::DataVisualizer { public: DataVisualizerHexadecimal() : DataVisualizer(ByteCount, CharCount) { } void draw(u64 address, const u8 *data, size_t size, bool upperCase) override { hex::unused(address); if (size == ByteCount) ImGui::Text(getFormatString(upperCase), *reinterpret_cast(data)); else ImGui::TextFormatted("{: {}s}", CharCount); } bool drawEditing(u64 address, u8 *data, size_t size, bool upperCase, bool startedEditing) override { hex::unused(address, startedEditing); if (size == ByteCount) { return drawDefaultEditingTextBox(address, getFormatString(upperCase), getImGuiDataType(), data, ImGuiInputTextFlags_CharsHexadecimal); } else return false; } private: constexpr static inline auto ByteCount = sizeof(T); constexpr static inline auto CharCount = ByteCount * 2; const static inline auto FormattingUpperCase = hex::format("%0{}X", CharCount); const static inline auto FormattingLowerCase = hex::format("%0{}x", CharCount); const char *getFormatString(bool upperCase) { if (upperCase) return FormattingUpperCase.c_str(); else return FormattingLowerCase.c_str(); } }; class DataVisualizerHexii : public hex::ContentRegistry::HexEditor::DataVisualizer { public: DataVisualizerHexii() : DataVisualizer(ByteCount, CharCount) { } void draw(u64 address, const u8 *data, size_t size, bool upperCase) override { hex::unused(address); if (size == ByteCount) { const u8 c = data[0]; switch (c) { case 0x00: ImGui::Text(" "); break; case 0xFF: ImGui::TextDisabled("##"); break; case ' ' ... '~': ImGui::Text(".%c", c); break; default: ImGui::Text(getFormatString(upperCase), c); break; } } else ImGui::TextFormatted("{: {}s}", CharCount); } bool drawEditing(u64 address, u8 *data, size_t size, bool upperCase, bool startedEditing) override { hex::unused(address, startedEditing); if (size == ByteCount) { return drawDefaultEditingTextBox(address, getFormatString(upperCase), getImGuiDataType(), data, ImGuiInputTextFlags_CharsHexadecimal); } else return false; } private: constexpr static inline auto ByteCount = 1; constexpr static inline auto CharCount = ByteCount * 2; const static inline auto FormattingUpperCase = hex::format("%0{}X", CharCount); const static inline auto FormattingLowerCase = hex::format("%0{}x", CharCount); const char *getFormatString(bool upperCase) { if (upperCase) return FormattingUpperCase.c_str(); else return FormattingLowerCase.c_str(); } }; template class DataVisualizerDecimal : public hex::ContentRegistry::HexEditor::DataVisualizer { public: DataVisualizerDecimal() : DataVisualizer(ByteCount, CharCount) { } void draw(u64 address, const u8 *data, size_t size, bool upperCase) override { hex::unused(address, upperCase); if (size == ByteCount) { if (hex::is_signed::value) ImGui::Text(getFormatString(), static_cast(*reinterpret_cast(data))); else ImGui::Text(getFormatString(), static_cast(*reinterpret_cast(data))); } else ImGui::TextFormatted("{: {}s}", CharCount); } bool drawEditing(u64 address, u8 *data, size_t size, bool upperCase, bool startedEditing) override { hex::unused(address, upperCase, startedEditing); if (size == ByteCount) { return ImGui::InputScalar( "##hex_input", getImGuiDataType(), data, nullptr, nullptr, nullptr, DataVisualizer::TextInputFlags); } else return false; } private: constexpr static inline auto ByteCount = sizeof(T); constexpr static inline auto CharCount = std::numeric_limits::digits10 + 2; const static inline auto FormatString = hex::format("%{}{}", CharCount, hex::is_signed::value ? "lld" : "llu"); const char *getFormatString() { return FormatString.c_str(); } }; template class DataVisualizerFloatingPoint : public hex::ContentRegistry::HexEditor::DataVisualizer { public: DataVisualizerFloatingPoint() : DataVisualizer(ByteCount, CharCount) { } void draw(u64 address, const u8 *data, size_t size, bool upperCase) override { hex::unused(address); if (size == ByteCount) ImGui::Text(getFormatString(upperCase), *reinterpret_cast(data)); else ImGui::TextFormatted("{: {}s}", CharCount); } bool drawEditing(u64 address, u8 *data, size_t size, bool upperCase, bool startedEditing) override { hex::unused(address, upperCase, startedEditing); if (size == ByteCount) { return ImGui::InputScalar( "##hex_input", getImGuiDataType(), data, nullptr, nullptr, nullptr, DataVisualizer::TextInputFlags | ImGuiInputTextFlags_CharsScientific); } else return false; } private: constexpr static inline auto ByteCount = sizeof(T); constexpr static inline auto CharCount = 14; const static inline auto FormatStringUpperCase = hex::format("%{}E", CharCount); const static inline auto FormatStringLowerCase = hex::format("%{}e", CharCount); const char *getFormatString(bool upperCase) { if (upperCase) return FormatStringUpperCase.c_str(); else return FormatStringLowerCase.c_str(); } }; class DataVisualizerRGBA8 : public hex::ContentRegistry::HexEditor::DataVisualizer { public: DataVisualizerRGBA8() : DataVisualizer(4, 2) { } void draw(u64 address, const u8 *data, size_t size, bool upperCase) override { hex::unused(address, upperCase); if (size == 4) ImGui::ColorButton("##color", ImColor(data[0], data[1], data[2], data[3]), ImGuiColorEditFlags_AlphaPreview | ImGuiColorEditFlags_NoLabel | ImGuiColorEditFlags_NoDragDrop, ImVec2(ImGui::GetColumnWidth(), ImGui::GetTextLineHeight())); else ImGui::ColorButton("##color", ImColor(0, 0, 0, 0xFF), ImGuiColorEditFlags_AlphaPreview | ImGuiColorEditFlags_NoLabel | ImGuiColorEditFlags_NoDragDrop, ImVec2(ImGui::GetColumnWidth(), ImGui::GetTextLineHeight())); } bool drawEditing(u64 address, u8 *data, size_t size, bool upperCase, bool startedEditing) override { hex::unused(address, data, size, upperCase); if (startedEditing) { this->m_currColor = { float(data[0]) / 0xFF, float(data[1]) / 0xFF, float(data[2]) / 0xFF, float(data[3]) / 0xFF }; ImGui::OpenPopup("##color_popup"); } ImGui::ColorButton("##color", ImColor(this->m_currColor[0], this->m_currColor[1], this->m_currColor[2], this->m_currColor[3]), ImGuiColorEditFlags_AlphaPreview | ImGuiColorEditFlags_NoLabel | ImGuiColorEditFlags_NoDragDrop, ImVec2(ImGui::GetColumnWidth(), ImGui::GetTextLineHeight())); if (ImGui::BeginPopup("##color_popup")) { if (ImGui::ColorPicker4("##picker", this->m_currColor.data(), ImGuiColorEditFlags_AlphaBar)) { for (u8 i = 0; i < 4; i++) data[i] = this->m_currColor[i] * 0xFF; } ImGui::EndPopup(); } else { return true; } return false; } std::array m_currColor; }; void registerDataVisualizers() { ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.hexadecimal.8bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.hexadecimal.16bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.hexadecimal.32bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.hexadecimal.64bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.unsigned.8bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.unsigned.16bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.unsigned.32bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.unsigned.64bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.signed.8bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.signed.16bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.signed.32bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.decimal.signed.64bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.floating_point.32bit"); ContentRegistry::HexEditor::addDataVisualizer>("hex.builtin.visualizer.floating_point.64bit"); ContentRegistry::HexEditor::addDataVisualizer("hex.builtin.visualizer.rgba8"); ContentRegistry::HexEditor::addDataVisualizer("hex.builtin.visualizer.hexii"); } }