#pragma clang diagnostic push #pragma ide diagnostic ignored "misc-no-recursion" // // Created by martin on 17.03.24. // #include "AnimGraphResource.h" #include #include #include "3rdparty/json/json.hpp" #include "AnimGraphBlendTree.h" #include "AnimGraphNodes.h" using json = nlohmann::json; // forward declarations static json sAnimGraphResourceBlendTreeToJson( const AnimGraphResource& anim_graph_resource); static bool sAnimGraphResourceBlendTreeFromJson( const json& json_data, AnimGraphResource* result_graph_resource); // // Socket <-> json // std::string sSocketTypeToStr(SocketType pin_type) { if (pin_type < SocketType::SocketTypeUndefined || pin_type >= SocketType::SocketTypeLast) { return "Unknown"; } return SocketTypeNames[static_cast(pin_type)]; } json sSocketToJson(const Socket& socket) { json result; result["name"] = socket.m_name; result["type"] = sSocketTypeToStr(socket.m_type); if (socket.m_type == SocketType::SocketTypeString && !socket.m_value_string.empty()) { result["value"] = socket.m_value_string; } else if (socket.m_value.flag) { if (socket.m_type == SocketType::SocketTypeBool) { result["value"] = socket.m_value.flag; } else if (socket.m_type == SocketType::SocketTypeAnimation) { } else if (socket.m_type == SocketType::SocketTypeInt) { result["value"] = socket.m_value.int_value; } else if (socket.m_type == SocketType::SocketTypeFloat) { result["value"] = socket.m_value.float_value; } else if (socket.m_type == SocketType::SocketTypeVec3) { result["value"][0] = socket.m_value.vec3.v[0]; result["value"][1] = socket.m_value.vec3.v[1]; result["value"][2] = socket.m_value.vec3.v[2]; } else if (socket.m_type == SocketType::SocketTypeQuat) { result["value"][0] = socket.m_value.quat.v[0]; result["value"][1] = socket.m_value.quat.v[1]; result["value"][2] = socket.m_value.quat.v[2]; result["value"][3] = socket.m_value.quat.v[3]; } else { std::cerr << "Invalid socket type '" << static_cast(socket.m_type) << "'." << std::endl; } } return result; } Socket sJsonToSocket(const json& json_data) { Socket result; result.m_type = SocketType::SocketTypeUndefined; result.m_reference.ptr = &result.m_value.int_value; result.m_name = json_data["name"]; std::string type_string = json_data["type"]; bool have_value = json_data.contains("value"); if (type_string == "Bool") { result.m_type = SocketType::SocketTypeBool; result.m_type_size = sizeof(bool); result.m_reference.ptr = &result.m_value.int_value; if (have_value) { result.m_value.flag = json_data["value"]; } } else if (type_string == "Animation") { result.m_type = SocketType::SocketTypeAnimation; result.m_type_size = sizeof(AnimData); } else if (type_string == "Int") { result.m_type = SocketType::SocketTypeInt; result.m_type_size = sizeof(int); if (have_value) { result.m_value.int_value = json_data["value"]; } } else if (type_string == "Float") { result.m_type = SocketType::SocketTypeFloat; result.m_type_size = sizeof(float); if (have_value) { result.m_value.float_value = json_data["value"]; } } else if (type_string == "Vec3") { result.m_type = SocketType::SocketTypeVec3; result.m_type_size = sizeof(Vec3); if (have_value) { result.m_value.vec3.x = json_data["value"][0]; result.m_value.vec3.y = json_data["value"][1]; result.m_value.vec3.z = json_data["value"][2]; } } else if (type_string == "Quat") { result.m_type = SocketType::SocketTypeQuat; result.m_type_size = sizeof(Quat); if (have_value) { result.m_value.quat.x = json_data["value"][0]; result.m_value.quat.y = json_data["value"][1]; result.m_value.quat.z = json_data["value"][2]; result.m_value.quat.w = json_data["value"][3]; } } else if (type_string == "String") { result.m_type = SocketType::SocketTypeString; result.m_type_size = sizeof(std::string); if (have_value) { result.m_value_string = json_data["value"]; } } else { std::cerr << "Invalid socket type '" << type_string << "'." << std::endl; } return result; } // // AnimGraphNode <-> json // json sAnimGraphNodeToJson( const AnimNodeResource* node, size_t node_index, const std::vector& connections) { json result; result["name"] = node->m_name; result["type"] = "AnimNodeResource"; result["node_type"] = node->m_node_type_name; for (size_t j = 0; j < 2; j++) { result["position"][j] = node->m_position[j]; } if (node->m_node_type_name == "BlendTree") { } for (const auto& socket : node->m_socket_accessor->m_inputs) { if (socket.m_type == SocketType::SocketTypeAnimation) { continue; } bool socket_connected = false; for (const auto& connection : connections) { if (connection.source_node_index == node_index && connection.source_socket_name == socket.m_name) { socket_connected = true; break; } } if (!socket_connected) { result["inputs"].push_back(sSocketToJson(socket)); } } for (auto& property : node->m_socket_accessor->m_properties) { result["properties"][property.m_name] = sSocketToJson(property); } return result; } AnimNodeResource* sAnimGraphNodeFromJson( const json& json_node, size_t node_index) { std::string node_type = json_node["node_type"]; if (node_type == "BlendTree") { AnimGraphResource* result = new AnimGraphResource(); sAnimGraphResourceBlendTreeFromJson(json_node, result); return result; } AnimNodeResource* result; result = new AnimNodeResource(); result->m_name = json_node["name"]; result->m_node_type_name = node_type; result->m_position[0] = json_node["position"][0]; result->m_position[1] = json_node["position"][1]; result->m_anim_node = AnimNodeFactory(result->m_node_type_name); result->m_socket_accessor = AnimNodeDescriptorFactory(result->m_node_type_name, result->m_anim_node); for (auto& property : result->m_socket_accessor->m_properties) { property = sJsonToSocket(json_node["properties"][property.m_name]); } if (node_index != 0 && node_index != 1 && json_node.contains("inputs")) { for (size_t j = 0, n = json_node["inputs"].size(); j < n; j++) { assert(json_node["inputs"][j].contains("name")); std::string input_name = json_node["inputs"][j]["name"]; Socket* input_socket = result->m_socket_accessor->GetInputSocket(input_name.c_str()); if (input_socket == nullptr) { std::cerr << "Could not find input socket with name " << input_name << " for node type " << result->m_node_type_name << std::endl; abort(); } *input_socket = sJsonToSocket(json_node["inputs"][j]); } } return result; } // // AnimGraphConnectionResource <-> Json // json sAnimGraphConnectionToJson(const BlendTreeConnectionResource* connection) { json result; result["type"] = "AnimGraphConnectionResource"; result["source_node_index"] = connection->source_node_index; result["source_socket_name"] = connection->source_socket_name; result["target_node_index"] = connection->target_node_index; result["target_socket_name"] = connection->target_socket_name; return result; } BlendTreeConnectionResource sAnimGraphConnectionFromJson( const json& json_node) { BlendTreeConnectionResource connection; connection.source_node_index = json_node["source_node_index"]; connection.source_socket_name = json_node["source_socket_name"]; connection.target_node_index = json_node["target_node_index"]; connection.target_socket_name = json_node["target_socket_name"]; return connection; } static json sAnimGraphResourceBlendTreeToJson( const AnimGraphResource& anim_graph_resource) { json result; result["name"] = anim_graph_resource.m_name; result["type"] = "AnimNodeResource"; result["node_type"] = "BlendTree"; const BlendTreeResource& blend_tree_resource = anim_graph_resource.m_blend_tree_resource; for (size_t i = 0; i < blend_tree_resource.GetNumNodes(); i++) { const AnimNodeResource* node = blend_tree_resource.GetNode(i); if (node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); result["nodes"][i] = sAnimGraphResourceBlendTreeToJson(*graph_resource); } else { result["nodes"][i] = sAnimGraphNodeToJson(node, i, blend_tree_resource.GetConnections()); } } for (size_t i = 0; i < blend_tree_resource.GetNumConnections(); i++) { const BlendTreeConnectionResource* connection = blend_tree_resource.GetConnection(i); result["connections"][i] = sAnimGraphConnectionToJson(connection); } // Graph inputs and outputs { const AnimNodeResource* graph_output_node = blend_tree_resource.GetNode(0); const std::vector graph_inputs = graph_output_node->m_socket_accessor->m_inputs; for (size_t i = 0; i < graph_inputs.size(); i++) { result["nodes"][0]["inputs"][i] = sSocketToJson(graph_inputs[i]); } const AnimNodeResource* graph_input_node = blend_tree_resource.GetNode(1); const std::vector graph_outputs = graph_input_node->m_socket_accessor->m_outputs; for (size_t i = 0; i < graph_outputs.size(); i++) { result["nodes"][1]["outputs"][i] = sSocketToJson(graph_outputs[i]); } } return result; } static bool sAnimGraphResourceBlendTreeFromJson( const json& json_data, AnimGraphResource* result_graph_resource) { BlendTreeResource& blend_tree_resource = result_graph_resource->m_blend_tree_resource; result_graph_resource->m_graph_type_name = "BlendTree"; result_graph_resource->m_node_type_name = "BlendTree"; result_graph_resource->m_name = json_data["name"]; // Load nodes for (size_t i = 0, n = json_data["nodes"].size(); i < n; i++) { const json& json_node = json_data["nodes"][i]; if (json_node["type"] != "AnimNodeResource") { std::cerr << "Invalid json object. Expected type 'AnimNodeResource' but got '" << json_node["type"] << "'." << std::endl; return false; } AnimNodeResource* node = sAnimGraphNodeFromJson(json_node, i); blend_tree_resource.AddNode(node); } // Graph outputs if (json_data["nodes"][0].contains("inputs")) { const json& graph_outputs = json_data["nodes"][0]["inputs"]; for (const auto& graph_output : graph_outputs) { AnimNodeResource* graph_node = blend_tree_resource.GetNode(0); graph_node->m_socket_accessor->m_inputs.push_back( sJsonToSocket(graph_output)); } } // Graph inputs (optional) if (json_data["nodes"][1].contains("outputs")) { const json& graph_inputs = json_data["nodes"][1]["outputs"]; for (const auto& graph_input : graph_inputs) { AnimNodeResource* graph_node = blend_tree_resource.GetNode(1); graph_node->m_socket_accessor->m_outputs.push_back( sJsonToSocket(graph_input)); } } // Load connections if (json_data.contains("connections")) { for (const auto& json_connection : json_data["connections"]) { if (json_connection["type"] != "AnimGraphConnectionResource") { std::cerr << "Invalid json object. Expected type " "'AnimGraphConnectionResource' " "but got '" << json_connection["type"] << "'." << std::endl; return false; } BlendTreeConnectionResource connection = sAnimGraphConnectionFromJson(json_connection); blend_tree_resource.ConnectSockets( blend_tree_resource.GetNode(connection.source_node_index), connection.source_socket_name, blend_tree_resource.GetNode(connection.target_node_index), connection.target_socket_name); } } return true; } bool BlendTreeResource::ConnectSockets( const AnimNodeResource* source_node, const std::string& source_socket_name, const AnimNodeResource* target_node, const std::string& target_socket_name) { if (!IsConnectionValid( source_node, source_socket_name, target_node, target_socket_name)) { return false; } int source_node_index = GetNodeIndex(source_node); int target_node_index = GetNodeIndex(target_node); if (source_node_index >= m_nodes.size() || target_node_index >= m_nodes.size()) { std::cerr << "Cannot connect nodes: could not find nodes." << std::endl; return false; } Socket* source_socket = GetNodeOutputSocket(source_node, source_socket_name); Socket* target_socket = GetNodeInputSocket(target_node, target_socket_name); if (source_socket == nullptr) { std::cerr << "Cannot connect nodes: could not find source socket '" << source_socket_name << "'." << std::endl; } if (target_socket == nullptr) { std::cerr << "Cannot connect nodes: could not find target socket '" << target_socket_name << "'." << std::endl; } if (target_socket == nullptr || source_socket == nullptr) { return false; } BlendTreeConnectionResource connection; connection.source_node_index = source_node_index; connection.source_socket_name = source_socket_name; connection.target_node_index = target_node_index; connection.target_socket_name = target_socket_name; m_connections.push_back(connection); m_node_input_connection_indices[target_node_index].emplace_back( m_connections.size() - 1); UpdateTreeTopologyInfo(); return true; } bool BlendTreeResource::DisconnectSockets( const AnimNodeResource* source_node, const std::string& source_socket_name, const AnimNodeResource* target_node, const std::string& target_socket_name) { int source_node_index = GetNodeIndex(source_node); int target_node_index = GetNodeIndex(target_node); if (source_node_index < 0 || target_node_index < 0) { return false; } int connection_index = -1; for (size_t i = 0, n = m_connections.size(); i < n; i++) { if (m_connections[i] == BlendTreeConnectionResource{ source_node_index, source_socket_name, target_node_index, target_socket_name}) { connection_index = i; break; } } if (connection_index == -1) { std::cerr << "Error: cannot disconnect sockets as connection is not found!" << std::endl; return false; } // remove connection m_connections.erase(m_connections.begin() + connection_index); // remove the input connection of the target node std::vector& target_input_connections = m_node_input_connection_indices[target_node_index]; std::vector::iterator end_iterator = std::remove( target_input_connections.begin(), target_input_connections.end(), connection_index); target_input_connections.erase(end_iterator); // Decrement all node input connection indices that are after the connection // we have removed above. for (size_t node_index = 0, n = m_nodes.size(); node_index < n; node_index++) { std::vector& node_input_connections = m_node_input_connection_indices[node_index]; for (size_t& node_connection_index : node_input_connections) { if (node_connection_index > connection_index) { node_connection_index--; } } } UpdateTreeTopologyInfo(); return true; } Socket* BlendTreeResource::GetNodeOutputSocket( const AnimNodeResource* node, const std::string& output_socket_name) const { Socket* output_socket = nullptr; if (node->m_socket_accessor) { output_socket = node->m_socket_accessor->GetOutputSocket(output_socket_name.c_str()); } if (output_socket == nullptr && node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); const BlendTreeResource& blend_tree_resource = graph_resource->m_blend_tree_resource; output_socket = blend_tree_resource.GetGraphOutputNode() ->m_socket_accessor->GetInputSocket(output_socket_name.c_str()); } return output_socket; } const Socket* BlendTreeResource::GetNodeOutputSocketByIndex( const AnimNodeResource* node, const size_t socket_output_index) const { const std::vector* output_sockets = nullptr; if (node->m_socket_accessor) { output_sockets = &node->m_socket_accessor->m_outputs; } else if (node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); const BlendTreeResource& blend_tree_resource = graph_resource->m_blend_tree_resource; output_sockets = &blend_tree_resource.GetGraphOutputNode()->m_socket_accessor->m_outputs; } if (output_sockets != nullptr && output_sockets->size() > socket_output_index) { return &output_sockets->operator[](socket_output_index); } return nullptr; } Socket* BlendTreeResource::GetNodeInputSocket( const AnimNodeResource* node, const std::string& input_socket_name) const { Socket* input_socket = nullptr; if (node->m_socket_accessor) { input_socket = node->m_socket_accessor->GetInputSocket(input_socket_name.c_str()); } if (input_socket == nullptr && node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); const BlendTreeResource& blend_tree_resource = graph_resource->m_blend_tree_resource; input_socket = blend_tree_resource.GetGraphInputNode() ->m_socket_accessor->GetOutputSocket(input_socket_name.c_str()); } return input_socket; } const Socket* BlendTreeResource::GetNodeInputSocketByIndex( const AnimNodeResource* node, const size_t socket_input_index) const { const std::vector* output_sockets = nullptr; if (node->m_socket_accessor) { output_sockets = &node->m_socket_accessor->m_inputs; } else if (node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); const BlendTreeResource& blend_tree_resource = graph_resource->m_blend_tree_resource; output_sockets = &blend_tree_resource.GetGraphOutputNode()->m_socket_accessor->m_outputs; } if (output_sockets != nullptr && output_sockets->size() > socket_input_index) { return &output_sockets->operator[](socket_input_index); } return nullptr; } std::vector BlendTreeResource::GetNodeOutputSockets( const AnimNodeResource* node) const { if (node->m_socket_accessor) { return node->m_socket_accessor->m_outputs; } if (node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); const BlendTreeResource& blend_tree_resource = graph_resource->m_blend_tree_resource; return blend_tree_resource.GetGraphOutputNode() ->m_socket_accessor->m_inputs; } return std::vector(); } std::vector BlendTreeResource::GetNodeInputSockets( const AnimNodeResource* node) const { if (node->m_socket_accessor) { return node->m_socket_accessor->m_inputs; } if (node->m_node_type_name == "BlendTree") { const AnimGraphResource* graph_resource = dynamic_cast(node); const BlendTreeResource& blend_tree_resource = graph_resource->m_blend_tree_resource; return blend_tree_resource.GetGraphInputNode() ->m_socket_accessor->m_outputs; } return std::vector(); } bool BlendTreeResource::IsConnectionValid( const AnimNodeResource* source_node, const std::string& source_socket_name, const AnimNodeResource* target_node, const std::string& target_socket_name) const { // Check self connection if (source_node == target_node) { return false; } // Check for loops int source_node_index = GetNodeIndex(source_node); int target_node_index = GetNodeIndex(target_node); if (std::find( m_node_inputs_subtree[source_node_index].cbegin(), m_node_inputs_subtree[source_node_index].cend(), target_node_index) != m_node_inputs_subtree[source_node_index].end()) { return false; } const Socket* source_socket = GetNodeOutputSocket(source_node, source_socket_name); const Socket* target_socket = GetNodeInputSocket(target_node, target_socket_name); if (source_socket == nullptr) { std::cerr << "Cannot connect nodes: could not find source socket '" << source_socket_name << "'." << std::endl; } if (target_socket == nullptr) { std::cerr << "Cannot connect nodes: could not find target socket '" << target_socket_name << "'." << std::endl; } if (target_socket == nullptr || source_socket == nullptr) { return false; } // Check socket types if (source_socket->m_type != target_socket->m_type) { return false; } return true; } const BlendTreeConnectionResource* BlendTreeResource::FindConnectionForSocket( const AnimNodeResource* node, const std::string& socket_name) const { int node_index = GetNodeIndex(node); std::vector::const_iterator connection_iter = std::find_if( m_connections.begin(), m_connections.end(), [node_index, socket_name](const BlendTreeConnectionResource& connection) { if ((connection.source_node_index == node_index && connection.source_socket_name == socket_name) || (connection.target_node_index == node_index && connection.target_socket_name == socket_name)) { return true; } return false; }); if (connection_iter != m_connections.end()) { return &*connection_iter; } return nullptr; } void BlendTreeResource::UpdateTreeTopologyInfo() { // TODO: Updating eval order and subtrees may get slow with many nodes. An // iterative approach would scale better. But let's leave that optimization // for a later time. UpdateNodeEvalOrder(); UpdateNodeSubtrees(); } void BlendTreeResource::UpdateNodeEvalOrderRecursive(const size_t node_index) { const std::vector& node_input_connection_indices = m_node_input_connection_indices[node_index]; for (size_t i = 0, n = node_input_connection_indices.size(); i < n; i++) { const BlendTreeConnectionResource& connection_resource = m_connections[node_input_connection_indices[i]]; if (connection_resource.source_node_index == 1) { continue; } UpdateNodeEvalOrderRecursive(connection_resource.source_node_index); } if (node_index != 0 && node_index != 1) { // In case we have multiple output connections from the node we here // ensure that use the node evaluation that is the furthest away from // the output. std::vector::iterator find_iter = std::find( m_node_eval_order.begin(), m_node_eval_order.end(), node_index); if (find_iter != m_node_eval_order.end()) { m_node_eval_order.erase(find_iter); } m_node_eval_order.push_back(node_index); } } void BlendTreeResource::UpdateNodeSubtrees() { for (size_t eval_index = 0, num_eval_nodes = m_node_eval_order.size(); eval_index < num_eval_nodes; eval_index++) { size_t node_index = m_node_eval_order[eval_index]; m_node_inputs_subtree[node_index].clear(); const std::vector& node_input_connection_indices = m_node_input_connection_indices[node_index]; for (size_t i = 0, n = node_input_connection_indices.size(); i < n; i++) { const BlendTreeConnectionResource& connection_resource = m_connections[node_input_connection_indices[i]]; m_node_inputs_subtree[node_index].emplace_back( connection_resource.source_node_index); m_node_inputs_subtree[node_index].insert( m_node_inputs_subtree[node_index].end(), m_node_inputs_subtree[connection_resource.source_node_index].cbegin(), m_node_inputs_subtree[connection_resource.source_node_index].cend()); } } } bool AnimGraphResource::LoadFromFile(const char* filename) { Clear(); std::ifstream input_file; input_file.open(filename); std::stringstream buffer; buffer << input_file.rdbuf(); json json_data = json::parse(buffer.str(), nullptr, false); if (json_data.is_discarded()) { std::cerr << "Error parsing json of file '" << filename << "'." << std::endl; return false; } if (json_data["type"] != "AnimNodeResource") { std::cerr << "Invalid json object. Expected type 'AnimNodeResource' but got '" << json_data["type"] << "'." << std::endl; return false; } if (json_data["node_type"] == "BlendTree") { return sAnimGraphResourceBlendTreeFromJson(json_data, this); } else if (json_data["node_type"] == "StateMachine") { return LoadStateMachineResourceFromJson(json_data); } std::cerr << "Invalid node_type. Expected type 'BlendTree' or " "'StateMachine' but got '" << json_data["node_type"] << "'." << std::endl; return false; } bool AnimGraphResource::SaveToFile(const char* filename) const { if (m_graph_type_name == "BlendTree") { return SaveBlendTreeResourceToFile(filename); } else if (m_graph_type_name == "StateMachine") { return SaveStateMachineResourceToFile(filename); } std::cerr << "Invalid AnimGraphResource type: " << m_graph_type_name << "." << std::endl; return false; } bool AnimGraphResource::SaveBlendTreeResourceToFile( const char* filename) const { json result; result = sAnimGraphResourceBlendTreeToJson(*this); std::ofstream output_file; output_file.open(filename); output_file << result.dump(4, ' ') << std::endl; output_file.close(); return true; } void AnimGraphResource::CreateBlendTreeInstance( AnimGraphBlendTree& result) const { if (m_node_type_name != "BlendTree") { std::cerr << "Invalid AnimGraphResource. Expected type 'BlendTree' but got '" << m_graph_type_name << "'." << std::endl; return; } result.m_name = m_name; NodeSocketDataOffsetMap node_offset_map; CreateBlendTreeRuntimeNodeInstances(result); PrepareBlendTreeIOData(result, node_offset_map); SetRuntimeNodeProperties(result); CreateBlendTreeConnectionInstances(result, node_offset_map); result.UpdateOrderedNodes(); result.ResetNodeStates(); } void AnimGraphResource::CreateBlendTreeRuntimeNodeInstances( AnimGraphBlendTree& result) const { for (const AnimNodeResource* node_resource : m_blend_tree_resource.GetNodes()) { AnimNode* node = AnimNodeFactory(node_resource->m_node_type_name); if (node_resource->m_node_type_name == "BlendTree") { const AnimGraphResource* embedded_blend_tree_resource = dynamic_cast(node_resource); assert(embedded_blend_tree_resource != nullptr); AnimGraphBlendTree* embedded_blend_tree = dynamic_cast(node); assert(embedded_blend_tree != nullptr); embedded_blend_tree_resource->CreateBlendTreeInstance( *embedded_blend_tree); embedded_blend_tree_resource->m_socket_accessor->m_inputs = embedded_blend_tree->m_node_descriptor->m_outputs; embedded_blend_tree_resource->m_socket_accessor->m_outputs = embedded_blend_tree->m_node_descriptor->m_inputs; } node->m_name = node_resource->m_name; node->m_node_type_name = node_resource->m_node_type_name; result.m_nodes.push_back(node); // runtime node connections result.m_node_input_connections.emplace_back(); result.m_node_output_connections.emplace_back(); } } void AnimGraphResource::PrepareBlendTreeIOData( AnimGraphBlendTree& instance, NodeSocketDataOffsetMap& node_offset_map) const { instance.m_node_descriptor = AnimNodeDescriptorFactory("BlendTree", instance.m_nodes[0]); instance.m_node_descriptor->m_outputs = m_blend_tree_resource.GetNode(1)->m_socket_accessor->m_outputs; instance.m_node_descriptor->m_inputs = m_blend_tree_resource.GetNode(0)->m_socket_accessor->m_inputs; // // graph inputs // int input_block_size = 0; std::vector& graph_inputs = instance.GetGraphInputs(); for (int i = 0; i < graph_inputs.size(); i++) { input_block_size += sizeof(void*); } if (input_block_size > 0) { instance.m_input_buffer = new char[input_block_size]; memset(instance.m_input_buffer, 0, input_block_size); } int input_block_offset = 0; for (int i = 0; i < graph_inputs.size(); i++) { graph_inputs[i].m_reference.ptr = (void*)&instance.m_input_buffer[input_block_offset]; instance.m_node_descriptor->m_outputs[i].m_reference.ptr = &instance.m_input_buffer[input_block_offset]; input_block_offset += sizeof(void*); } // // graph outputs // int output_block_size = 0; std::vector& graph_outputs = instance.GetGraphOutputs(); for (int i = 0; i < graph_outputs.size(); i++) { output_block_size += sizeof(void*); } if (output_block_size > 0) { instance.m_output_buffer = new char[output_block_size]; memset(instance.m_output_buffer, 0, output_block_size); } int output_block_offset = 0; for (int i = 0; i < graph_outputs.size(); i++) { instance.m_node_descriptor->m_inputs[i].m_reference.ptr = &instance.m_output_buffer[output_block_offset]; output_block_offset += sizeof(void*); } // // connecton data storage // size_t connection_data_storage_size = 0; for (const BlendTreeConnectionResource& connection : m_blend_tree_resource.GetConnections()) { const AnimNodeResource* source_node = m_blend_tree_resource.GetNode(connection.source_node_index); Socket* source_socket = source_node->m_socket_accessor->GetOutputSocket( connection.source_socket_name.c_str()); NodeSocketPair source_socket_pair{source_node, source_socket->m_name}; if (node_offset_map.find(source_socket_pair) == node_offset_map.end()) { node_offset_map.insert( {source_socket_pair, connection_data_storage_size}); connection_data_storage_size += source_socket->m_type_size; } } if (connection_data_storage_size > 0) { instance.m_connection_data_storage = new char[connection_data_storage_size]; memset(instance.m_connection_data_storage, 0, connection_data_storage_size); } } void AnimGraphResource::CreateBlendTreeConnectionInstances( AnimGraphBlendTree& instance, NodeSocketDataOffsetMap& node_offset_map) const { std::vector instance_node_descriptors( m_blend_tree_resource.GetNumNodes(), nullptr); for (int i = 0; i < m_blend_tree_resource.GetNumNodes(); i++) { instance_node_descriptors[i] = AnimNodeDescriptorFactory( m_blend_tree_resource.GetNode(i)->m_node_type_name, instance.m_nodes[i]); if (i > 1 && m_blend_tree_resource.GetNode(i)->m_node_type_name == "BlendTree") { instance_node_descriptors[i]->m_inputs = m_blend_tree_resource.GetNode(i)->m_socket_accessor->m_inputs; instance_node_descriptors[i]->m_outputs = m_blend_tree_resource.GetNode(i)->m_socket_accessor->m_outputs; } } instance_node_descriptors[0]->m_inputs = instance.m_node_descriptor->m_inputs; instance_node_descriptors[1]->m_outputs = instance.m_node_descriptor->m_outputs; for (const BlendTreeConnectionResource& connection : m_blend_tree_resource.GetConnections()) { NodeDescriptorBase* source_node_descriptor = instance_node_descriptors[connection.source_node_index]; NodeDescriptorBase* target_node_descriptor = instance_node_descriptors[connection.target_node_index]; AnimNode* source_node = instance.m_nodes[connection.source_node_index]; AnimNode* target_node = instance.m_nodes[connection.target_node_index]; Socket* source_socket = source_node_descriptor->GetOutputSocket( connection.source_socket_name.c_str()); Socket* target_socket = target_node_descriptor->GetInputSocket( connection.target_socket_name.c_str()); AnimGraphConnection instance_connection; if (source_node->m_node_type_name == "BlendTree") { // For embedded subgraphs we have to ensure two things: // 1. The parent graph target node must activate the source node within // the embedded subgraph. // 2. The parent graph target node must activate the AnimNode of the // parent graph which contains the embedded subgraph. Otherwise, the // embedded graph does not get evaluated. // For each case we have to add a connection here. First we insert the // default connection within the graph (i.e. for case 2). Then we alter // the source node such that it points to the node within the embedded // subgraph. AnimGraphConnection embedded_graph_activation_connection; embedded_graph_activation_connection.m_source_node = source_node; embedded_graph_activation_connection.m_source_socket_name = source_socket->m_name; embedded_graph_activation_connection.m_target_node = target_node; embedded_graph_activation_connection.m_target_socket_name = target_socket->m_name; embedded_graph_activation_connection.m_socket = *target_socket; instance.m_node_input_connections[connection.target_node_index].push_back( embedded_graph_activation_connection); instance.m_node_output_connections[connection.source_node_index] .push_back(embedded_graph_activation_connection); const AnimGraphResource* source_blend_tree_resource = dynamic_cast( m_blend_tree_resource.GetNode(connection.source_node_index)); AnimGraphBlendTree* source_blend_tree = dynamic_cast(source_node); size_t source_blend_tree_output_node_index = source_blend_tree_resource->m_blend_tree_resource .GetNodeIndexForOutputSocket(connection.source_socket_name); source_node = source_blend_tree->m_nodes[source_blend_tree_output_node_index]; instance_connection.m_crosses_hierarchy = true; } else if (target_node->m_node_type_name == "BlendTree") { // When a connection points to an embedded blend tree we have to ensure // that the embedded node knows about its connection partner in the parent // tree. This allows the embedded node to properly activate the node in // the parent graph. const AnimGraphResource* target_blend_tree_resource = dynamic_cast( m_blend_tree_resource.GetNode(connection.target_node_index)); AnimGraphBlendTree* target_blend_tree = dynamic_cast(target_node); size_t target_blend_tree_output_node_index = target_blend_tree_resource->m_blend_tree_resource .GetNodeIndexForInputSocket(connection.target_socket_name); target_node = target_blend_tree->m_nodes[target_blend_tree_output_node_index]; // Make sure that the embedded node input activates the parent graph node. std::vector& embeded_target_node_inputs = target_blend_tree ->m_node_input_connections[target_blend_tree_output_node_index]; for (size_t j = 0; j < embeded_target_node_inputs.size(); j++) { AnimGraphConnection& embedded_target_connection = embeded_target_node_inputs[j]; if (embedded_target_connection.m_source_socket_name == target_socket->m_name) { embedded_target_connection.m_source_node = source_node; embedded_target_connection.m_crosses_hierarchy = true; // In addition: make sure we expose the embedded socket to the // connection in the parent blend tree. That way // parent_tree.SetValue<>() correctly propagates the value to the // embedded node socket. target_socket = &embedded_target_connection.m_socket; } } } instance_connection.m_source_node = source_node; instance_connection.m_source_socket_name = source_socket->m_name; instance_connection.m_target_node = target_node; instance_connection.m_target_socket_name = target_socket->m_name; instance_connection.m_socket = *target_socket; size_t socket_data_offset = 0; if (connection.source_node_index == 1) { instance_connection.m_socket = *target_socket; } else { if (connection.target_node_index == 0) { instance_connection.m_socket = *source_socket; } NodeSocketPair node_socket_pair{ m_blend_tree_resource.GetNode(connection.source_node_index), source_socket->m_name}; NodeSocketDataOffsetMap::const_iterator socket_data_offset_iter = node_offset_map.find(node_socket_pair); if (socket_data_offset_iter != node_offset_map.end()) { socket_data_offset = socket_data_offset_iter->second; } else { std::cerr << "Error: could not find data offset for node '" << source_node->m_name << "' and socket '" << source_socket->m_name << "'." << std::endl; assert(false); abort(); } *target_socket->m_reference.ptr_ptr = &instance.m_connection_data_storage[socket_data_offset]; *source_socket->m_reference.ptr_ptr = &instance.m_connection_data_storage[socket_data_offset]; if (source_socket->m_type == SocketType::SocketTypeAnimation) { instance.m_animdata_blocks.push_back( (AnimData*)(&instance .m_connection_data_storage[socket_data_offset])); } } instance.m_node_input_connections[connection.target_node_index].push_back( instance_connection); instance.m_node_output_connections[connection.source_node_index].push_back( instance_connection); } // // const node inputs // std::vector const_inputs = m_blend_tree_resource.GetConstantNodeInputs(instance_node_descriptors); size_t const_node_inputs_buffer_size = 0; for (auto& const_input : const_inputs) { if (const_input->m_type == SocketType::SocketTypeString) { // TODO: implement string const node input support std::cerr << "Error: const inputs for strings not yet implemented!" << std::endl; abort(); } const_node_inputs_buffer_size += const_input->m_type_size; } if (const_node_inputs_buffer_size > 0) { instance.m_const_node_inputs = new char[const_node_inputs_buffer_size]; memset(instance.m_const_node_inputs, '\0', const_node_inputs_buffer_size); } size_t const_input_buffer_offset = 0; for (auto& i : const_inputs) { Socket* const_input = i; // TODO: implement string const node input support assert(const_input->m_type != SocketType::SocketTypeString); *const_input->m_reference.ptr_ptr = &instance.m_const_node_inputs[const_input_buffer_offset]; memcpy( *const_input->m_reference.ptr_ptr, &const_input->m_value, i->m_type_size); const_input_buffer_offset += i->m_type_size; } for (int i = 0; i < m_blend_tree_resource.GetNumNodes(); i++) { delete instance_node_descriptors[i]; } } void AnimGraphResource::SetRuntimeNodeProperties( AnimGraphBlendTree& result) const { for (int i = 2; i < m_blend_tree_resource.GetNumNodes(); i++) { const AnimNodeResource* node_resource = m_blend_tree_resource.GetNode(i); NodeDescriptorBase* node_instance_accessor = AnimNodeDescriptorFactory( node_resource->m_node_type_name, result.m_nodes[i]); std::vector& resource_properties = node_resource->m_socket_accessor->m_properties; for (const auto& property : resource_properties) { const std::string& name = property.m_name; switch (property.m_type) { case SocketType::SocketTypeBool: node_instance_accessor->SetProperty( name.c_str(), property.m_value.flag); break; case SocketType::SocketTypeInt: node_instance_accessor->SetProperty( name.c_str(), property.m_value.int_value); break; case SocketType::SocketTypeFloat: node_instance_accessor->SetProperty( name.c_str(), property.m_value.float_value); break; case SocketType::SocketTypeVec3: node_instance_accessor->SetProperty( name.c_str(), property.m_value.vec3); break; case SocketType::SocketTypeQuat: node_instance_accessor->SetProperty( name.c_str(), property.m_value.quat); break; case SocketType::SocketTypeString: node_instance_accessor->SetProperty( name.c_str(), property.m_value_string); break; default: std::cerr << "Invalid socket type " << static_cast(property.m_type) << std::endl; } } delete node_instance_accessor; } } bool AnimGraphResource::SaveStateMachineResourceToFile( const char* filename) const { assert(false && "Not yet implemented"); return false; } bool AnimGraphResource::LoadStateMachineResourceFromJson( nlohmann::json const& json_data) { assert(false && "Not yet implemented"); return false; } #pragma clang diagnostic pop