Added NodeConnectionDebug tool.

This commit is contained in:
Martin Felis 2025-03-02 12:48:39 +01:00
parent 55bcd9cd99
commit a2e8de0b70
2 changed files with 201 additions and 38 deletions

View File

@ -24,9 +24,29 @@ struct EditorState {
bool isGraphLoadedThisFrame = false; bool isGraphLoadedThisFrame = false;
ImVec2 mousePopupStart = {}; ImVec2 mousePopupStart = {};
char statusLine[1024] = {};
};
struct NodeConnectionDebugState {
struct SocketInfo {
ax::NodeEditor::PinId pin = {};
int nodeId = {};
const AnimNodeResource* nodeResource = {};
int socketId = {};
const Socket* socket = {};
};
SocketInfo sourceSocket = {};
SocketInfo targetSocket = {};
void Reset() {
sourceSocket = {};
targetSocket = {};
}
}; };
static EditorState sEditorState; static EditorState sEditorState;
static NodeConnectionDebugState sNodeConnectionDebugState;
constexpr int cPinIconSize = 24; constexpr int cPinIconSize = 24;
@ -503,56 +523,98 @@ void AnimGraphEditorBreadcrumbNavigation() {
} }
} }
void HandleConnectionCreation( void HandleConnectionCreation(BlendTreeResource& current_blend_tree) {
BlendTreeResource& current_blend_tree) { // Create Connections
if (ax::NodeEditor::BeginCreate()) { if (ax::NodeEditor::BeginCreate()) {
ax::NodeEditor::PinId input_pin_id, output_pin_id; ax::NodeEditor::PinId input_pin_id, output_pin_id;
if (ax::NodeEditor::QueryNewLink(&input_pin_id, &output_pin_id)) { if (ax::NodeEditor::QueryNewLink(&input_pin_id, &output_pin_id)) {
int source_node_index; ax::NodeEditor::PinId source_pin = input_pin_id;
int source_node_socket_index; ax::NodeEditor::PinId target_pin = output_pin_id;
int source_node_index = -1;
int source_node_socket_index = -1;
const AnimNodeResource* source_node = nullptr; const AnimNodeResource* source_node = nullptr;
const Socket* source_socket = nullptr; const Socket* source_socket = nullptr;
if (input_pin_id) { if (input_pin_id) {
if (IsPinInput(input_pin_id.Get())) {
source_pin = input_pin_id;
target_pin = output_pin_id;
} else {
target_pin = input_pin_id;
source_pin = output_pin_id;
}
}
if (output_pin_id) {
if (IsPinOutput(output_pin_id.Get())) {
target_pin = output_pin_id;
source_pin = input_pin_id;
} else {
source_pin = output_pin_id;
target_pin = input_pin_id;
}
}
if (source_pin) {
OutputPinIdToNodeIndexAndSocketIndex( OutputPinIdToNodeIndexAndSocketIndex(
input_pin_id.Get(), source_pin.Get(),
&source_node_index, &source_node_index,
&source_node_socket_index); &source_node_socket_index);
source_node = current_blend_tree.GetNode(source_node_index); source_node = current_blend_tree.GetNode(source_node_index);
if (source_node->m_virtual_socket_accessor->m_outputs.size()
< source_node_socket_index) { if (source_node != nullptr) {
source_node_socket_index = -1; if (source_node->m_virtual_socket_accessor->m_outputs.size()
} else { < source_node_socket_index) {
source_socket = current_blend_tree.GetNodeOutputSocketByIndex( source_node_socket_index = -1;
source_node, } else {
source_node_socket_index); source_socket = current_blend_tree.GetNodeOutputSocketByIndex(
source_node,
source_node_socket_index);
}
} }
} }
int target_node_index; int target_node_index = -1;
int target_node_socket_index; int target_node_socket_index = -1;
const AnimNodeResource* target_node = nullptr; const AnimNodeResource* target_node = nullptr;
const Socket* target_socket = nullptr; const Socket* target_socket = nullptr;
if (output_pin_id) { if (target_pin) {
InputPinIdToNodeIndexAndSocketIndex( InputPinIdToNodeIndexAndSocketIndex(
output_pin_id.Get(), target_pin.Get(),
&target_node_index, &target_node_index,
&target_node_socket_index); &target_node_socket_index);
target_node = current_blend_tree.GetNode(target_node_index); target_node = current_blend_tree.GetNode(target_node_index);
if (target_node->m_virtual_socket_accessor->m_inputs.size()
< target_node_socket_index) { if (target_node != nullptr) {
target_node_socket_index = -1; if (target_node->m_virtual_socket_accessor->m_inputs.size()
} else { < target_node_socket_index) {
target_socket = current_blend_tree.GetNodeInputSocketByIndex( target_node_socket_index = -1;
target_node, } else {
target_node_socket_index); target_socket = current_blend_tree.GetNodeInputSocketByIndex(
target_node,
target_node_socket_index);
}
} }
} }
if (input_pin_id && output_pin_id) { sNodeConnectionDebugState.sourceSocket.pin = source_pin;
sNodeConnectionDebugState.sourceSocket.nodeId = source_node_index;
sNodeConnectionDebugState.sourceSocket.nodeResource = source_node;
sNodeConnectionDebugState.sourceSocket.socketId =
source_node_socket_index;
sNodeConnectionDebugState.sourceSocket.socket = source_socket;
sNodeConnectionDebugState.targetSocket.pin = target_pin;
sNodeConnectionDebugState.targetSocket.nodeId = target_node_index;
sNodeConnectionDebugState.targetSocket.nodeResource = target_node;
sNodeConnectionDebugState.targetSocket.socketId =
target_node_socket_index;
sNodeConnectionDebugState.targetSocket.socket = target_socket;
if (source_pin && target_pin) {
if (source_socket == nullptr || target_socket == nullptr if (source_socket == nullptr || target_socket == nullptr
|| !current_blend_tree.IsConnectionValid( || !current_blend_tree.IsConnectionValid(
source_node, source_node,
@ -605,9 +667,12 @@ void BlendTreeRenderNodes(
current_blend_tree.GetNodeInputSockets(node_resource); current_blend_tree.GetNodeInputSockets(node_resource);
for (size_t j = 0, ni = node_inputs.size(); j < ni; j++) { for (size_t j = 0, ni = node_inputs.size(); j < ni; j++) {
Socket& socket = node_inputs[j]; Socket& socket = node_inputs[j];
builder.Input(NodeIndexAndSocketIndexToInputPinId( ax::NodeEditor::PinId input_pin = NodeIndexAndSocketIndexToInputPinId(
static_cast<int>(node_index), static_cast<int>(node_index),
static_cast<int>(j))); static_cast<int>(j));
builder.Input(input_pin);
assert(!input_pin.Invalid);
DrawSocketIcon( DrawSocketIcon(
socket.m_type, socket.m_type,
@ -679,19 +744,101 @@ void BlendTreeRenderConnections(BlendTreeResource& current_blend_tree) {
target_socket_pin_id); target_socket_pin_id);
} }
} }
void AnimGraphEditorDebugWidget() {
ImGui::Begin("Connection Debug Panel");
ImGui::BeginTable("Connection", 3);
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
ImGui::Text("Pin");
ImGui::TableNextColumn();
ImGui::Text("%x", sNodeConnectionDebugState.sourceSocket.pin.AsPointer());
ImGui::TableNextColumn();
ImGui::Text("%x", sNodeConnectionDebugState.targetSocket.pin.AsPointer());
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
ImGui::Text("Node");
ImGui::TableNextColumn();
if (sNodeConnectionDebugState.sourceSocket.nodeResource) {
ImGui::Text(
"%s (%p)",
sNodeConnectionDebugState.sourceSocket.nodeResource->m_node_type_name
.c_str(),
sNodeConnectionDebugState.sourceSocket.nodeResource);
}
ImGui::TableNextColumn();
if (sNodeConnectionDebugState.targetSocket.nodeResource) {
ImGui::Text(
"%s (%p)",
sNodeConnectionDebugState.targetSocket.nodeResource->m_node_type_name
.c_str(),
sNodeConnectionDebugState.targetSocket.nodeResource);
}
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
ImGui::Text("NodeId");
ImGui::TableNextColumn();
ImGui::Text("%d", sNodeConnectionDebugState.sourceSocket.nodeId);
ImGui::TableNextColumn();
ImGui::Text("%d", sNodeConnectionDebugState.targetSocket.nodeId);
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
ImGui::Text("Socket");
ImGui::TableNextColumn();
if (sNodeConnectionDebugState.sourceSocket.socket) {
ImGui::Text(
"%s (%p)",
sNodeConnectionDebugState.sourceSocket.socket->m_name.c_str(),
sNodeConnectionDebugState.sourceSocket.socket);
}
ImGui::TableNextColumn();
if (sNodeConnectionDebugState.targetSocket.socket) {
ImGui::Text(
"%s (%p)",
sNodeConnectionDebugState.targetSocket.socket->m_name.c_str(),
sNodeConnectionDebugState.targetSocket.socket);
}
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
ImGui::Text("SocketId");
ImGui::TableNextColumn();
ImGui::Text("%x", sNodeConnectionDebugState.sourceSocket.socketId);
ImGui::TableNextColumn();
ImGui::Text("%x", sNodeConnectionDebugState.targetSocket.socketId);
ImGui::EndTable();
ImGui::End();
}
void AnimGraphEditorUpdate(ax::NodeEditor::EditorContext* context) { void AnimGraphEditorUpdate(ax::NodeEditor::EditorContext* context) {
sEditorState.statusLine[0] = '\0';
sEditorState.statusLine[sizeof(sEditorState.statusLine) - 1] = '\0';
sNodeConnectionDebugState.Reset();
ax::NodeEditor::SetCurrentEditor(context); ax::NodeEditor::SetCurrentEditor(context);
AnimGraphEditorMenuBar(); AnimGraphEditorMenuBar();
AnimGraphEditorBreadcrumbNavigation(); AnimGraphEditorBreadcrumbNavigation();
ImGui::Columns(2); static ImGuiTableFlags flags =
ImGuiTableFlags_SizingStretchSame | ImGuiTableFlags_Resizable;
ImGui::BeginTable("GraphEditorWithSidebar", 2, flags);
ImGui::TableNextRow();
ImGui::TableSetColumnIndex(0);
// //
// Node editor canvas // Node editor canvas
// //
ax::NodeEditor::Begin("Graph Editor"); ImVec2 graph_size = ImGui::GetContentRegionAvail();
graph_size.y -= 20;
ax::NodeEditor::Begin("Graph Editor", graph_size);
AnimGraphResource* current_graph = AnimGraphResource* current_graph =
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]; sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex];
@ -700,19 +847,19 @@ void AnimGraphEditorUpdate(ax::NodeEditor::EditorContext* context) {
ax::NodeEditor::Utilities::BlueprintNodeBuilder builder; ax::NodeEditor::Utilities::BlueprintNodeBuilder builder;
BlendTreeRenderNodes(current_blend_tree, builder); BlendTreeRenderNodes(current_blend_tree, builder);
BlendTreeRenderConnections(current_blend_tree); BlendTreeRenderConnections(current_blend_tree);
HandleConnectionCreation(current_blend_tree); HandleConnectionCreation(current_blend_tree);
BlendTreeEditorNodePopup(); BlendTreeEditorNodePopup();
ax::NodeEditor::End(); ax::NodeEditor::End();
ImGui::Text("Status: %s", sEditorState.statusLine);
// //
// Sidebar // Sidebar
// //
ImGui::NextColumn(); ImGui::TableSetColumnIndex(1);
if (ax::NodeEditor::GetSelectedObjectCount() > 0) { if (ax::NodeEditor::GetSelectedObjectCount() > 0) {
ax::NodeEditor::NodeId selected_node_id = 0; ax::NodeEditor::NodeId selected_node_id = 0;
@ -727,7 +874,9 @@ void AnimGraphEditorUpdate(ax::NodeEditor::EditorContext* context) {
} }
} }
ImGui::Columns(1); ImGui::EndTable();
AnimGraphEditorDebugWidget();
// Clear flag, however it may be re-set further down when handling double // Clear flag, however it may be re-set further down when handling double
// clicking into subgraphs. // clicking into subgraphs.

View File

@ -13,6 +13,9 @@ struct SkinnedMesh;
struct AnimGraphBlendTree; struct AnimGraphBlendTree;
struct SyncTrack; struct SyncTrack;
constexpr int cMaxSocketsPerNode = 500;
constexpr int cMaxNodesPerGraph = 1000;
inline int GenerateInputAttributeId(int node_id, int input_index) { inline int GenerateInputAttributeId(int node_id, int input_index) {
return ((input_index + 1) << 14) + node_id; return ((input_index + 1) << 14) + node_id;
} }
@ -36,31 +39,42 @@ SplitOutputAttributeId(int attribute_id, int* node_id, int* output_index) {
inline int NodeIndexAndSocketIndexToInputPinId( inline int NodeIndexAndSocketIndexToInputPinId(
int node_index, int node_index,
int input_socket_index) { int input_socket_index) {
return node_index * 1000 + input_socket_index; return node_index * cMaxNodesPerGraph + input_socket_index;
} }
inline int NodeIndexAndSocketIndexToOutputPinId( inline int NodeIndexAndSocketIndexToOutputPinId(
int node_index, int node_index,
int output_socket_index) { int output_socket_index) {
return node_index * 1000 + 500 + output_socket_index; return node_index * cMaxNodesPerGraph + cMaxSocketsPerNode
+ output_socket_index;
} }
inline void InputPinIdToNodeIndexAndSocketIndex( inline void InputPinIdToNodeIndexAndSocketIndex(
unsigned long input_pin_id, unsigned long input_pin_id,
int* node_index, int* node_index,
int* socket_index) { int* socket_index) {
*socket_index = input_pin_id % 1000; *socket_index = static_cast<int>(input_pin_id) % cMaxNodesPerGraph;
*node_index = (input_pin_id - *socket_index) / 1000; *node_index =
(static_cast<int>(input_pin_id) - *socket_index) / cMaxNodesPerGraph;
} }
inline void OutputPinIdToNodeIndexAndSocketIndex( inline void OutputPinIdToNodeIndexAndSocketIndex(
unsigned long output_pin_id, unsigned long output_pin_id,
int* node_index, int* node_index,
int* socket_index) { int* socket_index) {
*socket_index = ((output_pin_id - 500) % 1000); *socket_index =
*node_index = (output_pin_id - *socket_index) / 1000; ((static_cast<int>(output_pin_id) - cMaxSocketsPerNode)
% cMaxNodesPerGraph);
*node_index =
(static_cast<int>(output_pin_id) - *socket_index) / cMaxNodesPerGraph;
} }
inline bool IsPinInput(unsigned long pin_id) {
return ((pin_id % cMaxNodesPerGraph) >= cMaxSocketsPerNode);
}
inline bool IsPinOutput(unsigned long pin_id) { return !IsPinInput(pin_id); }
void SyncTrackEditor(SyncTrack* sync_track); void SyncTrackEditor(SyncTrack* sync_track);
void SkinnedMeshWidget(SkinnedMesh* skinned_mesh); void SkinnedMeshWidget(SkinnedMesh* skinned_mesh);