Poly3d/imgui/GraphEditor.cpp

1112 lines
44 KiB
C++
Raw Normal View History

2024-11-25 23:08:00 +00:00
// https://github.com/CedricGuillemet/ImGuizmo
// v1.91.3 WIP
//
// The MIT License(MIT)
//
// Copyright(c) 2021 Cedric Guillemet
//
// Permission is hereby granted, free of charge, to any person obtaining a copy
// of this software and associated documentation files(the "Software"), to deal
// in the Software without restriction, including without limitation the rights
// to use, copy, modify, merge, publish, distribute, sublicense, and / or sell
// copies of the Software, and to permit persons to whom the Software is
// furnished to do so, subject to the following conditions :
//
// The above copyright notice and this permission notice shall be included in all
// copies or substantial portions of the Software.
//
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.IN NO EVENT SHALL THE
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
// SOFTWARE.
//
#define IMGUI_DEFINE_MATH_OPERATORS
#include "imgui.h"
#include "imgui_internal.h"
#include <math.h>
#include <vector>
#include <float.h>
#include <array>
#include "GraphEditor.h"
namespace GraphEditor {
static inline float Distance(const ImVec2& a, const ImVec2& b)
{
return sqrtf((a.x - b.x) * (a.x - b.x) + (a.y - b.y) * (a.y - b.y));
}
static inline float sign(float v)
{
return (v >= 0.f) ? 1.f : -1.f;
}
static ImVec2 GetInputSlotPos(Delegate& delegate, const Node& node, SlotIndex slotIndex, float factor)
{
ImVec2 Size = node.mRect.GetSize() * factor;
size_t InputsCount = delegate.GetTemplate(node.mTemplateIndex).mInputCount;
return ImVec2(node.mRect.Min.x * factor,
node.mRect.Min.y * factor + Size.y * ((float)slotIndex + 1) / ((float)InputsCount + 1) + 8.f);
}
static ImVec2 GetOutputSlotPos(Delegate& delegate, const Node& node, SlotIndex slotIndex, float factor)
{
ImVec2 Size = node.mRect.GetSize() * factor;
size_t OutputsCount = delegate.GetTemplate(node.mTemplateIndex).mOutputCount;
return ImVec2(node.mRect.Min.x * factor + Size.x,
node.mRect.Min.y * factor + Size.y * ((float)slotIndex + 1) / ((float)OutputsCount + 1) + 8.f);
}
static ImRect GetNodeRect(const Node& node, float factor)
{
ImVec2 Size = node.mRect.GetSize() * factor;
return ImRect(node.mRect.Min * factor, node.mRect.Min * factor + Size);
}
static ImVec2 editingNodeSource;
static bool editingInput = false;
static ImVec2 captureOffset;
enum NodeOperation
{
NO_None,
NO_EditingLink,
NO_QuadSelecting,
NO_MovingNodes,
NO_EditInput,
NO_PanView,
};
static NodeOperation nodeOperation = NO_None;
static void HandleZoomScroll(ImRect regionRect, ViewState& viewState, const Options& options)
{
ImGuiIO& io = ImGui::GetIO();
if (regionRect.Contains(io.MousePos))
{
if (io.MouseWheel < -FLT_EPSILON)
{
viewState.mFactorTarget *= 1.f - options.mZoomRatio;
}
if (io.MouseWheel > FLT_EPSILON)
{
viewState.mFactorTarget *= 1.0f + options.mZoomRatio;
}
}
ImVec2 mouseWPosPre = (io.MousePos - ImGui::GetCursorScreenPos()) / viewState.mFactor;
viewState.mFactorTarget = ImClamp(viewState.mFactorTarget, options.mMinZoom, options.mMaxZoom);
viewState.mFactor = ImLerp(viewState.mFactor, viewState.mFactorTarget, options.mZoomLerpFactor);
ImVec2 mouseWPosPost = (io.MousePos - ImGui::GetCursorScreenPos()) / viewState.mFactor;
if (ImGui::IsMousePosValid())
{
viewState.mPosition += mouseWPosPost - mouseWPosPre;
}
}
void GraphEditorClear()
{
nodeOperation = NO_None;
}
static void FitNodes(Delegate& delegate, ViewState& viewState, const ImVec2 viewSize, bool selectedNodesOnly)
{
const size_t nodeCount = delegate.GetNodeCount();
if (!nodeCount)
{
return;
}
bool validNode = false;
ImVec2 min(FLT_MAX, FLT_MAX);
ImVec2 max(-FLT_MAX, -FLT_MAX);
for (NodeIndex nodeIndex = 0; nodeIndex < nodeCount; nodeIndex++)
{
const Node& node = delegate.GetNode(nodeIndex);
if (selectedNodesOnly && !node.mSelected)
{
continue;
}
min = ImMin(min, node.mRect.Min);
min = ImMin(min, node.mRect.Max);
max = ImMax(max, node.mRect.Min);
max = ImMax(max, node.mRect.Max);
validNode = true;
}
if (!validNode)
{
return;
}
min -= viewSize * 0.05f;
max += viewSize * 0.05f;
ImVec2 nodesSize = max - min;
ImVec2 nodeCenter = (max + min) * 0.5f;
float ratioY = viewSize.y / nodesSize.y;
float ratioX = viewSize.x / nodesSize.x;
viewState.mFactor = viewState.mFactorTarget = ImMin(ImMin(ratioY, ratioX), 1.f);
viewState.mPosition = ImVec2(-nodeCenter.x, -nodeCenter.y) + (viewSize * 0.5f) / viewState.mFactorTarget;
}
static void DisplayLinks(Delegate& delegate,
ImDrawList* drawList,
const ImVec2 offset,
const float factor,
const ImRect regionRect,
NodeIndex hoveredNode,
const Options& options)
{
const size_t linkCount = delegate.GetLinkCount();
for (LinkIndex linkIndex = 0; linkIndex < linkCount; linkIndex++)
{
const auto link = delegate.GetLink(linkIndex);
const auto nodeInput = delegate.GetNode(link.mInputNodeIndex);
const auto nodeOutput = delegate.GetNode(link.mOutputNodeIndex);
ImVec2 p1 = offset + GetOutputSlotPos(delegate, nodeInput, link.mInputSlotIndex, factor);
ImVec2 p2 = offset + GetInputSlotPos(delegate, nodeOutput, link.mOutputSlotIndex, factor);
// con. view clipping
if ((p1.y < 0.f && p2.y < 0.f) || (p1.y > regionRect.Max.y && p2.y > regionRect.Max.y) ||
(p1.x < 0.f && p2.x < 0.f) || (p1.x > regionRect.Max.x && p2.x > regionRect.Max.x))
continue;
bool highlightCons = hoveredNode == link.mInputNodeIndex || hoveredNode == link.mOutputNodeIndex;
uint32_t col = delegate.GetTemplate(nodeInput.mTemplateIndex).mHeaderColor | (highlightCons ? 0xF0F0F0 : 0);
if (options.mDisplayLinksAsCurves)
{
// curves
drawList->AddBezierCubic(p1, p1 + ImVec2(50, 0) * factor, p2 + ImVec2(-50, 0) * factor, p2, 0xFF000000, options.mLineThickness * 1.5f * factor);
drawList->AddBezierCubic(p1, p1 + ImVec2(50, 0) * factor, p2 + ImVec2(-50, 0) * factor, p2, col, options.mLineThickness * 1.5f * factor);
/*
ImVec2 p10 = p1 + ImVec2(20.f * factor, 0.f);
ImVec2 p20 = p2 - ImVec2(20.f * factor, 0.f);
ImVec2 dif = p20 - p10;
ImVec2 p1a, p1b;
if (fabsf(dif.x) > fabsf(dif.y))
{
p1a = p10 + ImVec2(fabsf(fabsf(dif.x) - fabsf(dif.y)) * 0.5 * sign(dif.x), 0.f);
p1b = p1a + ImVec2(fabsf(dif.y) * sign(dif.x) , dif.y);
}
else
{
p1a = p10 + ImVec2(0.f, fabsf(fabsf(dif.y) - fabsf(dif.x)) * 0.5 * sign(dif.y));
p1b = p1a + ImVec2(dif.x, fabsf(dif.x) * sign(dif.y));
}
drawList->AddLine(p1, p10, col, 3.f * factor);
drawList->AddLine(p10, p1a, col, 3.f * factor);
drawList->AddLine(p1a, p1b, col, 3.f * factor);
drawList->AddLine(p1b, p20, col, 3.f * factor);
drawList->AddLine(p20, p2, col, 3.f * factor);
*/
}
else
{
// straight lines
std::array<ImVec2, 6> pts;
int ptCount = 0;
ImVec2 dif = p2 - p1;
ImVec2 p1a, p1b;
const float limitx = 12.f * factor;
if (dif.x < limitx)
{
ImVec2 p10 = p1 + ImVec2(limitx, 0.f);
ImVec2 p20 = p2 - ImVec2(limitx, 0.f);
dif = p20 - p10;
p1a = p10 + ImVec2(0.f, dif.y * 0.5f);
p1b = p1a + ImVec2(dif.x, 0.f);
pts = { p1, p10, p1a, p1b, p20, p2 };
ptCount = 6;
}
else
{
if (fabsf(dif.y) < 1.f)
{
pts = { p1, (p1 + p2) * 0.5f, p2 };
ptCount = 3;
}
else
{
if (fabsf(dif.y) < 10.f)
{
if (fabsf(dif.x) > fabsf(dif.y))
{
p1a = p1 + ImVec2(fabsf(fabsf(dif.x) - fabsf(dif.y)) * 0.5f * sign(dif.x), 0.f);
p1b = p1a + ImVec2(fabsf(dif.y) * sign(dif.x), dif.y);
}
else
{
p1a = p1 + ImVec2(0.f, fabsf(fabsf(dif.y) - fabsf(dif.x)) * 0.5f * sign(dif.y));
p1b = p1a + ImVec2(dif.x, fabsf(dif.x) * sign(dif.y));
}
}
else
{
if (fabsf(dif.x) > fabsf(dif.y))
{
float d = fabsf(dif.y) * sign(dif.x) * 0.5f;
p1a = p1 + ImVec2(d, dif.y * 0.5f);
p1b = p1a + ImVec2(fabsf(fabsf(dif.x) - fabsf(d) * 2.f) * sign(dif.x), 0.f);
}
else
{
float d = fabsf(dif.x) * sign(dif.y) * 0.5f;
p1a = p1 + ImVec2(dif.x * 0.5f, d);
p1b = p1a + ImVec2(0.f, fabsf(fabsf(dif.y) - fabsf(d) * 2.f) * sign(dif.y));
}
}
pts = { p1, p1a, p1b, p2 };
ptCount = 4;
}
}
float highLightFactor = factor * (highlightCons ? 2.0f : 1.f);
for (int pass = 0; pass < 2; pass++)
{
drawList->AddPolyline(pts.data(), ptCount, pass ? col : 0xFF000000, false, (pass ? options.mLineThickness : (options.mLineThickness * 1.5f)) * highLightFactor);
}
}
}
}
static void HandleQuadSelection(Delegate& delegate, ImDrawList* drawList, const ImVec2 offset, const float factor, ImRect contentRect, const Options& options)
{
if (!options.mAllowQuadSelection)
{
return;
}
ImGuiIO& io = ImGui::GetIO();
static ImVec2 quadSelectPos;
//auto& nodes = delegate->GetNodes();
auto nodeCount = delegate.GetNodeCount();
if (nodeOperation == NO_QuadSelecting && ImGui::IsWindowFocused())
{
const ImVec2 bmin = ImMin(quadSelectPos, io.MousePos);
const ImVec2 bmax = ImMax(quadSelectPos, io.MousePos);
drawList->AddRectFilled(bmin, bmax, options.mQuadSelection, 1.f);
drawList->AddRect(bmin, bmax, options.mQuadSelectionBorder, 1.f);
if (!io.MouseDown[0])
{
if (!io.KeyCtrl && !io.KeyShift)
{
for (size_t nodeIndex = 0; nodeIndex < nodeCount; nodeIndex++)
{
delegate.SelectNode(nodeIndex, false);
}
}
nodeOperation = NO_None;
ImRect selectionRect(bmin, bmax);
for (unsigned int nodeIndex = 0; nodeIndex < nodeCount; nodeIndex++)
{
const auto node = delegate.GetNode(nodeIndex);
ImVec2 nodeRectangleMin = offset + node.mRect.Min * factor;
ImVec2 nodeRectangleMax = nodeRectangleMin + node.mRect.GetSize() * factor;
if (selectionRect.Overlaps(ImRect(nodeRectangleMin, nodeRectangleMax)))
{
if (io.KeyCtrl)
{
delegate.SelectNode(nodeIndex, false);
}
else
{
delegate.SelectNode(nodeIndex, true);
}
}
else
{
if (!io.KeyShift)
{
delegate.SelectNode(nodeIndex, false);
}
}
}
}
}
else if (nodeOperation == NO_None && io.MouseDown[0] && ImGui::IsWindowFocused() &&
contentRect.Contains(io.MousePos))
{
nodeOperation = NO_QuadSelecting;
quadSelectPos = io.MousePos;
}
}
static bool HandleConnections(ImDrawList* drawList,
NodeIndex nodeIndex,
const ImVec2 offset,
const float factor,
Delegate& delegate,
const Options& options,
bool bDrawOnly,
SlotIndex& inputSlotOver,
SlotIndex& outputSlotOver,
const bool inMinimap)
{
static NodeIndex editingNodeIndex;
static SlotIndex editingSlotIndex;
ImGuiIO& io = ImGui::GetIO();
const auto node = delegate.GetNode(nodeIndex);
const auto nodeTemplate = delegate.GetTemplate(node.mTemplateIndex);
const auto linkCount = delegate.GetLinkCount();
size_t InputsCount = nodeTemplate.mInputCount;
size_t OutputsCount = nodeTemplate.mOutputCount;
inputSlotOver = -1;
outputSlotOver = -1;
// draw/use inputs/outputs
bool hoverSlot = false;
for (int i = 0; i < 2; i++)
{
float closestDistance = FLT_MAX;
SlotIndex closestConn = -1;
ImVec2 closestTextPos;
ImVec2 closestPos;
const size_t slotCount[2] = {InputsCount, OutputsCount};
for (SlotIndex slotIndex = 0; slotIndex < slotCount[i]; slotIndex++)
{
const char** con = i ? nodeTemplate.mOutputNames : nodeTemplate.mInputNames;
const char* conText = (con && con[slotIndex]) ? con[slotIndex] : "";
ImVec2 p =
offset + (i ? GetOutputSlotPos(delegate, node, slotIndex, factor) : GetInputSlotPos(delegate, node, slotIndex, factor));
float distance = Distance(p, io.MousePos);
bool overCon = (nodeOperation == NO_None || nodeOperation == NO_EditingLink) &&
(distance < options.mNodeSlotRadius * 2.f) && (distance < closestDistance);
ImVec2 textSize;
textSize = ImGui::CalcTextSize(conText);
ImVec2 textPos =
p + ImVec2(-options.mNodeSlotRadius * (i ? -1.f : 1.f) * (overCon ? 3.f : 2.f) - (i ? 0 : textSize.x),
-textSize.y / 2);
ImRect nodeRect = GetNodeRect(node, factor);
if (!inMinimap && (overCon || (nodeRect.Contains(io.MousePos - offset) && closestConn == -1 &&
(editingInput == (i != 0)) && nodeOperation == NO_EditingLink)))
{
closestDistance = distance;
closestConn = slotIndex;
closestTextPos = textPos;
closestPos = p;
if (i)
{
outputSlotOver = slotIndex;
}
else
{
inputSlotOver = slotIndex;
}
}
else
{
const ImU32* slotColorSource = i ? nodeTemplate.mOutputColors : nodeTemplate.mInputColors;
const ImU32 slotColor = slotColorSource ? slotColorSource[slotIndex] : options.mDefaultSlotColor;
drawList->AddCircleFilled(p, options.mNodeSlotRadius, IM_COL32(0, 0, 0, 200));
drawList->AddCircleFilled(p, options.mNodeSlotRadius * 0.75f, slotColor);
if (!options.mDrawIONameOnHover)
{
drawList->AddText(io.FontDefault, 14, textPos + ImVec2(2, 2), IM_COL32(0, 0, 0, 255), conText);
drawList->AddText(io.FontDefault, 14, textPos, IM_COL32(150, 150, 150, 255), conText);
}
}
}
if (closestConn != -1)
{
const char** con = i ? nodeTemplate.mOutputNames : nodeTemplate.mInputNames;
const char* conText = (con && con[closestConn]) ? con[closestConn] : "";
const ImU32* slotColorSource = i ? nodeTemplate.mOutputColors : nodeTemplate.mInputColors;
const ImU32 slotColor = slotColorSource ? slotColorSource[closestConn] : options.mDefaultSlotColor;
hoverSlot = true;
drawList->AddCircleFilled(closestPos, options.mNodeSlotRadius * options.mNodeSlotHoverFactor * 0.75f, IM_COL32(0, 0, 0, 200));
drawList->AddCircleFilled(closestPos, options.mNodeSlotRadius * options.mNodeSlotHoverFactor, slotColor);
drawList->AddText(io.FontDefault, 16, closestTextPos + ImVec2(1, 1), IM_COL32(0, 0, 0, 255), conText);
drawList->AddText(io.FontDefault, 16, closestTextPos, IM_COL32(250, 250, 250, 255), conText);
bool inputToOutput = (!editingInput && !i) || (editingInput && i);
if (nodeOperation == NO_EditingLink && !io.MouseDown[0] && !bDrawOnly)
{
if (inputToOutput)
{
// check loopback
Link nl;
if (editingInput)
nl = Link{nodeIndex, closestConn, editingNodeIndex, editingSlotIndex};
else
nl = Link{editingNodeIndex, editingSlotIndex, nodeIndex, closestConn};
if (!delegate.AllowedLink(nl.mOutputNodeIndex, nl.mInputNodeIndex))
{
break;
}
bool alreadyExisting = false;
for (size_t linkIndex = 0; linkIndex < linkCount; linkIndex++)
{
const auto link = delegate.GetLink(linkIndex);
if (!memcmp(&link, &nl, sizeof(Link)))
{
alreadyExisting = true;
break;
}
}
if (!alreadyExisting)
{
for (unsigned int linkIndex = 0; linkIndex < linkCount; linkIndex++)
{
const auto link = delegate.GetLink(linkIndex);
if (link.mOutputNodeIndex == nl.mOutputNodeIndex && link.mOutputSlotIndex == nl.mOutputSlotIndex)
{
delegate.DelLink(linkIndex);
break;
}
}
delegate.AddLink(nl.mInputNodeIndex, nl.mInputSlotIndex, nl.mOutputNodeIndex, nl.mOutputSlotIndex);
}
}
}
// when ImGui::IsWindowHovered() && !ImGui::IsAnyItemActive() is uncommented, one can't click the node
// input/output when mouse is over the node itself.
if (nodeOperation == NO_None &&
/*ImGui::IsWindowHovered() && !ImGui::IsAnyItemActive() &&*/ io.MouseClicked[0] && !bDrawOnly)
{
nodeOperation = NO_EditingLink;
editingInput = i == 0;
editingNodeSource = closestPos;
editingNodeIndex = nodeIndex;
editingSlotIndex = closestConn;
if (editingInput)
{
// remove existing link
for (unsigned int linkIndex = 0; linkIndex < linkCount; linkIndex++)
{
const auto link = delegate.GetLink(linkIndex);
if (link.mOutputNodeIndex == nodeIndex && link.mOutputSlotIndex == closestConn)
{
delegate.DelLink(linkIndex);
break;
}
}
}
}
}
}
return hoverSlot;
}
static void DrawGrid(ImDrawList* drawList, ImVec2 windowPos, const ViewState& viewState, const ImVec2 canvasSize, ImU32 gridColor, ImU32 gridColor2, float gridSize)
{
float gridSpace = gridSize * viewState.mFactor;
int divx = static_cast<int>(-viewState.mPosition.x / gridSize);
int divy = static_cast<int>(-viewState.mPosition.y / gridSize);
for (float x = fmodf(viewState.mPosition.x * viewState.mFactor, gridSpace); x < canvasSize.x; x += gridSpace, divx ++)
{
bool tenth = !(divx % 10);
drawList->AddLine(ImVec2(x, 0.0f) + windowPos, ImVec2(x, canvasSize.y) + windowPos, tenth ? gridColor2 : gridColor);
}
for (float y = fmodf(viewState.mPosition.y * viewState.mFactor, gridSpace); y < canvasSize.y; y += gridSpace, divy ++)
{
bool tenth = !(divy % 10);
drawList->AddLine(ImVec2(0.0f, y) + windowPos, ImVec2(canvasSize.x, y) + windowPos, tenth ? gridColor2 : gridColor);
}
}
// return true if node is hovered
static bool DrawNode(ImDrawList* drawList,
NodeIndex nodeIndex,
const ImVec2 offset,
const float factor,
Delegate& delegate,
bool overInput,
const Options& options,
const bool inMinimap,
const ImRect& viewPort)
{
ImGuiIO& io = ImGui::GetIO();
const auto node = delegate.GetNode(nodeIndex);
IM_ASSERT((node.mRect.GetWidth() != 0.f) && (node.mRect.GetHeight() != 0.f) && "Nodes must have a non-zero rect.");
const auto nodeTemplate = delegate.GetTemplate(node.mTemplateIndex);
const ImVec2 nodeRectangleMin = offset + node.mRect.Min * factor;
const bool old_any_active = ImGui::IsAnyItemActive();
ImGui::SetCursorScreenPos(nodeRectangleMin);
const ImVec2 nodeSize = node.mRect.GetSize() * factor;
// test nested IO
drawList->ChannelsSetCurrent(1); // Background
const size_t InputsCount = nodeTemplate.mInputCount;
const size_t OutputsCount = nodeTemplate.mOutputCount;
/*
for (int i = 0; i < 2; i++)
{
const size_t slotCount[2] = {InputsCount, OutputsCount};
for (size_t slotIndex = 0; slotIndex < slotCount[i]; slotIndex++)
{
const char* con = i ? nodeTemplate.mOutputNames[slotIndex] : nodeTemplate.mInputNames[slotIndex];//node.mOutputs[slot_idx] : node->mInputs[slot_idx];
if (!delegate->IsIOPinned(nodeIndex, slot_idx, i == 1))
{
}
continue;
ImVec2 p = offset + (i ? GetOutputSlotPos(delegate, node, slotIndex, factor) : GetInputSlotPos(delegate, node, slotIndex, factor));
const float arc = 28.f * (float(i) * 0.3f + 1.0f) * (i ? 1.f : -1.f);
const float ofs = 0.f;
ImVec2 pts[3] = {p + ImVec2(arc + ofs, 0.f), p + ImVec2(0.f + ofs, -arc), p + ImVec2(0.f + ofs, arc)};
drawList->AddTriangleFilled(pts[0], pts[1], pts[2], i ? 0xFFAA5030 : 0xFF30AA50);
drawList->AddTriangle(pts[0], pts[1], pts[2], 0xFF000000, 2.f);
}
}
*/
ImGui::SetCursorScreenPos(nodeRectangleMin);
float maxHeight = ImMin(viewPort.Max.y, nodeRectangleMin.y + nodeSize.y) - nodeRectangleMin.y;
float maxWidth = ImMin(viewPort.Max.x, nodeRectangleMin.x + nodeSize.x) - nodeRectangleMin.x;
ImGui::InvisibleButton("node", ImVec2(maxWidth, maxHeight));
// must be called right after creating the control we want to be able to move
bool nodeMovingActive = ImGui::IsItemActive();
// Save the size of what we have emitted and whether any of the widgets are being used
bool nodeWidgetsActive = (!old_any_active && ImGui::IsAnyItemActive());
ImVec2 nodeRectangleMax = nodeRectangleMin + nodeSize;
bool nodeHovered = false;
if (ImGui::IsItemHovered() && nodeOperation == NO_None && !overInput)
{
nodeHovered = true;
}
if (ImGui::IsWindowFocused())
{
if ((nodeWidgetsActive || nodeMovingActive) && !inMinimap)
{
if (!node.mSelected)
{
if (!io.KeyShift)
{
const auto nodeCount = delegate.GetNodeCount();
for (size_t i = 0; i < nodeCount; i++)
{
delegate.SelectNode(i, false);
}
}
delegate.SelectNode(nodeIndex, true);
}
}
}
if (nodeMovingActive && io.MouseDown[0] && nodeHovered && !inMinimap)
{
if (nodeOperation != NO_MovingNodes)
{
nodeOperation = NO_MovingNodes;
}
}
const bool currentSelectedNode = node.mSelected;
const ImU32 node_bg_color = nodeHovered ? nodeTemplate.mBackgroundColorOver : nodeTemplate.mBackgroundColor;
drawList->AddRect(nodeRectangleMin,
nodeRectangleMax,
currentSelectedNode ? options.mSelectedNodeBorderColor : options.mNodeBorderColor,
options.mRounding,
ImDrawFlags_RoundCornersAll,
currentSelectedNode ? options.mBorderSelectionThickness : options.mBorderThickness);
ImVec2 imgPos = nodeRectangleMin + ImVec2(14, 25);
ImVec2 imgSize = nodeRectangleMax + ImVec2(-5, -5) - imgPos;
float imgSizeComp = std::min(imgSize.x, imgSize.y);
drawList->AddRectFilled(nodeRectangleMin, nodeRectangleMax, node_bg_color, options.mRounding);
/*float progress = delegate->NodeProgress(nodeIndex);
if (progress > FLT_EPSILON && progress < 1.f - FLT_EPSILON)
{
ImVec2 progressLineA = nodeRectangleMax - ImVec2(nodeSize.x - 2.f, 3.f);
ImVec2 progressLineB = progressLineA + ImVec2(nodeSize.x * factor - 4.f, 0.f);
drawList->AddLine(progressLineA, progressLineB, 0xFF400000, 3.f);
drawList->AddLine(progressLineA, ImLerp(progressLineA, progressLineB, progress), 0xFFFF0000, 3.f);
}*/
ImVec2 imgPosMax = imgPos + ImVec2(imgSizeComp, imgSizeComp);
//ImVec2 imageSize = delegate->GetEvaluationSize(nodeIndex);
/*float imageRatio = 1.f;
if (imageSize.x > 0.f && imageSize.y > 0.f)
{
imageRatio = imageSize.y / imageSize.x;
}
ImVec2 quadSize = imgPosMax - imgPos;
ImVec2 marge(0.f, 0.f);
if (imageRatio > 1.f)
{
marge.x = (quadSize.x - quadSize.y / imageRatio) * 0.5f;
}
else
{
marge.y = (quadSize.y - quadSize.y * imageRatio) * 0.5f;
}*/
//delegate->DrawNodeImage(drawList, ImRect(imgPos, imgPosMax), marge, nodeIndex);
drawList->AddRectFilled(nodeRectangleMin,
ImVec2(nodeRectangleMax.x, nodeRectangleMin.y + 20),
nodeTemplate.mHeaderColor, options.mRounding);
drawList->PushClipRect(nodeRectangleMin, ImVec2(nodeRectangleMax.x, nodeRectangleMin.y + 20), true);
drawList->AddText(nodeRectangleMin + ImVec2(2, 2), IM_COL32(0, 0, 0, 255), node.mName);
drawList->PopClipRect();
ImRect customDrawRect(nodeRectangleMin + ImVec2(options.mRounding, 20 + options.mRounding), nodeRectangleMax - ImVec2(options.mRounding, options.mRounding));
if (customDrawRect.Max.y > customDrawRect.Min.y && customDrawRect.Max.x > customDrawRect.Min.x)
{
delegate.CustomDraw(drawList, customDrawRect, nodeIndex);
}
/*
const ImTextureID bmpInfo = (ImTextureID)(uint64_t)delegate->GetBitmapInfo(nodeIndex).idx;
if (bmpInfo)
{
ImVec2 bmpInfoPos(nodeRectangleMax - ImVec2(26, 12));
ImVec2 bmpInfoSize(20, 20);
if (delegate->NodeIsCompute(nodeIndex))
{
drawList->AddImageQuad(bmpInfo,
bmpInfoPos,
bmpInfoPos + ImVec2(bmpInfoSize.x, 0.f),
bmpInfoPos + bmpInfoSize,
bmpInfoPos + ImVec2(0., bmpInfoSize.y));
}
else if (delegate->NodeIs2D(nodeIndex))
{
drawList->AddImageQuad(bmpInfo,
bmpInfoPos,
bmpInfoPos + ImVec2(bmpInfoSize.x, 0.f),
bmpInfoPos + bmpInfoSize,
bmpInfoPos + ImVec2(0., bmpInfoSize.y));
}
else if (delegate->NodeIsCubemap(nodeIndex))
{
drawList->AddImageQuad(bmpInfo,
bmpInfoPos + ImVec2(0., bmpInfoSize.y),
bmpInfoPos + bmpInfoSize,
bmpInfoPos + ImVec2(bmpInfoSize.x, 0.f),
bmpInfoPos);
}
}*/
return nodeHovered;
}
bool DrawMiniMap(ImDrawList* drawList, Delegate& delegate, ViewState& viewState, const Options& options, const ImVec2 windowPos, const ImVec2 canvasSize)
{
if (Distance(options.mMinimap.Min, options.mMinimap.Max) <= FLT_EPSILON)
{
return false;
}
const size_t nodeCount = delegate.GetNodeCount();
if (!nodeCount)
{
return false;
}
ImVec2 min(FLT_MAX, FLT_MAX);
ImVec2 max(-FLT_MAX, -FLT_MAX);
const ImVec2 margin(50, 50);
for (NodeIndex nodeIndex = 0; nodeIndex < nodeCount; nodeIndex++)
{
const Node& node = delegate.GetNode(nodeIndex);
min = ImMin(min, node.mRect.Min - margin);
min = ImMin(min, node.mRect.Max + margin);
max = ImMax(max, node.mRect.Min - margin);
max = ImMax(max, node.mRect.Max + margin);
}
// add view in world space
const ImVec2 worldSizeView = canvasSize / viewState.mFactor;
const ImVec2 viewMin(-viewState.mPosition.x, -viewState.mPosition.y);
const ImVec2 viewMax = viewMin + worldSizeView;
min = ImMin(min, viewMin);
max = ImMax(max, viewMax);
const ImVec2 nodesSize = max - min;
const ImVec2 middleWorld = (min + max) * 0.5f;
const ImVec2 minScreen = windowPos + options.mMinimap.Min * canvasSize;
const ImVec2 maxScreen = windowPos + options.mMinimap.Max * canvasSize;
const ImVec2 viewSize = maxScreen - minScreen;
const ImVec2 middleScreen = (minScreen + maxScreen) * 0.5f;
const float ratioY = viewSize.y / nodesSize.y;
const float ratioX = viewSize.x / nodesSize.x;
const float factor = ImMin(ImMin(ratioY, ratioX), 1.f);
drawList->AddRectFilled(minScreen, maxScreen, IM_COL32(30, 30, 30, 200), 3, ImDrawFlags_RoundCornersAll);
for (NodeIndex nodeIndex = 0; nodeIndex < nodeCount; nodeIndex++)
{
const Node& node = delegate.GetNode(nodeIndex);
const auto nodeTemplate = delegate.GetTemplate(node.mTemplateIndex);
ImRect rect = node.mRect;
rect.Min -= middleWorld;
rect.Min *= factor;
rect.Min += middleScreen;
rect.Max -= middleWorld;
rect.Max *= factor;
rect.Max += middleScreen;
drawList->AddRectFilled(rect.Min, rect.Max, nodeTemplate.mBackgroundColor, 1, ImDrawFlags_RoundCornersAll);
if (node.mSelected)
{
drawList->AddRect(rect.Min, rect.Max, options.mSelectedNodeBorderColor, 1, ImDrawFlags_RoundCornersAll);
}
}
// add view
ImVec2 viewMinScreen = (viewMin - middleWorld) * factor + middleScreen;
ImVec2 viewMaxScreen = (viewMax - middleWorld) * factor + middleScreen;
drawList->AddRectFilled(viewMinScreen, viewMaxScreen, IM_COL32(255, 255, 255, 32), 1, ImDrawFlags_RoundCornersAll);
drawList->AddRect(viewMinScreen, viewMaxScreen, IM_COL32(255, 255, 255, 128), 1, ImDrawFlags_RoundCornersAll);
ImGuiIO& io = ImGui::GetIO();
const bool mouseInMinimap = ImRect(minScreen, maxScreen).Contains(io.MousePos);
if (mouseInMinimap && io.MouseClicked[0])
{
const ImVec2 clickedRatio = (io.MousePos - minScreen) / viewSize;
const ImVec2 worldPosCenter = ImVec2(ImLerp(min.x, max.x, clickedRatio.x), ImLerp(min.y, max.y, clickedRatio.y));
ImVec2 worldPosViewMin = worldPosCenter - worldSizeView * 0.5;
ImVec2 worldPosViewMax = worldPosCenter + worldSizeView * 0.5;
if (worldPosViewMin.x < min.x)
{
worldPosViewMin.x = min.x;
worldPosViewMax.x = worldPosViewMin.x + worldSizeView.x;
}
if (worldPosViewMin.y < min.y)
{
worldPosViewMin.y = min.y;
worldPosViewMax.y = worldPosViewMin.y + worldSizeView.y;
}
if (worldPosViewMax.x > max.x)
{
worldPosViewMax.x = max.x;
worldPosViewMin.x = worldPosViewMax.x - worldSizeView.x;
}
if (worldPosViewMax.y > max.y)
{
worldPosViewMax.y = max.y;
worldPosViewMin.y = worldPosViewMax.y - worldSizeView.y;
}
viewState.mPosition = ImVec2(-worldPosViewMin.x, -worldPosViewMin.y);
}
return mouseInMinimap;
}
void Show(Delegate& delegate, const Options& options, ViewState& viewState, bool enabled, FitOnScreen* fit)
{
ImGui::PushStyleVar(ImGuiStyleVar_ChildBorderSize, 0.f);
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(0.f, 0.f));
ImGui::PushStyleVar(ImGuiStyleVar_FrameBorderSize, 0.f);
const ImVec2 windowPos = ImGui::GetCursorScreenPos();
const ImVec2 canvasSize = ImGui::GetContentRegionAvail();
const ImVec2 scrollRegionLocalPos(0, 0);
ImRect regionRect(windowPos, windowPos + canvasSize);
HandleZoomScroll(regionRect, viewState, options);
ImVec2 offset = ImGui::GetCursorScreenPos() + viewState.mPosition * viewState.mFactor;
captureOffset = viewState.mPosition * viewState.mFactor;
//ImGui::InvisibleButton("GraphEditorButton", canvasSize);
ImGui::BeginChild(71711, canvasSize, ImGuiChildFlags_FrameStyle);
ImGui::SetCursorPos(windowPos);
ImGui::BeginGroup();
ImGuiIO& io = ImGui::GetIO();
// Create our child canvas
ImGui::PushStyleVar(ImGuiStyleVar_FramePadding, ImVec2(1, 1));
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(0, 0));
ImGui::PushStyleColor(ImGuiCol_ChildBg, IM_COL32(30, 30, 30, 200));
ImDrawList* drawList = ImGui::GetWindowDrawList();
ImGui::PushClipRect(regionRect.Min, regionRect.Max, true);
drawList->AddRectFilled(windowPos, windowPos + canvasSize, options.mBackgroundColor);
// Background or Display grid
if (options.mRenderGrid)
{
DrawGrid(drawList, windowPos, viewState, canvasSize, options.mGridColor, options.mGridColor2, options.mGridSize);
}
// Fit view
if (fit && ((*fit == Fit_AllNodes) || (*fit == Fit_SelectedNodes)))
{
FitNodes(delegate, viewState, canvasSize, (*fit == Fit_SelectedNodes));
}
if (enabled)
{
static NodeIndex hoveredNode = -1;
// Display links
drawList->ChannelsSplit(3);
// minimap
drawList->ChannelsSetCurrent(2); // minimap
const bool inMinimap = DrawMiniMap(drawList, delegate, viewState, options, windowPos, canvasSize);
// Focus rectangle
if (ImGui::IsWindowFocused())
{
drawList->AddRect(regionRect.Min, regionRect.Max, options.mFrameFocus, 1.f, 0, 2.f);
}
drawList->ChannelsSetCurrent(1); // Background
// Links
DisplayLinks(delegate, drawList, offset, viewState.mFactor, regionRect, hoveredNode, options);
// edit node link
if (nodeOperation == NO_EditingLink)
{
ImVec2 p1 = editingNodeSource;
ImVec2 p2 = io.MousePos;
drawList->AddLine(p1, p2, IM_COL32(200, 200, 200, 255), 3.0f);
}
// Display nodes
drawList->PushClipRect(regionRect.Min, regionRect.Max, true);
hoveredNode = -1;
SlotIndex inputSlotOver = -1;
SlotIndex outputSlotOver = -1;
NodeIndex nodeOver = -1;
const auto nodeCount = delegate.GetNodeCount();
for (int i = 0; i < 2; i++)
{
for (NodeIndex nodeIndex = 0; nodeIndex < nodeCount; nodeIndex++)
{
//const auto* node = &nodes[nodeIndex];
const auto node = delegate.GetNode(nodeIndex);
if (node.mSelected != (i != 0))
{
continue;
}
// node view clipping
ImRect nodeRect = GetNodeRect(node, viewState.mFactor);
nodeRect.Min += offset;
nodeRect.Max += offset;
if (!regionRect.Overlaps(nodeRect))
{
continue;
}
ImGui::PushID((int)nodeIndex);
SlotIndex inputSlot = -1;
SlotIndex outputSlot = -1;
bool overInput = (!inMinimap) && HandleConnections(drawList, nodeIndex, offset, viewState.mFactor, delegate, options, false, inputSlot, outputSlot, inMinimap);
// shadow
/*
ImVec2 shadowOffset = ImVec2(30, 30);
ImVec2 shadowPivot = (nodeRect.Min + nodeRect.Max) /2.f;
ImVec2 shadowPointMiddle = shadowPivot + shadowOffset;
ImVec2 shadowPointTop = ImVec2(shadowPivot.x, nodeRect.Min.y) + shadowOffset;
ImVec2 shadowPointBottom = ImVec2(shadowPivot.x, nodeRect.Max.y) + shadowOffset;
ImVec2 shadowPointLeft = ImVec2(nodeRect.Min.x, shadowPivot.y) + shadowOffset;
ImVec2 shadowPointRight = ImVec2(nodeRect.Max.x, shadowPivot.y) + shadowOffset;
// top left
drawList->AddRectFilledMultiColor(nodeRect.Min + shadowOffset, shadowPointMiddle, IM_COL32(0 ,0, 0, 0), IM_COL32(0,0,0,0), IM_COL32(0, 0, 0, 255), IM_COL32(0, 0, 0, 0));
// top right
drawList->AddRectFilledMultiColor(shadowPointTop, shadowPointRight, IM_COL32(0 ,0, 0, 0), IM_COL32(0,0,0,0), IM_COL32(0, 0, 0, 0), IM_COL32(0, 0, 0, 255));
// bottom left
drawList->AddRectFilledMultiColor(shadowPointLeft, shadowPointBottom, IM_COL32(0 ,0, 0, 0), IM_COL32(0, 0, 0, 255), IM_COL32(0, 0, 0, 0), IM_COL32(0,0,0,0));
// bottom right
drawList->AddRectFilledMultiColor(shadowPointMiddle, nodeRect.Max + shadowOffset, IM_COL32(0, 0, 0, 255), IM_COL32(0 ,0, 0, 0), IM_COL32(0,0,0,0), IM_COL32(0, 0, 0, 0));
*/
if (DrawNode(drawList, nodeIndex, offset, viewState.mFactor, delegate, overInput, options, inMinimap, regionRect))
{
hoveredNode = nodeIndex;
}
HandleConnections(drawList, nodeIndex, offset, viewState.mFactor, delegate, options, true, inputSlot, outputSlot, inMinimap);
if (inputSlot != -1 || outputSlot != -1)
{
inputSlotOver = inputSlot;
outputSlotOver = outputSlot;
nodeOver = nodeIndex;
}
ImGui::PopID();
}
}
drawList->PopClipRect();
if (nodeOperation == NO_MovingNodes)
{
if (ImGui::IsMouseDragging(0, 1))
{
ImVec2 delta = io.MouseDelta / viewState.mFactor;
if (fabsf(delta.x) >= 1.f || fabsf(delta.y) >= 1.f)
{
delegate.MoveSelectedNodes(delta);
}
}
}
drawList->ChannelsSetCurrent(0);
// quad selection
if (!inMinimap)
{
HandleQuadSelection(delegate, drawList, offset, viewState.mFactor, regionRect, options);
}
drawList->ChannelsMerge();
// releasing mouse button means it's done in any operation
if (nodeOperation == NO_PanView)
{
if (!io.MouseDown[2])
{
nodeOperation = NO_None;
}
}
else if (nodeOperation != NO_None && !io.MouseDown[0])
{
nodeOperation = NO_None;
}
// right click
if (!inMinimap && nodeOperation == NO_None && regionRect.Contains(io.MousePos) &&
(ImGui::IsMouseClicked(1) /*|| (ImGui::IsWindowFocused() && ImGui::IsKeyPressedMap(ImGuiKey_Tab))*/))
{
delegate.RightClick(nodeOver, inputSlotOver, outputSlotOver);
}
// Scrolling
if (ImGui::IsWindowHovered() && !ImGui::IsAnyItemActive() && io.MouseClicked[2] && nodeOperation == NO_None)
{
nodeOperation = NO_PanView;
}
if (nodeOperation == NO_PanView)
{
viewState.mPosition += io.MouseDelta / viewState.mFactor;
}
}
ImGui::PopClipRect();
ImGui::PopStyleColor(1);
ImGui::PopStyleVar(2);
ImGui::EndGroup();
ImGui::EndChild();
ImGui::PopStyleVar(3);
// change fit to none
if (fit)
{
*fit = Fit_None;
}
}
bool EditOptions(Options& options)
{
bool updated = false;
if (ImGui::CollapsingHeader("Colors", nullptr))
{
ImColor backgroundColor(options.mBackgroundColor);
ImColor gridColor(options.mGridColor);
ImColor selectedNodeBorderColor(options.mSelectedNodeBorderColor);
ImColor nodeBorderColor(options.mNodeBorderColor);
ImColor quadSelection(options.mQuadSelection);
ImColor quadSelectionBorder(options.mQuadSelectionBorder);
ImColor defaultSlotColor(options.mDefaultSlotColor);
ImColor frameFocus(options.mFrameFocus);
updated |= ImGui::ColorEdit4("Background", (float*)&backgroundColor);
updated |= ImGui::ColorEdit4("Grid", (float*)&gridColor);
updated |= ImGui::ColorEdit4("Selected Node Border", (float*)&selectedNodeBorderColor);
updated |= ImGui::ColorEdit4("Node Border", (float*)&nodeBorderColor);
updated |= ImGui::ColorEdit4("Quad Selection", (float*)&quadSelection);
updated |= ImGui::ColorEdit4("Quad Selection Border", (float*)&quadSelectionBorder);
updated |= ImGui::ColorEdit4("Default Slot", (float*)&defaultSlotColor);
updated |= ImGui::ColorEdit4("Frame when has focus", (float*)&frameFocus);
options.mBackgroundColor = backgroundColor;
options.mGridColor = gridColor;
options.mSelectedNodeBorderColor = selectedNodeBorderColor;
options.mNodeBorderColor = nodeBorderColor;
options.mQuadSelection = quadSelection;
options.mQuadSelectionBorder = quadSelectionBorder;
options.mDefaultSlotColor = defaultSlotColor;
options.mFrameFocus = frameFocus;
}
if (ImGui::CollapsingHeader("Options", nullptr))
{
updated |= ImGui::InputFloat4("Minimap", &options.mMinimap.Min.x);
updated |= ImGui::InputFloat("Line Thickness", &options.mLineThickness);
updated |= ImGui::InputFloat("Grid Size", &options.mGridSize);
updated |= ImGui::InputFloat("Rounding", &options.mRounding);
updated |= ImGui::InputFloat("Zoom Ratio", &options.mZoomRatio);
updated |= ImGui::InputFloat("Zoom Lerp Factor", &options.mZoomLerpFactor);
updated |= ImGui::InputFloat("Border Selection Thickness", &options.mBorderSelectionThickness);
updated |= ImGui::InputFloat("Border Thickness", &options.mBorderThickness);
updated |= ImGui::InputFloat("Slot Radius", &options.mNodeSlotRadius);
updated |= ImGui::InputFloat("Slot Hover Factor", &options.mNodeSlotHoverFactor);
updated |= ImGui::InputFloat2("Zoom min/max", &options.mMinZoom);
updated |= ImGui::InputFloat("Slot Hover Factor", &options.mSnap);
if (ImGui::RadioButton("Curved Links", options.mDisplayLinksAsCurves))
{
options.mDisplayLinksAsCurves = !options.mDisplayLinksAsCurves;
updated = true;
}
if (ImGui::RadioButton("Straight Links", !options.mDisplayLinksAsCurves))
{
options.mDisplayLinksAsCurves = !options.mDisplayLinksAsCurves;
updated = true;
}
updated |= ImGui::Checkbox("Allow Quad Selection", &options.mAllowQuadSelection);
updated |= ImGui::Checkbox("Render Grid", &options.mRenderGrid);
updated |= ImGui::Checkbox("Draw IO names on hover", &options.mDrawIONameOnHover);
}
return updated;
}
} // namespace