2022-02-11 16:51:18 +01:00
|
|
|
//
|
|
|
|
// Created by martin on 11.02.22.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "AnimGraphEditor.h"
|
|
|
|
|
2023-03-30 18:11:54 +02:00
|
|
|
#include <sstream>
|
|
|
|
|
2024-03-01 14:26:52 +01:00
|
|
|
#include "3rdparty/imgui-node-editor/imgui_node_editor.h"
|
2024-04-01 12:33:23 +02:00
|
|
|
#include "AnimGraphResource.h"
|
2023-03-30 18:11:54 +02:00
|
|
|
#include "SkinnedMesh.h"
|
2022-03-25 11:46:44 +01:00
|
|
|
#include "imgui.h"
|
2022-02-12 12:06:25 +01:00
|
|
|
#include "imnodes.h"
|
2022-03-25 11:46:44 +01:00
|
|
|
#include "misc/cpp/imgui_stdlib.h"
|
2022-02-18 23:33:30 +01:00
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
struct EditorState {
|
|
|
|
AnimGraphResource* rootGraphResource = nullptr;
|
2024-05-01 10:58:33 +02:00
|
|
|
|
|
|
|
std::vector<AnimGraphResource*> hierarchyStack;
|
|
|
|
size_t hierarchyStackIndex = 0;
|
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
bool isGraphLoadedThisFrame = false;
|
2024-05-01 11:59:30 +02:00
|
|
|
ImVec2 mousePopupStart = {};
|
2024-04-30 18:40:54 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
static EditorState sEditorState;
|
2023-03-26 23:39:11 +02:00
|
|
|
|
2022-02-14 22:37:19 +01:00
|
|
|
ImNodesPinShape sGetSocketShapeFromSocketType(const SocketType& socket_type) {
|
|
|
|
switch (socket_type) {
|
|
|
|
case SocketType::SocketTypeAnimation:
|
|
|
|
return ImNodesPinShape_QuadFilled;
|
2023-04-02 21:40:49 +02:00
|
|
|
case SocketType::SocketTypeInt:
|
|
|
|
return ImNodesPinShape_CircleFilled;
|
2022-02-14 22:37:19 +01:00
|
|
|
case SocketType::SocketTypeFloat:
|
|
|
|
return ImNodesPinShape_CircleFilled;
|
|
|
|
case SocketType::SocketTypeVec3:
|
|
|
|
return ImNodesPinShape_TriangleFilled;
|
|
|
|
case SocketType::SocketTypeQuat:
|
|
|
|
return ImNodesPinShape_Triangle;
|
|
|
|
case SocketType::SocketTypeBool:
|
|
|
|
return ImNodesPinShape_Circle;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return ImNodesPinShape_Quad;
|
|
|
|
}
|
2022-02-11 16:51:18 +01:00
|
|
|
|
2022-03-25 11:46:44 +01:00
|
|
|
void NodeSocketEditor(Socket& socket) {
|
|
|
|
int mode_current = static_cast<int>(socket.m_type);
|
|
|
|
ImGui::InputText("Name", &socket.m_name);
|
|
|
|
if (ImGui::Combo(
|
|
|
|
"Type",
|
|
|
|
&mode_current,
|
|
|
|
SocketTypeNames,
|
|
|
|
sizeof(SocketTypeNames) / sizeof(char*))) {
|
|
|
|
socket.m_type = static_cast<SocketType>(mode_current);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-05 00:44:37 +02:00
|
|
|
void RemoveBlendTreeConnectionsForSocket(
|
|
|
|
BlendTreeResource& blend_tree_resource,
|
|
|
|
AnimNodeResource* node_resource,
|
2022-03-25 11:46:44 +01:00
|
|
|
Socket& socket) {
|
2024-04-25 21:12:08 +02:00
|
|
|
const BlendTreeConnectionResource* connection =
|
|
|
|
blend_tree_resource.FindConnectionForSocket(node_resource, socket.m_name);
|
|
|
|
while (connection != nullptr) {
|
|
|
|
blend_tree_resource.DisconnectSockets(
|
|
|
|
blend_tree_resource.GetNode(connection->source_node_index),
|
|
|
|
connection->source_socket_name,
|
|
|
|
blend_tree_resource.GetNode(connection->target_node_index),
|
|
|
|
connection->target_socket_name);
|
|
|
|
|
|
|
|
connection = blend_tree_resource.FindConnectionForSocket(
|
|
|
|
node_resource,
|
|
|
|
socket.m_name);
|
2022-03-25 11:46:44 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-30 18:11:54 +02:00
|
|
|
void SyncTrackEditor(SyncTrack* sync_track) {
|
|
|
|
ImGui::SliderFloat("duration", &sync_track->m_duration, 0.001f, 10.f);
|
|
|
|
|
|
|
|
ImGui::Text("Marker");
|
|
|
|
ImGui::SameLine();
|
|
|
|
ImGui::Text("%d", sync_track->m_num_intervals);
|
|
|
|
ImGui::SameLine();
|
|
|
|
if (ImGui::Button("+")) {
|
|
|
|
if (sync_track->m_num_intervals < cSyncTrackMaxIntervals) {
|
2023-04-01 14:16:20 +02:00
|
|
|
sync_track->m_num_intervals++;
|
2023-03-30 18:11:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
ImGui::SameLine();
|
|
|
|
if (ImGui::Button("-")) {
|
|
|
|
if (sync_track->m_num_intervals > 0) {
|
2023-04-01 14:16:20 +02:00
|
|
|
sync_track->m_num_intervals--;
|
2023-03-30 18:11:54 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::Text("Marker:");
|
|
|
|
for (int i = 0; i < sync_track->m_num_intervals; i++) {
|
|
|
|
ImGui::Text("%2d:", i);
|
|
|
|
ImGui::SameLine();
|
|
|
|
std::ostringstream marker_stream;
|
|
|
|
marker_stream << i;
|
|
|
|
ImGui::SliderFloat(
|
|
|
|
marker_stream.str().c_str(),
|
|
|
|
&sync_track->m_sync_markers[i],
|
|
|
|
0.f,
|
|
|
|
1.f);
|
|
|
|
}
|
|
|
|
|
2023-04-01 14:16:20 +02:00
|
|
|
if (ImGui::Button("Update Intervals")) {
|
2023-03-30 18:11:54 +02:00
|
|
|
sync_track->CalcIntervals();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void SkinnedMeshWidget(SkinnedMesh* skinned_mesh) {
|
|
|
|
if (ImGui::TreeNode("Bones")) {
|
|
|
|
for (int i = 0; i < skinned_mesh->m_skeleton.num_joints(); i++) {
|
|
|
|
ImGui::Text("%s", skinned_mesh->m_skeleton.joint_names()[i]);
|
|
|
|
}
|
|
|
|
ImGui::TreePop();
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::Text("Animations");
|
|
|
|
|
|
|
|
const char* items[255] = {0};
|
|
|
|
static int selected = -1;
|
|
|
|
for (int i = 0; i < skinned_mesh->m_animations.size(); i++) {
|
|
|
|
items[i] = skinned_mesh->m_animation_names[i].c_str();
|
|
|
|
}
|
|
|
|
|
2023-04-01 14:16:20 +02:00
|
|
|
ImGui::Combo(
|
|
|
|
"Animation",
|
|
|
|
&selected,
|
|
|
|
items,
|
|
|
|
skinned_mesh->m_animations.size());
|
2023-03-30 18:11:54 +02:00
|
|
|
|
|
|
|
ImGui::Text("Sync Track");
|
|
|
|
if (selected >= 0 && selected < skinned_mesh->m_animations.size()) {
|
|
|
|
SyncTrackEditor(&skinned_mesh->m_animation_sync_track[selected]);
|
|
|
|
skinned_mesh->m_override_anim = selected;
|
|
|
|
|
|
|
|
ImGui::Checkbox("Override Animation", &skinned_mesh->m_sync_track_override);
|
|
|
|
if (skinned_mesh->m_sync_track_override) {
|
|
|
|
ImGui::SliderFloat("Ratio", &skinned_mesh->m_override_ratio, 0.f, 1.f);
|
|
|
|
|
|
|
|
ozz::animation::SamplingJob sampling_job;
|
|
|
|
sampling_job.animation = skinned_mesh->m_animations[selected];
|
|
|
|
sampling_job.context = &skinned_mesh->m_sampling_context;
|
|
|
|
sampling_job.ratio = skinned_mesh->m_override_ratio;
|
|
|
|
sampling_job.output = make_span(skinned_mesh->m_local_matrices);
|
|
|
|
if (!sampling_job.Run()) {
|
|
|
|
ozz::log::Err() << "Error sampling animation." << std::endl;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-25 11:46:44 +01:00
|
|
|
void AnimGraphEditorRenderSidebar(
|
2024-04-05 00:44:37 +02:00
|
|
|
BlendTreeResource& blend_tree_resource,
|
|
|
|
AnimNodeResource* node_resource) {
|
2024-05-01 12:50:16 +02:00
|
|
|
ImGui::Text(
|
|
|
|
"[%s (%2.2f, %2.2f)]",
|
|
|
|
node_resource->m_node_type_name.c_str(),
|
|
|
|
node_resource->m_position[0],
|
|
|
|
node_resource->m_position[1]);
|
2022-02-18 22:24:19 +01:00
|
|
|
|
|
|
|
char node_name_buffer[256];
|
|
|
|
memset(node_name_buffer, 0, sizeof(node_name_buffer));
|
|
|
|
strncpy(
|
|
|
|
node_name_buffer,
|
2024-04-05 00:44:37 +02:00
|
|
|
node_resource->m_name.c_str(),
|
|
|
|
std::min(node_resource->m_name.size(), sizeof(node_name_buffer)));
|
2022-02-18 22:24:19 +01:00
|
|
|
|
|
|
|
if (ImGui::InputText("Name", node_name_buffer, sizeof(node_name_buffer))) {
|
2024-04-05 00:44:37 +02:00
|
|
|
node_resource->m_name = node_name_buffer;
|
2022-02-18 22:24:19 +01:00
|
|
|
}
|
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
int num_properties = 0;
|
|
|
|
if (node_resource->m_socket_accessor != nullptr) {
|
|
|
|
num_properties = node_resource->m_socket_accessor->m_properties.size();
|
|
|
|
}
|
2022-02-18 22:24:19 +01:00
|
|
|
for (int i = 0; i < num_properties; i++) {
|
2024-04-05 00:44:37 +02:00
|
|
|
Socket& property = node_resource->m_socket_accessor->m_properties[i];
|
2023-04-02 21:40:49 +02:00
|
|
|
if (property.m_type == SocketType::SocketTypeInt) {
|
|
|
|
ImGui::InputInt(
|
|
|
|
property.m_name.c_str(),
|
|
|
|
reinterpret_cast<int*>(&property.m_value.int_value),
|
|
|
|
1);
|
|
|
|
} else if (property.m_type == SocketType::SocketTypeFloat) {
|
2022-02-18 22:24:19 +01:00
|
|
|
ImGui::SliderFloat(
|
|
|
|
property.m_name.c_str(),
|
2023-03-28 22:00:58 +02:00
|
|
|
reinterpret_cast<float*>(&property.m_value.float_value),
|
2022-02-18 22:24:19 +01:00
|
|
|
-100.f,
|
|
|
|
100.f);
|
|
|
|
} else if (property.m_type == SocketType::SocketTypeBool) {
|
2023-04-02 21:24:12 +02:00
|
|
|
bool flag_value = property.GetValue<bool>();
|
2024-03-03 20:20:24 +01:00
|
|
|
if (ImGui::Checkbox(property.m_name.c_str(), &flag_value)) {
|
2023-04-02 21:24:12 +02:00
|
|
|
property.SetValue(flag_value);
|
|
|
|
}
|
2022-02-18 22:24:19 +01:00
|
|
|
} else if (property.m_type == SocketType::SocketTypeString) {
|
2023-03-26 23:39:11 +02:00
|
|
|
char string_buf[1024];
|
2023-04-01 14:16:20 +02:00
|
|
|
memset(string_buf, '\0', sizeof(string_buf));
|
|
|
|
memcpy(
|
|
|
|
string_buf,
|
|
|
|
property.m_value_string.c_str(),
|
|
|
|
std::min(
|
|
|
|
static_cast<size_t>(1024),
|
|
|
|
property.m_value_string.size() + 1));
|
2022-02-18 22:24:19 +01:00
|
|
|
if (ImGui::InputText(
|
|
|
|
property.m_name.c_str(),
|
|
|
|
string_buf,
|
|
|
|
sizeof(string_buf))) {
|
2023-04-01 14:16:20 +02:00
|
|
|
property.m_value_string = string_buf;
|
2022-02-18 22:24:19 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-25 11:46:44 +01:00
|
|
|
|
2024-04-05 00:44:37 +02:00
|
|
|
if (node_resource == blend_tree_resource.GetGraphOutputNode()) {
|
2022-03-25 11:46:44 +01:00
|
|
|
ImGui::Text("Outputs");
|
|
|
|
|
|
|
|
// Graph outputs are the inputs of the output node!
|
2024-04-05 00:44:37 +02:00
|
|
|
std::vector<Socket>& outputs = node_resource->m_socket_accessor->m_inputs;
|
2022-03-25 11:46:44 +01:00
|
|
|
|
|
|
|
std::vector<Socket>::iterator iter = outputs.begin();
|
|
|
|
while (iter != outputs.end()) {
|
|
|
|
Socket& output = *iter;
|
|
|
|
ImGui::PushID(&output);
|
|
|
|
NodeSocketEditor(output);
|
|
|
|
if (ImGui::Button("X")) {
|
2024-04-05 00:44:37 +02:00
|
|
|
RemoveBlendTreeConnectionsForSocket(
|
|
|
|
blend_tree_resource,
|
|
|
|
node_resource,
|
|
|
|
output);
|
2022-03-25 11:46:44 +01:00
|
|
|
iter = outputs.erase(iter);
|
|
|
|
} else {
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
ImGui::PopID();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-05 00:44:37 +02:00
|
|
|
if (node_resource == blend_tree_resource.GetGraphInputNode()) {
|
2022-03-25 11:46:44 +01:00
|
|
|
ImGui::Text("Inputs");
|
|
|
|
|
|
|
|
// Graph inputs are the outputs of the input node!
|
2024-04-05 00:44:37 +02:00
|
|
|
std::vector<Socket>& inputs = node_resource->m_socket_accessor->m_outputs;
|
2022-03-25 11:46:44 +01:00
|
|
|
|
|
|
|
std::vector<Socket>::iterator iter = inputs.begin();
|
|
|
|
while (iter != inputs.end()) {
|
|
|
|
Socket& input = *iter;
|
|
|
|
ImGui::PushID(&input);
|
|
|
|
NodeSocketEditor(input);
|
|
|
|
if (ImGui::Button("X")) {
|
2024-04-05 00:44:37 +02:00
|
|
|
RemoveBlendTreeConnectionsForSocket(
|
|
|
|
blend_tree_resource,
|
|
|
|
node_resource,
|
|
|
|
input);
|
2022-03-25 11:46:44 +01:00
|
|
|
iter = inputs.erase(iter);
|
|
|
|
} else {
|
|
|
|
iter++;
|
|
|
|
}
|
|
|
|
ImGui::PopID();
|
|
|
|
}
|
|
|
|
}
|
2022-02-18 22:24:19 +01:00
|
|
|
}
|
|
|
|
|
2024-04-25 21:12:08 +02:00
|
|
|
void AnimGraphEditorClear() {
|
2024-05-01 12:50:16 +02:00
|
|
|
if (ax::NodeEditor::GetCurrentEditor() != nullptr) {
|
|
|
|
ax::NodeEditor::ClearSelection();
|
|
|
|
}
|
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
delete sEditorState.rootGraphResource;
|
|
|
|
|
|
|
|
sEditorState.rootGraphResource = new AnimGraphResource();
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.rootGraphResource->m_name = "Root";
|
2024-04-30 18:40:54 +02:00
|
|
|
sEditorState.rootGraphResource->m_graph_type_name = "BlendTree";
|
|
|
|
sEditorState.rootGraphResource->m_blend_tree_resource.InitGraphConnectors();
|
2024-05-01 10:58:33 +02:00
|
|
|
|
2024-05-01 12:50:16 +02:00
|
|
|
sEditorState.hierarchyStack.clear();
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack.push_back(sEditorState.rootGraphResource);
|
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex] =
|
|
|
|
sEditorState.hierarchyStack.back();
|
2024-05-01 12:50:16 +02:00
|
|
|
sEditorState.hierarchyStackIndex = 0;
|
2024-04-25 21:12:08 +02:00
|
|
|
}
|
|
|
|
|
2024-03-01 14:26:52 +01:00
|
|
|
void AnimGraphEditorUpdate(ax::NodeEditor::EditorContext* context) {
|
2024-04-25 21:40:09 +02:00
|
|
|
ax::NodeEditor::SetCurrentEditor(context);
|
|
|
|
|
2024-05-01 10:58:33 +02:00
|
|
|
//
|
|
|
|
// Menu bar
|
|
|
|
//
|
2024-03-01 14:26:52 +01:00
|
|
|
ImGui::BeginMenuBar();
|
|
|
|
if (ImGui::Button("Save")) {
|
2024-04-30 18:40:54 +02:00
|
|
|
sEditorState.rootGraphResource->SaveToFile("editor_graph.json");
|
2024-03-01 14:26:52 +01:00
|
|
|
}
|
|
|
|
if (ImGui::Button("Load")) {
|
2024-05-01 12:50:16 +02:00
|
|
|
AnimGraphEditorClear();
|
2024-04-30 18:40:54 +02:00
|
|
|
sEditorState.rootGraphResource->LoadFromFile("editor_graph.json");
|
|
|
|
sEditorState.isGraphLoadedThisFrame = true;
|
2024-03-01 14:26:52 +01:00
|
|
|
}
|
|
|
|
if (ImGui::Button("Clear")) {
|
2024-04-25 21:12:08 +02:00
|
|
|
AnimGraphEditorClear();
|
2024-03-01 14:26:52 +01:00
|
|
|
}
|
2024-04-25 21:40:09 +02:00
|
|
|
if (ImGui::Button("Content")) {
|
|
|
|
ax::NodeEditor::NavigateToContent();
|
|
|
|
}
|
2024-03-01 14:26:52 +01:00
|
|
|
char graph_name_buffer[256];
|
|
|
|
memset(graph_name_buffer, 0, sizeof(graph_name_buffer));
|
|
|
|
strncpy(
|
|
|
|
graph_name_buffer,
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_name.c_str(),
|
2024-03-01 14:26:52 +01:00
|
|
|
sizeof(graph_name_buffer));
|
|
|
|
if (ImGui::InputText("Name", graph_name_buffer, sizeof(graph_name_buffer))) {
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]->m_name =
|
|
|
|
graph_name_buffer;
|
2024-03-01 14:26:52 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::EndMenuBar();
|
|
|
|
|
2024-05-01 10:58:33 +02:00
|
|
|
//
|
|
|
|
// Breadcrumb navigation
|
|
|
|
//
|
|
|
|
for (size_t i = 0, n = sEditorState.hierarchyStack.size(); i < n; i++) {
|
|
|
|
AnimGraphResource* graph_resource =
|
|
|
|
dynamic_cast<AnimGraphResource*>(sEditorState.hierarchyStack[i]);
|
|
|
|
ImGui::PushID(graph_resource);
|
|
|
|
|
|
|
|
bool highlight_button = i == sEditorState.hierarchyStackIndex;
|
|
|
|
|
|
|
|
if (highlight_button) {
|
|
|
|
ImGui::PushStyleColor(
|
|
|
|
ImGuiCol_Button,
|
|
|
|
(ImVec4)ImColor::HSV(1. / 7.0f, 0.6f, 0.6f));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::Button(graph_resource->m_name.c_str())) {
|
|
|
|
sEditorState.hierarchyStackIndex = i;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (highlight_button) {
|
|
|
|
ImGui::PopStyleColor(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::PopID();
|
|
|
|
|
|
|
|
if (i < n - 1) {
|
|
|
|
ImGui::SameLine();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-04-25 21:40:09 +02:00
|
|
|
ImGui::Columns(2);
|
|
|
|
|
2024-03-01 14:26:52 +01:00
|
|
|
//
|
|
|
|
// Node editor canvas
|
|
|
|
//
|
|
|
|
ax::NodeEditor::Begin("Graph Editor");
|
|
|
|
|
2024-05-01 12:50:16 +02:00
|
|
|
for (size_t node_index = 0,
|
2024-05-01 10:58:33 +02:00
|
|
|
n = sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNumNodes();
|
2024-05-01 12:50:16 +02:00
|
|
|
node_index < n;
|
|
|
|
node_index++) {
|
2024-04-05 00:44:37 +02:00
|
|
|
AnimNodeResource* node_resource =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
2024-05-01 12:50:16 +02:00
|
|
|
->m_blend_tree_resource.GetNode(node_index);
|
|
|
|
ax::NodeEditor::NodeId node_id(node_resource);
|
2024-03-01 14:26:52 +01:00
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
if (sEditorState.isGraphLoadedThisFrame) {
|
2024-03-03 20:20:24 +01:00
|
|
|
ax::NodeEditor::SetNodePosition(
|
|
|
|
node_id,
|
2024-04-05 00:44:37 +02:00
|
|
|
ImVec2(node_resource->m_position[0], node_resource->m_position[1]));
|
2024-03-03 20:20:24 +01:00
|
|
|
}
|
2024-03-01 14:26:52 +01:00
|
|
|
ax::NodeEditor::BeginNode(node_id);
|
2024-04-05 00:44:37 +02:00
|
|
|
ImGui::Text("%s", node_resource->m_node_type_name.c_str());
|
2024-03-01 14:26:52 +01:00
|
|
|
|
|
|
|
// Inputs
|
2024-04-30 18:40:54 +02:00
|
|
|
std::vector<Socket> node_inputs =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNodeInputSockets(node_resource);
|
2024-03-01 14:26:52 +01:00
|
|
|
for (size_t j = 0, ni = node_inputs.size(); j < ni; j++) {
|
|
|
|
Socket& socket = node_inputs[j];
|
2024-03-03 20:20:24 +01:00
|
|
|
ax::NodeEditor::BeginPin(
|
2024-04-25 21:12:08 +02:00
|
|
|
NodeIndexAndSocketIndexToInputPinId(
|
2024-05-01 12:50:16 +02:00
|
|
|
static_cast<int>(node_index),
|
2024-04-25 21:12:08 +02:00
|
|
|
static_cast<int>(j)),
|
2024-03-03 20:20:24 +01:00
|
|
|
ax::NodeEditor::PinKind::Input);
|
2024-03-01 14:26:52 +01:00
|
|
|
ImGui::Text("%s", socket.m_name.c_str());
|
|
|
|
ax::NodeEditor::EndPin();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Outputs
|
2024-04-30 18:40:54 +02:00
|
|
|
std::vector<Socket> node_outputs =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNodeOutputSockets(node_resource);
|
2024-03-01 14:26:52 +01:00
|
|
|
for (size_t j = 0, ni = node_outputs.size(); j < ni; j++) {
|
2024-03-03 20:20:24 +01:00
|
|
|
Socket& socket = node_outputs[j];
|
|
|
|
ax::NodeEditor::BeginPin(
|
2024-04-25 21:12:08 +02:00
|
|
|
NodeIndexAndSocketIndexToOutputPinId(
|
2024-05-01 12:50:16 +02:00
|
|
|
static_cast<int>(node_index),
|
2024-04-25 21:12:08 +02:00
|
|
|
static_cast<int>(j)),
|
2024-03-03 20:20:24 +01:00
|
|
|
ax::NodeEditor::PinKind::Output);
|
2024-03-01 14:26:52 +01:00
|
|
|
ImGui::Text("%s", socket.m_name.c_str());
|
|
|
|
ax::NodeEditor::EndPin();
|
|
|
|
}
|
|
|
|
|
|
|
|
ax::NodeEditor::EndNode();
|
2024-04-25 21:12:08 +02:00
|
|
|
|
|
|
|
ImVec2 node_position = ax::NodeEditor::GetNodePosition(node_id);
|
|
|
|
node_resource->m_position[0] = node_position.x;
|
|
|
|
node_resource->m_position[1] = node_position.y;
|
2024-03-01 14:26:52 +01:00
|
|
|
}
|
|
|
|
|
2024-03-03 20:20:24 +01:00
|
|
|
int link_id = 0;
|
2024-04-05 00:44:37 +02:00
|
|
|
for (size_t connection_id = 0,
|
2024-05-01 10:58:33 +02:00
|
|
|
n = sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNumConnections();
|
2024-04-05 00:44:37 +02:00
|
|
|
connection_id < n;
|
2024-03-03 20:20:24 +01:00
|
|
|
connection_id++) {
|
2024-04-24 21:58:47 +02:00
|
|
|
const BlendTreeConnectionResource* connection_resource =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetConnection(connection_id);
|
2024-04-05 00:44:37 +02:00
|
|
|
|
|
|
|
const AnimNodeResource* source_node_resource =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNode(
|
|
|
|
connection_resource->source_node_index);
|
2024-04-05 00:44:37 +02:00
|
|
|
int source_socket_index =
|
|
|
|
source_node_resource->m_socket_accessor->GetOutputIndex(
|
2024-04-24 21:58:47 +02:00
|
|
|
connection_resource->source_socket_name.c_str());
|
2024-04-05 00:44:37 +02:00
|
|
|
|
|
|
|
const AnimNodeResource* target_node_resource =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNode(
|
|
|
|
connection_resource->target_node_index);
|
2024-04-05 00:44:37 +02:00
|
|
|
int target_socket_index =
|
|
|
|
target_node_resource->m_socket_accessor->GetInputIndex(
|
2024-04-24 21:58:47 +02:00
|
|
|
connection_resource->target_socket_name.c_str());
|
2024-04-05 00:44:37 +02:00
|
|
|
|
2024-04-25 21:12:08 +02:00
|
|
|
int source_socket_pin_id = NodeIndexAndSocketIndexToOutputPinId(
|
2024-04-24 21:58:47 +02:00
|
|
|
static_cast<int>(connection_resource->source_node_index),
|
2024-04-05 00:44:37 +02:00
|
|
|
source_socket_index);
|
2024-04-25 21:12:08 +02:00
|
|
|
int target_socket_pin_id = NodeIndexAndSocketIndexToInputPinId(
|
2024-04-24 21:58:47 +02:00
|
|
|
static_cast<int>(connection_resource->target_node_index),
|
2024-04-05 00:44:37 +02:00
|
|
|
target_socket_index);
|
2024-03-03 20:20:24 +01:00
|
|
|
|
2024-04-25 21:12:08 +02:00
|
|
|
ax::NodeEditor::Link(link_id++, source_socket_pin_id, target_socket_pin_id);
|
2024-03-03 20:20:24 +01:00
|
|
|
}
|
2024-03-01 14:26:52 +01:00
|
|
|
|
|
|
|
// Create Connections
|
|
|
|
if (ax::NodeEditor::BeginCreate()) {
|
|
|
|
ax::NodeEditor::PinId input_pin_id, output_pin_id;
|
|
|
|
if (ax::NodeEditor::QueryNewLink(&input_pin_id, &output_pin_id)) {
|
|
|
|
if (input_pin_id && output_pin_id) {
|
2024-05-01 13:03:57 +02:00
|
|
|
int source_node_index;
|
|
|
|
int source_node_socket_index;
|
2024-04-25 21:12:08 +02:00
|
|
|
|
2024-05-01 13:03:57 +02:00
|
|
|
OutputPinIdToNodeIndexAndSocketIndex(
|
|
|
|
input_pin_id.Get(),
|
|
|
|
&source_node_index,
|
|
|
|
&source_node_socket_index);
|
|
|
|
|
|
|
|
const AnimNodeResource* source_node =
|
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNode(source_node_index);
|
|
|
|
if (source_node->m_socket_accessor->m_outputs.size()
|
|
|
|
< source_node_socket_index) {
|
|
|
|
source_node_socket_index = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
int target_node_index;
|
|
|
|
int target_node_socket_index;
|
|
|
|
|
|
|
|
InputPinIdToNodeIndexAndSocketIndex(
|
|
|
|
output_pin_id.Get(),
|
|
|
|
&target_node_index,
|
|
|
|
&target_node_socket_index);
|
|
|
|
|
|
|
|
const AnimNodeResource* target_node =
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
2024-05-01 13:03:57 +02:00
|
|
|
->m_blend_tree_resource.GetNode(target_node_index);
|
|
|
|
if (target_node->m_socket_accessor->m_inputs.size()
|
|
|
|
< target_node_socket_index) {
|
|
|
|
target_node_socket_index = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
const Socket* source_socket =
|
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNodeOutputSocketByIndex(
|
2024-04-30 18:40:54 +02:00
|
|
|
source_node,
|
2024-05-01 13:03:57 +02:00
|
|
|
source_node_socket_index);
|
|
|
|
const Socket* target_socket =
|
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.GetNodeInputSocketByIndex(
|
2024-04-30 18:40:54 +02:00
|
|
|
target_node,
|
2024-05-01 13:03:57 +02:00
|
|
|
target_node_socket_index);
|
|
|
|
|
|
|
|
if (source_socket == nullptr || target_socket == nullptr
|
|
|
|
|| !sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.IsConnectionValid(
|
|
|
|
source_node,
|
|
|
|
source_socket->m_name,
|
|
|
|
target_node,
|
|
|
|
target_socket->m_name)) {
|
|
|
|
ax::NodeEditor::RejectNewItem();
|
|
|
|
} else if (ax::NodeEditor::AcceptNewItem()) {
|
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.ConnectSockets(
|
|
|
|
source_node,
|
|
|
|
source_socket->m_name,
|
|
|
|
target_node,
|
|
|
|
target_socket->m_name);
|
2024-03-01 14:26:52 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ax::NodeEditor::EndCreate();
|
|
|
|
|
2024-04-25 21:12:08 +02:00
|
|
|
// Popup menu
|
|
|
|
{
|
|
|
|
const bool open_popup = ImGui::IsMouseReleased(ImGuiMouseButton_Right);
|
|
|
|
|
2024-05-01 11:59:30 +02:00
|
|
|
if (open_popup && ImGui::IsWindowHovered()) {
|
|
|
|
ax::NodeEditor::Suspend();
|
2024-04-25 21:12:08 +02:00
|
|
|
ImGui::OpenPopup("add node");
|
2024-05-01 11:59:30 +02:00
|
|
|
ax::NodeEditor::Resume();
|
|
|
|
sEditorState.mousePopupStart = ImGui::GetMousePos();
|
2024-04-25 21:12:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ax::NodeEditor::Suspend();
|
|
|
|
ImGui::PushStyleVar(ImGuiStyleVar_WindowPadding, ImVec2(8.f, 8.f));
|
|
|
|
if (ImGui::BeginPopup("add node")) {
|
|
|
|
std::string node_type_name = "";
|
|
|
|
if (ImGui::MenuItem("AnimSampler")) {
|
|
|
|
node_type_name = "AnimSampler";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::MenuItem("Blend2")) {
|
|
|
|
node_type_name = "Blend2";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::MenuItem("SpeedScale")) {
|
|
|
|
node_type_name = "SpeedScale";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::MenuItem("LockTranslationNode")) {
|
|
|
|
node_type_name = "LockTranslationNode";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::MenuItem("MathAddNode")) {
|
|
|
|
node_type_name = "MathAddNode";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::MenuItem("MathFloatToVec3Node")) {
|
|
|
|
node_type_name = "MathFloatToVec3Node";
|
|
|
|
}
|
|
|
|
|
|
|
|
if (ImGui::MenuItem("ConstScalarNode")) {
|
|
|
|
node_type_name = "ConstScalarNode";
|
|
|
|
}
|
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
if (ImGui::MenuItem("BlendTree")) {
|
|
|
|
node_type_name = "BlendTree";
|
|
|
|
}
|
|
|
|
|
2024-04-25 21:40:09 +02:00
|
|
|
if (!node_type_name.empty()) {
|
2024-04-25 21:12:08 +02:00
|
|
|
AnimNodeResource* node_resource =
|
|
|
|
AnimNodeResourceFactory(node_type_name);
|
2024-05-01 13:03:57 +02:00
|
|
|
ax::NodeEditor::SetNodePosition(
|
|
|
|
ax::NodeEditor::NodeId(node_resource),
|
|
|
|
sEditorState.mousePopupStart);
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource.AddNode(node_resource);
|
2024-04-25 21:12:08 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::EndPopup();
|
|
|
|
}
|
|
|
|
ImGui::PopStyleVar();
|
|
|
|
ax::NodeEditor::Resume();
|
|
|
|
}
|
|
|
|
|
2024-03-01 14:26:52 +01:00
|
|
|
ax::NodeEditor::End();
|
|
|
|
|
2024-04-25 21:40:09 +02:00
|
|
|
//
|
|
|
|
// Sidebar
|
|
|
|
//
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
if (ax::NodeEditor::GetSelectedObjectCount() > 0) {
|
|
|
|
ax::NodeEditor::NodeId selected_node_id = 0;
|
|
|
|
|
|
|
|
ax::NodeEditor::GetSelectedNodes(&selected_node_id, 1);
|
|
|
|
|
|
|
|
if (selected_node_id.Get() != 0) {
|
|
|
|
AnimGraphEditorRenderSidebar(
|
2024-05-01 10:58:33 +02:00
|
|
|
sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex]
|
|
|
|
->m_blend_tree_resource,
|
2024-05-01 12:50:16 +02:00
|
|
|
selected_node_id.AsPointer<AnimNodeResource>());
|
2024-04-25 21:40:09 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ImGui::Columns(1);
|
2024-03-03 20:20:24 +01:00
|
|
|
|
2024-05-01 12:50:16 +02:00
|
|
|
// Clear flag, however it may be re-set further down when handling double
|
|
|
|
// clicking into subgraphs.
|
|
|
|
sEditorState.isGraphLoadedThisFrame = false;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Handle double click into subgraphs
|
|
|
|
//
|
2024-04-30 18:40:54 +02:00
|
|
|
ax::NodeEditor::NodeId double_clicked_node_id =
|
|
|
|
ax::NodeEditor::GetDoubleClickedNode();
|
|
|
|
if (!double_clicked_node_id.Invalid) {
|
|
|
|
AnimNodeResource* clicked_node_resource =
|
2024-05-01 12:50:16 +02:00
|
|
|
double_clicked_node_id.AsPointer<AnimNodeResource>();
|
|
|
|
|
|
|
|
if (clicked_node_resource != nullptr
|
|
|
|
&& clicked_node_resource->m_node_type_name == "BlendTree") {
|
2024-05-01 10:58:33 +02:00
|
|
|
AnimGraphResource* clicked_graph_resource =
|
2024-04-30 18:40:54 +02:00
|
|
|
dynamic_cast<AnimGraphResource*>(clicked_node_resource);
|
2024-05-01 12:50:16 +02:00
|
|
|
|
|
|
|
assert(clicked_graph_resource != nullptr);
|
|
|
|
|
2024-05-01 10:58:33 +02:00
|
|
|
if (sEditorState.hierarchyStack.size()
|
|
|
|
> sEditorState.hierarchyStackIndex + 1
|
|
|
|
&& sEditorState.hierarchyStack[sEditorState.hierarchyStackIndex + 1]
|
|
|
|
== clicked_graph_resource) {
|
|
|
|
sEditorState.hierarchyStackIndex++;
|
|
|
|
} else {
|
|
|
|
sEditorState.hierarchyStack.resize(
|
|
|
|
sEditorState.hierarchyStackIndex + 1);
|
|
|
|
sEditorState.hierarchyStack.push_back(clicked_graph_resource);
|
|
|
|
sEditorState.hierarchyStackIndex++;
|
|
|
|
}
|
|
|
|
|
2024-05-01 12:50:16 +02:00
|
|
|
sEditorState.isGraphLoadedThisFrame = true;
|
2024-04-30 18:40:54 +02:00
|
|
|
ax::NodeEditor::ClearSelection();
|
2022-02-15 21:06:12 +01:00
|
|
|
}
|
2022-02-11 16:51:18 +01:00
|
|
|
}
|
|
|
|
|
2024-04-30 18:40:54 +02:00
|
|
|
ax::NodeEditor::SetCurrentEditor(nullptr);
|
2023-03-26 23:39:11 +02:00
|
|
|
}
|
|
|
|
|
2024-04-05 00:44:37 +02:00
|
|
|
void AnimGraphEditorGetRuntimeGraph(AnimGraphBlendTree& blend_tree) {
|
2024-04-30 18:40:54 +02:00
|
|
|
sEditorState.rootGraphResource->CreateBlendTreeInstance(blend_tree);
|
2022-02-11 16:51:18 +01:00
|
|
|
}
|