Skip to content

Commit

Permalink
Implements motion extraction unit tests.
Browse files Browse the repository at this point in the history
  • Loading branch information
guillaumeblanc committed Apr 1, 2024
1 parent 3890f7b commit 76d7307
Showing 1 changed file with 319 additions and 6 deletions.
325 changes: 319 additions & 6 deletions test/animation/offline/motion_extractor_tests.cc
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,9 @@
#include "ozz/animation/offline/raw_track.h"
#include "ozz/animation/offline/skeleton_builder.h"
#include "ozz/animation/runtime/skeleton.h"
#include "ozz/base/maths/gtest_math_helper.h"
#include "ozz/base/maths/math_constant.h"
#include "ozz/base/maths/math_ex.h"
#include "ozz/base/memory/unique_ptr.h"

using ozz::animation::Skeleton;
Expand All @@ -46,19 +48,22 @@ using ozz::animation::offline::SkeletonBuilder;
TEST(Error, MotionExtractor) {
RawAnimation input;
input.tracks.resize(1);
ASSERT_TRUE(input.Validate());

RawSkeleton raw_skeleton;
raw_skeleton.roots.resize(1);
const Skeleton skeleton = std::move(*SkeletonBuilder()(raw_skeleton));
ASSERT_TRUE(raw_skeleton.Validate());
const auto skeleton = SkeletonBuilder()(raw_skeleton);
ASSERT_TRUE(skeleton);

RawFloat3Track motion_position;
RawQuaternionTrack motion_rotation;
RawAnimation output;

{ // Same input and output.
MotionExtractor extractor;
EXPECT_FALSE(
extractor(input, skeleton, &motion_position, &motion_rotation, &input));
EXPECT_FALSE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&input));
}

{
Expand All @@ -69,15 +74,323 @@ TEST(Error, MotionExtractor) {
EXPECT_FALSE(input_invalid.Validate());

MotionExtractor extractor;
EXPECT_FALSE(extractor(input_invalid, skeleton, &motion_position,
EXPECT_FALSE(extractor(input_invalid, *skeleton, &motion_position,
&motion_rotation, &output));
}

{
// Non matching animation & skeleton
RawSkeleton raw_skeleton_2;
raw_skeleton_2.roots.resize(2);
const auto skeleton_2 = SkeletonBuilder()(raw_skeleton_2);
ASSERT_TRUE(skeleton_2);

MotionExtractor extractor;
EXPECT_FALSE(extractor(input, *skeleton_2, &motion_position,
&motion_rotation, &output));
}

{ // Null output
MotionExtractor extractor;
EXPECT_FALSE(
extractor(input, *skeleton, nullptr, &motion_rotation, &output));
EXPECT_FALSE(
extractor(input, *skeleton, &motion_position, nullptr, &output));
EXPECT_FALSE(extractor(input, *skeleton, &motion_position, &motion_rotation,
nullptr));
}

{ // Invalid root
MotionExtractor extractor;
extractor.root_joint = 93;
EXPECT_FALSE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&output));
}

{ // Valid
MotionExtractor extractor;
EXPECT_TRUE(extractor(input, skeleton, &motion_position, &motion_rotation,
EXPECT_TRUE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&output));
}
}

TEST(Extract, MotionExtractor) {}
TEST(Extract, MotionExtractor) {
RawAnimation input;
input.duration = 46.f;
input.name = "test";
input.tracks.resize(2);

input.tracks[0].translations.push_back({.5f, {1.f, 2.f, 3.f}});
input.tracks[0].translations.push_back({5.f, {4.f, 5.f, 6.f}});
input.tracks[0].rotations.push_back(
{1.f, ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f,
ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f})});
input.tracks[0].rotations.push_back(
{12.f, ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f,
1.1f * ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f})});
input.tracks[0].rotations.push_back(
{46.f, ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f,
1.2f * ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f})});

input.tracks[1].translations.push_back({2.f, {7.f, 8.f, 9.f}});
input.tracks[1].translations.push_back({10.f, {10.f, 11.f, 12.f}});
input.tracks[1].rotations.push_back(
{23.f, ozz::math::Quaternion::FromEuler(
ozz::math::Float3(-ozz::math::kPi / 5.f, ozz::math::kPi / 7.f,
ozz::math::kPi / 2.f))});

ASSERT_TRUE(input.Validate());

RawSkeleton raw_skeleton;
raw_skeleton.roots.resize(1);
raw_skeleton.roots[0].children.resize(1);
ASSERT_TRUE(raw_skeleton.Validate());
const auto skeleton = SkeletonBuilder()(raw_skeleton);

RawFloat3Track motion_position;
RawQuaternionTrack motion_rotation;
RawAnimation baked;

const auto anim_cmpnt_eq = [](const auto& _a, const auto& _b) {
if (_a.size() != _b.size()) {
return false;
}
for (size_t i = 0; i < _a.size(); ++i) {
if (_a[i].time != _b[i].time) {
return false;
}
if (_a[i].value != _b[i].value) {
return false;
}
}
return true;
};
const auto anim_track_eq = [&anim_cmpnt_eq](const auto& _a, const auto& _b) {
return anim_cmpnt_eq(_a.translations, _b.translations) &&
anim_cmpnt_eq(_a.rotations, _b.rotations) &&
anim_cmpnt_eq(_a.scales, _b.scales);
};

{ // No extraction
const MotionExtractor extractor{
.root_joint = 0,
.position_settings = {false, false, false,
MotionExtractor::Reference::kIdentity, true},
.rotation_settings = {false, false, false,
MotionExtractor::Reference::kIdentity, true}};
EXPECT_TRUE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&baked));

// Animation
EXPECT_EQ(input.name, baked.name);
EXPECT_FLOAT_EQ(input.duration, baked.duration);
EXPECT_EQ(input.num_tracks(), baked.num_tracks());
EXPECT_TRUE(anim_track_eq(input.tracks[0], baked.tracks[0]));
EXPECT_TRUE(anim_track_eq(input.tracks[1], baked.tracks[1]));

// Motion
const auto& positions = motion_position.keyframes;
ASSERT_EQ(positions.size(), 2u);
EXPECT_FLOAT_EQ(positions[0].ratio, .5f / 46.f);
EXPECT_FLOAT3_EQ(positions[0].value, 0.f, 0.f, 0.f);
EXPECT_FLOAT_EQ(positions[1].ratio, 5.f / 46.f);
EXPECT_FLOAT3_EQ(positions[1].value, 0.f, 0.f, 0.f);

const auto& rotations = motion_rotation.keyframes;
ASSERT_EQ(rotations.size(), 3u);
EXPECT_FLOAT_EQ(rotations[0].ratio, 1.f / 46.f);
EXPECT_QUATERNION_EQ(rotations[0].value, 0.f, 0.f, 0.f, 1.f);
EXPECT_FLOAT_EQ(rotations[1].ratio, 12.f / 46.f);
EXPECT_QUATERNION_EQ(rotations[1].value, 0.f, 0.f, 0.f, 1.f);
EXPECT_FLOAT_EQ(rotations[2].ratio, 46.f / 46.f);
EXPECT_QUATERNION_EQ(rotations[2].value, 0.f, 0.f, 0.f, 1.f);
}

{ // No baking
const MotionExtractor extractor{
.root_joint = 0,
.position_settings = {true, true, true,
MotionExtractor::Reference::kIdentity, false},
.rotation_settings = {true, true, true,
MotionExtractor::Reference::kIdentity, false}};
EXPECT_TRUE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&baked));

// Animation
EXPECT_EQ(input.name, baked.name);
EXPECT_FLOAT_EQ(input.duration, baked.duration);
EXPECT_EQ(input.num_tracks(), baked.num_tracks());
EXPECT_TRUE(anim_track_eq(input.tracks[0], baked.tracks[0]));
EXPECT_TRUE(anim_track_eq(input.tracks[1], baked.tracks[1]));

// Track
const auto& positions = motion_position.keyframes;
ASSERT_EQ(positions.size(), 2u);
EXPECT_FLOAT_EQ(positions[0].ratio, .5f / 46.f);
EXPECT_FLOAT3_EQ(positions[0].value, 1.f, 2.f, 3.f);
EXPECT_FLOAT_EQ(positions[1].ratio, 5.f / 46.f);
EXPECT_FLOAT3_EQ(positions[1].value, 4.f, 5.f, 6.f);

const auto& rotations = motion_rotation.keyframes;
ASSERT_EQ(rotations.size(), 3u);
EXPECT_FLOAT_EQ(rotations[0].ratio, 1.f / 46.f);
auto r1 = ozz::math::Quaternion::FromEuler(ozz::math::Float3(
ozz::math::kPi / 2.f, ozz::math::kPi / 3.f, ozz::math::kPi / 4.f));
EXPECT_QUATERNION_EQ(rotations[0].value, r1.x, r1.y, r1.z, r1.w);
auto r2 = ozz::math::Quaternion::FromEuler(
ozz::math::Float3(ozz::math::kPi / 2.f, 1.1f * ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f));
EXPECT_QUATERNION_EQ(rotations[1].value, r2.x, r2.y, r2.z, r2.w);
auto r3 = ozz::math::Quaternion::FromEuler(
ozz::math::Float3(ozz::math::kPi / 2.f, 1.2f * ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f));
EXPECT_QUATERNION_EQ(rotations[2].value, r3.x, r3.y, r3.z, r3.w);
}

{ // Extract all
const MotionExtractor extractor{
.root_joint = 0,
.position_settings = {true, true, true,
MotionExtractor::Reference::kIdentity, true},
.rotation_settings = {true, true, true,
MotionExtractor::Reference::kIdentity, true}};
EXPECT_TRUE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&baked));

// Animation
EXPECT_EQ(input.name, baked.name);
EXPECT_FLOAT_EQ(input.duration, baked.duration);
EXPECT_EQ(input.num_tracks(), baked.num_tracks());

const auto& track0 = baked.tracks[0];
ASSERT_EQ(track0.translations.size(), 2u);
EXPECT_FLOAT3_EQ(track0.translations[0].value, 0.f, 0.f, 0.f);
EXPECT_FLOAT3_EQ(track0.translations[1].value, 0.f, 0.f, 0.f);
ASSERT_EQ(track0.rotations.size(), 3u);
EXPECT_QUATERNION_EQ(track0.rotations[0].value, 0.f, 0.f, 0.f, 1.f);
EXPECT_QUATERNION_EQ(track0.rotations[1].value, 0.f, 0.f, 0.f, 1.f);
EXPECT_QUATERNION_EQ(track0.rotations[2].value, 0.f, 0.f, 0.f, 1.f);

EXPECT_TRUE(anim_track_eq(input.tracks[1], baked.tracks[1]));

// Track
const auto& positions = motion_position.keyframes;
ASSERT_EQ(positions.size(), 2u);
EXPECT_FLOAT_EQ(positions[0].ratio, .5f / 46.f);
EXPECT_FLOAT3_EQ(positions[0].value, 1.f, 2.f, 3.f);
EXPECT_FLOAT_EQ(positions[1].ratio, 5.f / 46.f);
EXPECT_FLOAT3_EQ(positions[1].value, 4.f, 5.f, 6.f);

const auto& rotations = motion_rotation.keyframes;
ASSERT_EQ(rotations.size(), 3u);
EXPECT_FLOAT_EQ(rotations[0].ratio, 1.f / 46.f);
auto mr1 = ozz::math::Quaternion::FromEuler(
{ozz::math::kPi / 2.f, ozz::math::kPi / 3.f, ozz::math::kPi / 4.f});
EXPECT_QUATERNION_EQ(rotations[0].value, mr1.x, mr1.y, mr1.z, mr1.w);
auto mr2 = ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f,
1.1f * ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f});
EXPECT_QUATERNION_EQ(rotations[1].value, mr2.x, mr2.y, mr2.z, mr2.w);
auto mr3 = ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f,
1.2f * ozz::math::kPi / 3.f,
ozz::math::kPi / 4.f});
EXPECT_QUATERNION_EQ(rotations[2].value, mr3.x, mr3.y, mr3.z, mr3.w);
}

{ // Extract only y position
const MotionExtractor extractor{
.root_joint = 0,
.position_settings = {false, true, false,
MotionExtractor::Reference::kIdentity, true},
.rotation_settings = {false, false, false,
MotionExtractor::Reference::kIdentity, true}};
EXPECT_TRUE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&baked));

// Animation
EXPECT_EQ(input.name, baked.name);
EXPECT_FLOAT_EQ(input.duration, baked.duration);
EXPECT_EQ(input.num_tracks(), baked.num_tracks());

const auto& track0 = baked.tracks[0];
ASSERT_EQ(track0.translations.size(), 2u);
EXPECT_FLOAT3_EQ(track0.translations[0].value, 1.f, 0.f, 3.f);
EXPECT_FLOAT3_EQ(track0.translations[1].value, 4.f, 0.f, 6.f);

EXPECT_TRUE(anim_cmpnt_eq(track0.rotations, track0.rotations));
EXPECT_TRUE(anim_track_eq(input.tracks[1], baked.tracks[1]));

// Motion
const auto& positions = motion_position.keyframes;
ASSERT_EQ(positions.size(), 2u);
EXPECT_FLOAT_EQ(positions[0].ratio, .5f / 46.f);
EXPECT_FLOAT3_EQ(positions[0].value, 0.f, 2.f, 0.f);
EXPECT_FLOAT_EQ(positions[1].ratio, 5.f / 46.f);
EXPECT_FLOAT3_EQ(positions[1].value, 0.f, 5.f, 0.f);

const auto& rotations = motion_rotation.keyframes;
ASSERT_EQ(rotations.size(), 3u);
EXPECT_FLOAT_EQ(rotations[0].ratio, 1.f / 46.f);
EXPECT_QUATERNION_EQ(rotations[0].value, 0.f, 0.f, 0.f, 1.f);
EXPECT_FLOAT_EQ(rotations[1].ratio, 12.f / 46.f);
EXPECT_QUATERNION_EQ(rotations[1].value, 0.f, 0.f, 0.f, 1.f);
EXPECT_FLOAT_EQ(rotations[2].ratio, 46.f / 46.f);
EXPECT_QUATERNION_EQ(rotations[2].value, 0.f, 0.f, 0.f, 1.f);
}

{ // Extract only x position and y/yaw rotation
const MotionExtractor extractor{
.root_joint = 0,
.position_settings = {true, false, false,
MotionExtractor::Reference::kIdentity, true},
.rotation_settings = {false, true, false,
MotionExtractor::Reference::kIdentity, true}};
EXPECT_TRUE(extractor(input, *skeleton, &motion_position, &motion_rotation,
&baked));

EXPECT_EQ(input.name, baked.name);
EXPECT_FLOAT_EQ(input.duration, baked.duration);
EXPECT_EQ(input.num_tracks(), baked.num_tracks());

const auto& track0 = baked.tracks[0];
ASSERT_EQ(track0.translations.size(), 2u);
// Inverse of extracted y/yaw applied to translation
EXPECT_FLOAT3_EQ(track0.translations[0].value, -3.f, 2.f, 0.f);
EXPECT_FLOAT3_EQ(track0.translations[1].value, -6.f, 5.f, 0.f);

// Y/yaw extracted
ASSERT_EQ(track0.rotations.size(), 3u);
auto r1 = ozz::math::Quaternion::FromEuler(
{0, ozz::math::kPi / 3.f, ozz::math::kPi / 4.f});
EXPECT_QUATERNION_EQ(track0.rotations[0].value, r1.x, r1.y, r1.z, r1.w);
auto r2 = ozz::math::Quaternion::FromEuler(
{0, 1.1f * ozz::math::kPi / 3.f, ozz::math::kPi / 4.f});
EXPECT_QUATERNION_EQ(track0.rotations[1].value, r2.x, r2.y, r2.z, r2.w);
auto r3 = ozz::math::Quaternion::FromEuler(
{0, 1.2f * ozz::math::kPi / 3.f, ozz::math::kPi / 4.f});
EXPECT_QUATERNION_EQ(track0.rotations[2].value, r3.x, r3.y, r3.z, r3.w);

EXPECT_TRUE(anim_track_eq(input.tracks[1], baked.tracks[1]));

// Motion
const auto& positions = motion_position.keyframes;
ASSERT_EQ(positions.size(), 2u);
EXPECT_FLOAT_EQ(positions[0].ratio, .5f / 46.f);
EXPECT_FLOAT3_EQ(positions[0].value, 1.f, 0.f, 0.f);
EXPECT_FLOAT_EQ(positions[1].ratio, 5.f / 46.f);
EXPECT_FLOAT3_EQ(positions[1].value, 4.f, 0.f, 0.f);

const auto& rotations = motion_rotation.keyframes;
ASSERT_EQ(rotations.size(), 3u);
EXPECT_FLOAT_EQ(rotations[0].ratio, 1.f / 46.f);
auto mr1 = ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f, 0, 0});
EXPECT_QUATERNION_EQ(rotations[0].value, mr1.x, mr1.y, mr1.z, mr1.w);
auto mr2 = ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f, 0, 0});
EXPECT_QUATERNION_EQ(rotations[1].value, mr2.x, mr2.y, mr2.z, mr2.w);
auto mr3 = ozz::math::Quaternion::FromEuler({ozz::math::kPi / 2.f, 0, 0});
EXPECT_QUATERNION_EQ(rotations[2].value, mr3.x, mr3.y, mr3.z, mr3.w);
}
}

0 comments on commit 76d7307

Please sign in to comment.