AnimGraphEvalTests now properly evaluates.
This commit is contained in:
parent
1870a9d214
commit
b4eda31242
@ -48,6 +48,13 @@ struct AnimGraphBlendTree : public AnimNode {
|
|||||||
|
|
||||||
// AnimNode overrides
|
// AnimNode overrides
|
||||||
bool Init(AnimGraphContext& context) override;
|
bool Init(AnimGraphContext& context) override;
|
||||||
|
|
||||||
|
/// Determines which nodes in the BlendTree are active.
|
||||||
|
///
|
||||||
|
/// Note: this does not use the provided input_connections, instead it marks
|
||||||
|
/// all nodes directly connected to the BlendTree outputs as active and then
|
||||||
|
/// propagates the node state throught the tree. For this each active node's
|
||||||
|
/// AnimNode::MarkActiveInputs() gets called.
|
||||||
void MarkActiveInputs(
|
void MarkActiveInputs(
|
||||||
const std::vector<AnimGraphConnection>& input_connections) override;
|
const std::vector<AnimGraphConnection>& input_connections) override;
|
||||||
void CalcSyncTrack(
|
void CalcSyncTrack(
|
||||||
|
@ -33,12 +33,12 @@ struct Blend2Node : public AnimNode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (input.m_target_socket_name == "Input0" && *i_blend_weight < 0.999) {
|
if (input.m_target_socket_name == "Input0" && *i_blend_weight < 0.999) {
|
||||||
input_node->m_state = AnimNodeEvalState::Activated;
|
input_node->Activate(m_tick_number);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (input.m_target_socket_name == "Input1" && *i_blend_weight > 0.001) {
|
if (input.m_target_socket_name == "Input1" && *i_blend_weight > 0.001) {
|
||||||
input_node->m_state = AnimNodeEvalState::Activated;
|
input_node->Activate(m_tick_number);
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -59,6 +59,8 @@ struct NodeDescriptor<Blend2Node> : public NodeDescriptorBase {
|
|||||||
RegisterProperty("Sync", &node->m_sync_blend);
|
RegisterProperty("Sync", &node->m_sync_blend);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
|
|
||||||
void UpdateFlags() override {
|
void UpdateFlags() override {
|
||||||
Socket* weight_input_socket = FindSocket("Weight", m_inputs);
|
Socket* weight_input_socket = FindSocket("Weight", m_inputs);
|
||||||
assert(weight_input_socket != nullptr);
|
assert(weight_input_socket != nullptr);
|
||||||
@ -104,6 +106,8 @@ struct NodeDescriptor<SpeedScaleNode> : public NodeDescriptorBase {
|
|||||||
|
|
||||||
RegisterOutput("Output", &node->o_output);
|
RegisterOutput("Output", &node->o_output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -132,6 +136,8 @@ struct NodeDescriptor<AnimSamplerNode> : public NodeDescriptorBase {
|
|||||||
|
|
||||||
RegisterProperty("Filename", &node->m_filename);
|
RegisterProperty("Filename", &node->m_filename);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -159,6 +165,8 @@ struct NodeDescriptor<LockTranslationNode> : public NodeDescriptorBase {
|
|||||||
RegisterProperty("LockAxisY", &node->m_lock_y);
|
RegisterProperty("LockAxisY", &node->m_lock_y);
|
||||||
RegisterProperty("LockAxisZ", &node->m_lock_z);
|
RegisterProperty("LockAxisZ", &node->m_lock_z);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -177,6 +185,8 @@ struct NodeDescriptor<ConstScalarNode> : public NodeDescriptorBase {
|
|||||||
RegisterOutput("ScalarOutput", &node->o_value);
|
RegisterOutput("ScalarOutput", &node->o_value);
|
||||||
RegisterProperty("ScalarValue", &node->value);
|
RegisterProperty("ScalarValue", &node->value);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -202,6 +212,8 @@ struct NodeDescriptor<MathAddNode> : public NodeDescriptorBase {
|
|||||||
RegisterInput("Input1", &node->i_input1);
|
RegisterInput("Input1", &node->i_input1);
|
||||||
RegisterOutput("Output", &node->o_output);
|
RegisterOutput("Output", &node->o_output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
//
|
//
|
||||||
@ -232,6 +244,8 @@ struct NodeDescriptor<MathFloatToVec3Node> : public NodeDescriptorBase {
|
|||||||
RegisterInput("Input2", &node->i_input2);
|
RegisterInput("Input2", &node->i_input2);
|
||||||
RegisterOutput("Output", &node->o_output);
|
RegisterOutput("Output", &node->o_output);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
virtual ~NodeDescriptor() = default;
|
||||||
};
|
};
|
||||||
|
|
||||||
AnimNode* AnimNodeFactory(const std::string& name);
|
AnimNode* AnimNodeFactory(const std::string& name);
|
||||||
|
@ -45,8 +45,7 @@ struct AnimNode {
|
|||||||
for (const auto& input : input_connections) {
|
for (const auto& input : input_connections) {
|
||||||
AnimNode* input_node = input.m_source_node;
|
AnimNode* input_node = input.m_source_node;
|
||||||
if (input_node != nullptr) {
|
if (input_node != nullptr) {
|
||||||
input_node->m_tick_number = m_tick_number;
|
input_node->Activate(m_tick_number);
|
||||||
input_node->m_state = AnimNodeEvalState::Activated;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -64,13 +63,18 @@ struct AnimNode {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void UpdateTime(float time_last, float time_now) {
|
virtual void UpdateTime(const float time_last, const float time_now) {
|
||||||
m_time_last = time_last;
|
m_time_last = time_last;
|
||||||
m_time_now = time_now;
|
m_time_now = time_now;
|
||||||
m_state = AnimNodeEvalState::TimeUpdated;
|
m_state = AnimNodeEvalState::TimeUpdated;
|
||||||
}
|
}
|
||||||
|
|
||||||
virtual void Evaluate(AnimGraphContext& context){};
|
virtual void Evaluate(AnimGraphContext& context) {};
|
||||||
|
|
||||||
|
void Activate(const int tick_number) {
|
||||||
|
m_tick_number = tick_number;
|
||||||
|
m_state = AnimNodeEvalState::Activated;
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif //ANIMTESTBED_ANIMNODE_H
|
#endif //ANIMTESTBED_ANIMNODE_H
|
||||||
|
@ -4,7 +4,6 @@
|
|||||||
|
|
||||||
#include "AnimGraph/AnimGraphBlendTree.h"
|
#include "AnimGraph/AnimGraphBlendTree.h"
|
||||||
#include "AnimGraph/AnimGraphResource.h"
|
#include "AnimGraph/AnimGraphResource.h"
|
||||||
#include "AnimGraphEditor/AnimGraphEditor.h"
|
|
||||||
#include "catch.hpp"
|
#include "catch.hpp"
|
||||||
#include "ozz/animation/offline/animation_builder.h"
|
#include "ozz/animation/offline/animation_builder.h"
|
||||||
#include "ozz/animation/offline/raw_animation.h"
|
#include "ozz/animation/offline/raw_animation.h"
|
||||||
@ -14,7 +13,6 @@
|
|||||||
#include "ozz/animation/runtime/sampling_job.h"
|
#include "ozz/animation/runtime/sampling_job.h"
|
||||||
#include "ozz/animation/runtime/skeleton.h"
|
#include "ozz/animation/runtime/skeleton.h"
|
||||||
#include "ozz/base/io/archive.h"
|
#include "ozz/base/io/archive.h"
|
||||||
#include "ozz/base/io/stream.h"
|
|
||||||
#include "ozz/base/log.h"
|
#include "ozz/base/log.h"
|
||||||
#include "ozz/base/maths/soa_transform.h"
|
#include "ozz/base/maths/soa_transform.h"
|
||||||
|
|
||||||
@ -66,7 +64,7 @@ struct SimpleAnimFixture {
|
|||||||
bone0_translations.push_back(translation_key);
|
bone0_translations.push_back(translation_key);
|
||||||
|
|
||||||
translation_key.time = 1.f;
|
translation_key.time = 1.f;
|
||||||
translation_key.value = ozz::math::Float3(1.f, 0.f, 9.f);
|
translation_key.value = ozz::math::Float3(1.f, 0.f, 0.f);
|
||||||
bone0_translations.push_back(translation_key);
|
bone0_translations.push_back(translation_key);
|
||||||
|
|
||||||
bone0_track.translations = bone0_translations;
|
bone0_track.translations = bone0_translations;
|
||||||
@ -214,31 +212,97 @@ TEST_CASE_METHOD(
|
|||||||
|
|
||||||
AnimData graph_anim_output;
|
AnimData graph_anim_output;
|
||||||
graph_anim_output.m_local_matrices.resize(skeleton->num_joints());
|
graph_anim_output.m_local_matrices.resize(skeleton->num_joints());
|
||||||
blend_tree.SetOutput("GraphOutput", &graph_anim_output);
|
blend_tree.SetOutput("Output", &graph_anim_output);
|
||||||
|
|
||||||
// Evaluate graph
|
WHEN("Blend Weight == 0.") {
|
||||||
graph_float_input = 0.1f;
|
// Evaluate graph
|
||||||
|
graph_float_input = 0.f;
|
||||||
|
|
||||||
blend_tree.StartUpdateTick();
|
blend_tree.StartUpdateTick();
|
||||||
blend_tree.MarkActiveInputs(blend_tree.GetGraphOutputConnections());
|
blend_tree.MarkActiveInputs({});
|
||||||
|
|
||||||
CHECK(
|
THEN("Only Blend2 and first input of Blend2 node is active.") {
|
||||||
blend_tree.m_nodes[trans_x_node_index]->m_state
|
CHECK(
|
||||||
== AnimNodeEvalState::Activated);
|
blend_tree.m_nodes[trans_x_node_index]->m_state
|
||||||
CHECK(
|
== AnimNodeEvalState::Activated);
|
||||||
blend_tree.m_nodes[trans_y_node_index]->m_state
|
CHECK(
|
||||||
== AnimNodeEvalState::Activated);
|
blend_tree.m_nodes[trans_y_node_index]->m_state
|
||||||
CHECK(
|
== AnimNodeEvalState::Deactivated);
|
||||||
blend_tree.m_nodes[blend_node_index]->m_state
|
CHECK(
|
||||||
== AnimNodeEvalState::Activated);
|
blend_tree.m_nodes[blend_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Activated);
|
||||||
|
}
|
||||||
|
|
||||||
blend_tree.UpdateTime(0.0, 0.5f);
|
blend_tree.UpdateTime(0.0, 0.5f);
|
||||||
blend_tree.Evaluate(graph_context);
|
blend_tree.Evaluate(graph_context);
|
||||||
|
|
||||||
CHECK(
|
CHECK(
|
||||||
graph_anim_output.m_local_matrices[0].translation.x[0]
|
graph_anim_output.m_local_matrices[0].translation.x[0]
|
||||||
== Approx(0.5).margin(0.1));
|
== Approx(0.5).margin(0.01));
|
||||||
CHECK(
|
CHECK(
|
||||||
graph_anim_output.m_local_matrices[0].translation.y[0]
|
graph_anim_output.m_local_matrices[0].translation.y[0]
|
||||||
== Approx(0.05).margin(0.01));
|
== Approx(0.0).margin(0.01));
|
||||||
|
}
|
||||||
|
|
||||||
|
WHEN("Blend Weight 0.1") {
|
||||||
|
// Evaluate graph
|
||||||
|
graph_float_input = 0.1f;
|
||||||
|
|
||||||
|
blend_tree.StartUpdateTick();
|
||||||
|
blend_tree.MarkActiveInputs({});
|
||||||
|
|
||||||
|
THEN("All nodes are active.") {
|
||||||
|
CHECK(
|
||||||
|
blend_tree.m_nodes[trans_x_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Activated);
|
||||||
|
CHECK(
|
||||||
|
blend_tree.m_nodes[trans_y_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Activated);
|
||||||
|
CHECK(
|
||||||
|
blend_tree.m_nodes[blend_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Activated);
|
||||||
|
}
|
||||||
|
|
||||||
|
blend_tree.UpdateTime(0.0, 0.5f);
|
||||||
|
blend_tree.Evaluate(graph_context);
|
||||||
|
|
||||||
|
CHECK(
|
||||||
|
graph_anim_output.m_local_matrices[0].translation.x[0]
|
||||||
|
== Approx(0.45).margin(0.01));
|
||||||
|
CHECK(
|
||||||
|
graph_anim_output.m_local_matrices[0].translation.y[0]
|
||||||
|
== Approx(0.05).margin(0.01));
|
||||||
|
}
|
||||||
|
|
||||||
|
WHEN("Blend Weight 1.") {
|
||||||
|
// Evaluate graph
|
||||||
|
graph_float_input = 1.f;
|
||||||
|
|
||||||
|
blend_tree.StartUpdateTick();
|
||||||
|
blend_tree.MarkActiveInputs({});
|
||||||
|
|
||||||
|
THEN("Only Blend2 and second input of Blend2 are active.") {
|
||||||
|
CHECK(
|
||||||
|
blend_tree.m_nodes[trans_x_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Deactivated);
|
||||||
|
CHECK(
|
||||||
|
blend_tree.m_nodes[trans_y_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Activated);
|
||||||
|
CHECK(
|
||||||
|
blend_tree.m_nodes[blend_node_index]->m_state
|
||||||
|
== AnimNodeEvalState::Activated);
|
||||||
|
}
|
||||||
|
|
||||||
|
blend_tree.UpdateTime(0.0, 0.5f);
|
||||||
|
blend_tree.Evaluate(graph_context);
|
||||||
|
|
||||||
|
CHECK(
|
||||||
|
graph_anim_output.m_local_matrices[0].translation.x[0]
|
||||||
|
== Approx(0.).margin(0.01));
|
||||||
|
CHECK(
|
||||||
|
graph_anim_output.m_local_matrices[0].translation.y[0]
|
||||||
|
== Approx(0.5).margin(0.01));
|
||||||
|
}
|
||||||
|
|
||||||
|
delete blend_tree_resource;
|
||||||
}
|
}
|
Loading…
x
Reference in New Issue
Block a user