2021-11-12 20:10:56 +01:00
|
|
|
//
|
|
|
|
// Created by martin on 12.11.21.
|
|
|
|
//
|
|
|
|
|
|
|
|
#include "AnimationController.h"
|
|
|
|
|
|
|
|
#include <imgui.h>
|
|
|
|
#include <ozz/animation/runtime/blending_job.h>
|
|
|
|
#include <ozz/animation/runtime/local_to_model_job.h>
|
|
|
|
#include <ozz/animation/runtime/sampling_job.h>
|
|
|
|
|
2021-11-12 22:12:25 +01:00
|
|
|
#include "AnimNodes/AnimSamplerNode.h"
|
|
|
|
#include "AnimNodes/BlendNode.h"
|
|
|
|
#include "AnimNodes/SpeedScaleNode.h"
|
2021-11-12 20:10:56 +01:00
|
|
|
#include "SkinnedMesh.h"
|
|
|
|
|
|
|
|
AnimationController::AnimationController(SkinnedMesh* skinned_mesh)
|
2021-11-12 21:33:53 +01:00
|
|
|
: m_current_time(0.f), m_skinned_mesh(skinned_mesh) {
|
2021-11-12 20:10:56 +01:00
|
|
|
const int num_soa_joints = skinned_mesh->m_skeleton.num_soa_joints();
|
|
|
|
const int num_joints = skinned_mesh->m_skeleton.num_joints();
|
|
|
|
skinned_mesh->m_local_matrices.resize(num_soa_joints);
|
|
|
|
skinned_mesh->m_model_matrices.resize(num_joints);
|
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
m_local_matrices.resize(num_soa_joints);
|
2021-11-12 20:10:56 +01:00
|
|
|
|
|
|
|
assert(skinned_mesh->m_animations.size() > 1);
|
|
|
|
SetBlendAnims(skinned_mesh->m_animations[1], skinned_mesh->m_animations[0]);
|
|
|
|
ResetAnims();
|
|
|
|
|
2021-11-12 21:54:21 +01:00
|
|
|
AnimSamplerNode* sampler_node0 = new AnimSamplerNode(this);
|
|
|
|
sampler_node0->m_name = "AnimSampler0";
|
|
|
|
sampler_node0->SetAnimation(skinned_mesh->m_animations[1]);
|
|
|
|
m_anim_nodes.push_back(sampler_node0);
|
|
|
|
|
|
|
|
AnimSamplerNode* sampler_node1 = new AnimSamplerNode(this);
|
2021-11-13 00:08:32 +01:00
|
|
|
sampler_node1->m_name = "AnimSampler1";
|
2021-11-12 21:54:21 +01:00
|
|
|
sampler_node1->SetAnimation(skinned_mesh->m_animations[2]);
|
|
|
|
m_anim_nodes.push_back(sampler_node1);
|
2021-11-12 21:33:53 +01:00
|
|
|
|
|
|
|
SpeedScaleNode* speed_node = new SpeedScaleNode(this);
|
|
|
|
speed_node->m_name = "SpeedNode0";
|
2021-11-12 21:54:21 +01:00
|
|
|
speed_node->m_input_node = sampler_node0;
|
2021-11-12 21:33:53 +01:00
|
|
|
m_anim_nodes.push_back(speed_node);
|
|
|
|
|
2021-11-12 21:54:21 +01:00
|
|
|
BlendNode* blend_node = new BlendNode(this);
|
|
|
|
blend_node->m_name = "Blend0";
|
|
|
|
blend_node->m_input_A = speed_node;
|
|
|
|
blend_node->m_input_B = sampler_node1;
|
2021-11-13 00:08:32 +01:00
|
|
|
blend_node->m_sync_inputs = true;
|
2021-11-12 21:54:21 +01:00
|
|
|
m_anim_nodes.push_back(blend_node);
|
|
|
|
|
2021-11-13 00:08:32 +01:00
|
|
|
SpeedScaleNode* speed_node1 = new SpeedScaleNode(this);
|
|
|
|
speed_node1->m_name = "SpeedNode1";
|
|
|
|
speed_node1->m_input_node = blend_node;
|
|
|
|
m_anim_nodes.push_back(speed_node1);
|
|
|
|
|
|
|
|
m_output_node = speed_node1;
|
2021-11-12 22:12:25 +01:00
|
|
|
|
|
|
|
m_output_node->CollectNodeOrdering(m_ordered_nodes);
|
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
m_output_node->Reset();
|
2021-11-12 20:10:56 +01:00
|
|
|
}
|
|
|
|
|
2021-11-12 21:33:53 +01:00
|
|
|
|
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
AnimationController::~AnimationController() {
|
2021-11-12 21:33:53 +01:00
|
|
|
while (m_anim_nodes.size() > 0) {
|
|
|
|
delete m_anim_nodes[m_anim_nodes.size() - 1];
|
|
|
|
m_anim_nodes.pop_back();
|
2021-11-12 20:10:56 +01:00
|
|
|
}
|
2021-11-12 21:33:53 +01:00
|
|
|
|
|
|
|
m_output_node = nullptr;
|
2021-11-12 21:16:43 +01:00
|
|
|
}
|
2021-11-12 20:10:56 +01:00
|
|
|
|
2021-11-13 00:08:32 +01:00
|
|
|
void AnimationController::ResetAnims() {
|
|
|
|
for (int i = 0; i < m_ordered_nodes.size(); i++) {
|
|
|
|
m_ordered_nodes[i]->Reset();
|
|
|
|
}
|
|
|
|
}
|
2021-11-12 21:33:53 +01:00
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
void AnimationController::Update(float dt) {
|
|
|
|
if (m_output_node == nullptr) {
|
|
|
|
return;
|
2021-11-12 20:10:56 +01:00
|
|
|
}
|
|
|
|
|
2021-11-13 00:08:32 +01:00
|
|
|
m_output_node->UpdateIsSynced(false);
|
|
|
|
|
|
|
|
for (int i = m_ordered_nodes.size() - 1; i >= 0; i--) {
|
|
|
|
AnimNode* node = m_ordered_nodes[i];
|
|
|
|
if (node->m_is_time_synced) {
|
|
|
|
node->EvalAnimDuration();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m_output_node->UpdateTime(dt);
|
2021-11-12 21:33:53 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-11-12 20:10:56 +01:00
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
void AnimationController::Evaluate() {
|
|
|
|
if (m_output_node == nullptr) {
|
|
|
|
return;
|
2021-11-12 20:10:56 +01:00
|
|
|
}
|
2021-11-12 21:16:43 +01:00
|
|
|
m_output_node->Evaluate(&m_local_matrices);
|
2021-11-12 20:10:56 +01:00
|
|
|
|
|
|
|
// convert joint matrices from local to model space
|
|
|
|
ozz::animation::LocalToModelJob ltm_job;
|
|
|
|
ltm_job.skeleton = &m_skinned_mesh->m_skeleton;
|
2021-11-12 21:16:43 +01:00
|
|
|
ltm_job.input = make_span(m_local_matrices);
|
2021-11-12 20:10:56 +01:00
|
|
|
ltm_job.output = make_span(m_skinned_mesh->m_model_matrices);
|
|
|
|
ltm_job.Run();
|
|
|
|
};
|
|
|
|
|
2021-11-12 21:33:53 +01:00
|
|
|
|
|
|
|
|
2021-11-12 20:10:56 +01:00
|
|
|
void AnimationController::DrawDebugUi() {
|
2021-11-13 00:08:32 +01:00
|
|
|
ImGui::SetNextWindowSize(ImVec2(500, 300), ImGuiCond_FirstUseEver);
|
2021-11-12 20:10:56 +01:00
|
|
|
ImGui::Begin("AnimationController");
|
2021-11-13 00:08:32 +01:00
|
|
|
|
|
|
|
if (ImGui::Button("Reset")) {
|
|
|
|
ResetAnims();
|
|
|
|
}
|
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
if (m_output_node && ImGui::TreeNode("Output")) {
|
|
|
|
m_output_node->DrawDebugUi();
|
2021-11-12 20:10:56 +01:00
|
|
|
|
2021-11-12 21:16:43 +01:00
|
|
|
ImGui::TreePop();
|
2021-11-12 20:10:56 +01:00
|
|
|
}
|
|
|
|
|
2021-11-12 22:12:25 +01:00
|
|
|
if (m_output_node && ImGui::TreeNode("Nodes")) {
|
2021-11-13 00:08:32 +01:00
|
|
|
ImGui::Columns(4, "NodeOverview"); // 4-ways, with border
|
|
|
|
ImGui::Separator();
|
|
|
|
ImGui::Text("Name"); ImGui::NextColumn();
|
|
|
|
ImGui::Text("Synced"); ImGui::NextColumn();
|
|
|
|
ImGui::Text("Duration"); ImGui::NextColumn();
|
|
|
|
ImGui::Text("Time"); ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::Separator();
|
|
|
|
static int selected = -1;
|
|
|
|
|
2021-11-12 22:12:25 +01:00
|
|
|
for (int i = 0; i < m_ordered_nodes.size(); i++) {
|
2021-11-13 00:08:32 +01:00
|
|
|
AnimNode* node = m_ordered_nodes[i];
|
|
|
|
if (ImGui::Selectable(node->m_name.c_str(), selected == i, ImGuiSelectableFlags_SpanAllColumns))
|
|
|
|
selected = i;
|
|
|
|
bool hovered = ImGui::IsItemHovered();
|
|
|
|
ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::Text(node->m_is_time_synced ? "X" : "-"); ImGui::NextColumn();
|
|
|
|
|
|
|
|
ImGui::PushID((void*)&node->m_anim_duration);
|
|
|
|
ImGui::Text("%2.3f", node->m_anim_duration); ImGui::NextColumn();
|
|
|
|
ImGui::PopID();
|
|
|
|
|
|
|
|
ImGui::PushID((void*)&node->m_current_time);
|
|
|
|
ImGui::Text("%2.3f", node->m_current_time); ImGui::NextColumn();
|
|
|
|
ImGui::PopID();
|
2021-11-12 22:12:25 +01:00
|
|
|
}
|
|
|
|
|
2021-11-13 00:08:32 +01:00
|
|
|
ImGui::Columns(1);
|
|
|
|
ImGui::Separator();
|
|
|
|
|
2021-11-12 22:12:25 +01:00
|
|
|
ImGui::TreePop();
|
|
|
|
}
|
|
|
|
|
2021-11-12 20:10:56 +01:00
|
|
|
ImGui::End();
|
|
|
|
}
|