199 lines
6.8 KiB
C
199 lines
6.8 KiB
C
|
#pragma once
|
||
|
|
||
|
#include "imgui.h"
|
||
|
#include "imgui_internal.h"
|
||
|
|
||
|
inline static void CopyIOEvents(ImGuiContext *src, ImGuiContext *dst,
|
||
|
ImVec2 origin, float scale) {
|
||
|
dst->InputEventsQueue = src->InputEventsTrail;
|
||
|
for (ImGuiInputEvent &e : dst->InputEventsQueue) {
|
||
|
if (e.Type == ImGuiInputEventType_MousePos) {
|
||
|
e.MousePos.PosX = (e.MousePos.PosX - origin.x) / scale;
|
||
|
e.MousePos.PosY = (e.MousePos.PosY - origin.y) / scale;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
inline static void AppendDrawData(ImDrawList *src, ImVec2 origin, float scale) {
|
||
|
// TODO optimize if vtx_start == 0 || if idx_start == 0
|
||
|
ImDrawList *dl = ImGui::GetWindowDrawList();
|
||
|
const int vtx_start = dl->VtxBuffer.size();
|
||
|
const int idx_start = dl->IdxBuffer.size();
|
||
|
dl->VtxBuffer.resize(dl->VtxBuffer.size() + src->VtxBuffer.size());
|
||
|
dl->IdxBuffer.resize(dl->IdxBuffer.size() + src->IdxBuffer.size());
|
||
|
dl->CmdBuffer.reserve(dl->CmdBuffer.size() + src->CmdBuffer.size());
|
||
|
dl->_VtxWritePtr = dl->VtxBuffer.Data + vtx_start;
|
||
|
dl->_IdxWritePtr = dl->IdxBuffer.Data + idx_start;
|
||
|
const ImDrawVert *vtx_read = src->VtxBuffer.Data;
|
||
|
const ImDrawIdx *idx_read = src->IdxBuffer.Data;
|
||
|
for (int i = 0, c = src->VtxBuffer.size(); i < c; ++i) {
|
||
|
dl->_VtxWritePtr[i].uv = vtx_read[i].uv;
|
||
|
dl->_VtxWritePtr[i].col = vtx_read[i].col;
|
||
|
dl->_VtxWritePtr[i].pos = vtx_read[i].pos * scale + origin;
|
||
|
}
|
||
|
for (int i = 0, c = src->IdxBuffer.size(); i < c; ++i) {
|
||
|
dl->_IdxWritePtr[i] = idx_read[i] + vtx_start;
|
||
|
}
|
||
|
for (auto cmd : src->CmdBuffer) {
|
||
|
cmd.IdxOffset += idx_start;
|
||
|
IM_ASSERT(cmd.VtxOffset == 0);
|
||
|
cmd.ClipRect.x = cmd.ClipRect.x * scale + origin.x;
|
||
|
cmd.ClipRect.y = cmd.ClipRect.y * scale + origin.y;
|
||
|
cmd.ClipRect.z = cmd.ClipRect.z * scale + origin.x;
|
||
|
cmd.ClipRect.w = cmd.ClipRect.w * scale + origin.y;
|
||
|
dl->CmdBuffer.push_back(cmd);
|
||
|
}
|
||
|
|
||
|
dl->_VtxCurrentIdx += src->VtxBuffer.size();
|
||
|
dl->_VtxWritePtr = dl->VtxBuffer.Data + dl->VtxBuffer.size();
|
||
|
dl->_IdxWritePtr = dl->IdxBuffer.Data + dl->IdxBuffer.size();
|
||
|
}
|
||
|
|
||
|
struct ContainedContextConfig {
|
||
|
bool extra_window_wrapper = false;
|
||
|
ImVec2 size = {0.f, 0.f};
|
||
|
ImU32 color = IM_COL32_WHITE;
|
||
|
bool zoom_enabled = true;
|
||
|
float zoom_min = 0.3f;
|
||
|
float zoom_max = 2.f;
|
||
|
float zoom_divisions = 10.f;
|
||
|
float zoom_smoothness = 5.f;
|
||
|
float default_zoom = 1.f;
|
||
|
ImGuiKey reset_zoom_key = ImGuiKey_R;
|
||
|
ImGuiMouseButton scroll_button = ImGuiMouseButton_Middle;
|
||
|
};
|
||
|
|
||
|
class ContainedContext {
|
||
|
public:
|
||
|
~ContainedContext();
|
||
|
ContainedContextConfig &config() { return m_config; }
|
||
|
void begin();
|
||
|
void end();
|
||
|
[[nodiscard]] float scale() const { return m_scale; }
|
||
|
[[nodiscard]] const ImVec2 &origin() const { return m_origin; }
|
||
|
[[nodiscard]] bool hovered() const { return m_hovered; }
|
||
|
[[nodiscard]] const ImVec2 &scroll() const { return m_scroll; }
|
||
|
ImGuiContext *getRawContext() { return m_ctx; }
|
||
|
|
||
|
private:
|
||
|
ContainedContextConfig m_config;
|
||
|
|
||
|
ImVec2 m_origin;
|
||
|
ImVec2 m_pos;
|
||
|
ImGuiContext *m_ctx = nullptr;
|
||
|
ImGuiContext *m_original_ctx = nullptr;
|
||
|
|
||
|
bool m_anyWindowHovered = false;
|
||
|
bool m_anyItemActive = false;
|
||
|
bool m_hovered = false;
|
||
|
|
||
|
float m_scale = m_config.default_zoom, m_scaleTarget = m_config.default_zoom;
|
||
|
ImVec2 m_scroll = {0.f, 0.f}, m_scrollTarget = {0.f, 0.f};
|
||
|
};
|
||
|
|
||
|
inline ContainedContext::~ContainedContext() {
|
||
|
if (m_ctx)
|
||
|
ImGui::DestroyContext(m_ctx);
|
||
|
}
|
||
|
|
||
|
inline void ContainedContext::begin() {
|
||
|
ImGui::PushID(this);
|
||
|
ImGui::PushStyleColor(ImGuiCol_ChildBg, m_config.color);
|
||
|
ImGui::BeginChild("view_port", m_config.size, 0, ImGuiWindowFlags_NoMove);
|
||
|
ImGui::PopStyleColor();
|
||
|
// m_size = ImGui::GetWindowSize();
|
||
|
m_pos = ImGui::GetWindowPos();
|
||
|
|
||
|
ImVec2 size = ImGui::GetContentRegionAvail();
|
||
|
m_origin = ImGui::GetCursorScreenPos();
|
||
|
m_original_ctx = ImGui::GetCurrentContext();
|
||
|
const ImGuiStyle &orig_style = ImGui::GetStyle();
|
||
|
if (!m_ctx)
|
||
|
m_ctx = ImGui::CreateContext(ImGui::GetIO().Fonts);
|
||
|
ImGui::SetCurrentContext(m_ctx);
|
||
|
ImGuiStyle &new_style = ImGui::GetStyle();
|
||
|
new_style = orig_style;
|
||
|
|
||
|
CopyIOEvents(m_original_ctx, m_ctx, m_origin, m_scale);
|
||
|
|
||
|
ImGui::GetIO().DisplaySize = size / m_scale;
|
||
|
ImGui::GetIO().ConfigInputTrickleEventQueue = false;
|
||
|
ImGui::NewFrame();
|
||
|
|
||
|
if (!m_config.extra_window_wrapper)
|
||
|
return;
|
||
|
ImGui::SetNextWindowPos(ImVec2(0, 0), ImGuiCond_Appearing);
|
||
|
ImGui::SetNextWindowSize(ImGui::GetMainViewport()->WorkSize);
|
||
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0, 0));
|
||
|
ImGui::Begin("viewport_container", nullptr,
|
||
|
ImGuiWindowFlags_NoDecoration | ImGuiWindowFlags_NoBackground |
|
||
|
ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoScrollbar |
|
||
|
ImGuiWindowFlags_NoScrollWithMouse);
|
||
|
ImGui::PopStyleVar();
|
||
|
}
|
||
|
|
||
|
inline void ContainedContext::end() {
|
||
|
m_anyWindowHovered = ImGui::IsWindowHovered(ImGuiHoveredFlags_AnyWindow);
|
||
|
if (m_config.extra_window_wrapper && ImGui::IsWindowHovered())
|
||
|
m_anyWindowHovered = false;
|
||
|
|
||
|
m_anyItemActive = ImGui::IsAnyItemActive();
|
||
|
|
||
|
if (m_config.extra_window_wrapper)
|
||
|
ImGui::End();
|
||
|
|
||
|
ImGui::Render();
|
||
|
|
||
|
ImDrawData *draw_data = ImGui::GetDrawData();
|
||
|
|
||
|
ImGui::SetCurrentContext(m_original_ctx);
|
||
|
m_original_ctx = nullptr;
|
||
|
|
||
|
for (int i = 0; i < draw_data->CmdListsCount; ++i)
|
||
|
AppendDrawData(draw_data->CmdLists[i], m_origin, m_scale);
|
||
|
|
||
|
m_hovered = ImGui::IsWindowHovered(ImGuiHoveredFlags_ChildWindows) &&
|
||
|
!m_anyWindowHovered;
|
||
|
|
||
|
// Zooming
|
||
|
if (m_config.zoom_enabled && m_hovered && ImGui::GetIO().MouseWheel != 0.f) {
|
||
|
m_scaleTarget += ImGui::GetIO().MouseWheel / m_config.zoom_divisions;
|
||
|
m_scaleTarget =
|
||
|
m_scaleTarget < m_config.zoom_min ? m_config.zoom_min : m_scaleTarget;
|
||
|
m_scaleTarget =
|
||
|
m_scaleTarget > m_config.zoom_max ? m_config.zoom_max : m_scaleTarget;
|
||
|
|
||
|
if (m_config.zoom_smoothness == 0.f) {
|
||
|
m_scroll += (ImGui::GetMousePos() - m_pos) / m_scaleTarget -
|
||
|
(ImGui::GetMousePos() - m_pos) / m_scale;
|
||
|
m_scale = m_scaleTarget;
|
||
|
}
|
||
|
}
|
||
|
if (abs(m_scaleTarget - m_scale) >= 0.015f / m_config.zoom_smoothness) {
|
||
|
float cs = (m_scaleTarget - m_scale) / m_config.zoom_smoothness;
|
||
|
m_scroll += (ImGui::GetMousePos() - m_pos) / (m_scale + cs) -
|
||
|
(ImGui::GetMousePos() - m_pos) / m_scale;
|
||
|
m_scale += (m_scaleTarget - m_scale) / m_config.zoom_smoothness;
|
||
|
|
||
|
if (abs(m_scaleTarget - m_scale) < 0.015f / m_config.zoom_smoothness) {
|
||
|
m_scroll += (ImGui::GetMousePos() - m_pos) / m_scaleTarget -
|
||
|
(ImGui::GetMousePos() - m_pos) / m_scale;
|
||
|
m_scale = m_scaleTarget;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Zoom reset
|
||
|
if (ImGui::IsKeyPressed(m_config.reset_zoom_key, false))
|
||
|
m_scaleTarget = m_config.default_zoom;
|
||
|
|
||
|
// Scrolling
|
||
|
if (m_hovered && !m_anyItemActive &&
|
||
|
ImGui::IsMouseDragging(m_config.scroll_button, 0.f)) {
|
||
|
m_scroll += ImGui::GetIO().MouseDelta / m_scale;
|
||
|
m_scrollTarget = m_scroll;
|
||
|
}
|
||
|
|
||
|
ImGui::EndChild();
|
||
|
ImGui::PopID();
|
||
|
}
|