isle-portable/LEGO1/lego/sources/anim/legoanim.cpp
jonschz 974cd7ce7c
Implement LegoCarBuildAnimPresenter::FUN_10079160() and others (#1111)
* Implement `LegoCarBuildAnimPresenter::FUN_10079160()` and others

* Address review comments

---------

Co-authored-by: jonschz <jonschz@users.noreply.github.com>
2024-10-11 20:21:33 +02:00

926 lines
19 KiB
C++

#include "legoanim.h"
#include "mxgeometry/mxmatrix.h"
#include <limits.h>
DECOMP_SIZE_ASSERT(LegoAnimKey, 0x08)
DECOMP_SIZE_ASSERT(LegoTranslationKey, 0x14)
DECOMP_SIZE_ASSERT(LegoRotationKey, 0x18)
DECOMP_SIZE_ASSERT(LegoScaleKey, 0x14)
DECOMP_SIZE_ASSERT(LegoMorphKey, 0x0c)
DECOMP_SIZE_ASSERT(LegoUnknownKey, 0x0c)
DECOMP_SIZE_ASSERT(LegoAnimNodeData, 0x34)
DECOMP_SIZE_ASSERT(LegoAnimActorEntry, 0x08)
DECOMP_SIZE_ASSERT(LegoAnimScene, 0x24)
DECOMP_SIZE_ASSERT(LegoAnim, 0x18)
// FUNCTION: LEGO1 0x1009f000
LegoUnknownKey::LegoUnknownKey()
{
m_z = 0.0f;
}
// FUNCTION: LEGO1 0x1009f020
LegoResult LegoUnknownKey::Read(LegoStorage* p_storage)
{
LegoResult result;
if ((result = LegoAnimKey::Read(p_storage)) != SUCCESS) {
return result;
}
result = p_storage->Read(&m_z, sizeof(m_z));
return result == SUCCESS ? SUCCESS : result;
}
// FUNCTION: LEGO1 0x1009f0a0
LegoAnimScene::LegoAnimScene()
{
m_unk0x00 = 0;
m_unk0x04 = NULL;
m_unk0x08 = 0;
m_unk0x0c = NULL;
m_unk0x10 = 0;
m_unk0x14 = NULL;
m_unk0x18 = 0;
m_unk0x1c = 0;
m_unk0x20 = 0;
}
// FUNCTION: LEGO1 0x1009f0d0
LegoAnimScene::~LegoAnimScene()
{
if (m_unk0x04 != NULL) {
delete[] m_unk0x04;
m_unk0x04 = NULL;
}
if (m_unk0x0c != NULL) {
delete[] m_unk0x0c;
m_unk0x0c = NULL;
}
if (m_unk0x14 != NULL) {
delete[] m_unk0x14;
m_unk0x14 = NULL;
}
}
// FUNCTION: LEGO1 0x1009f200
LegoResult LegoAnimScene::Read(LegoStorage* p_storage)
{
LegoResult result;
LegoS32 i;
if ((result = p_storage->Read(&m_unk0x00, sizeof(m_unk0x00))) != SUCCESS) {
return result;
}
if (m_unk0x00) {
m_unk0x04 = new LegoTranslationKey[m_unk0x00];
for (i = 0; i < m_unk0x00; i++) {
if ((result = m_unk0x04[i].Read(p_storage)) != SUCCESS) {
goto done;
}
}
}
if ((result = p_storage->Read(&m_unk0x08, sizeof(m_unk0x08))) != SUCCESS) {
return result;
}
if (m_unk0x08) {
m_unk0x0c = new LegoTranslationKey[m_unk0x08];
for (i = 0; i < m_unk0x08; i++) {
if ((result = m_unk0x0c[i].Read(p_storage)) != SUCCESS) {
goto done;
}
}
}
if ((result = p_storage->Read(&m_unk0x10, sizeof(m_unk0x10))) != SUCCESS) {
return result;
}
if (m_unk0x10) {
m_unk0x14 = new LegoUnknownKey[m_unk0x10];
for (i = 0; i < m_unk0x10; i++) {
if ((result = m_unk0x14[i].Read(p_storage)) != SUCCESS) {
goto done;
}
}
}
return SUCCESS;
done:
if (m_unk0x04 != NULL) {
delete[] m_unk0x04;
m_unk0x00 = 0;
m_unk0x04 = NULL;
}
if (m_unk0x0c != NULL) {
delete[] m_unk0x0c;
m_unk0x08 = 0;
m_unk0x0c = NULL;
}
if (m_unk0x14 != NULL) {
delete[] m_unk0x14;
m_unk0x10 = 0;
m_unk0x14 = NULL;
}
return result;
}
// FUNCTION: LEGO1 0x1009f490
// FUNCTION: BETA10 0x10181a83
LegoResult LegoAnimScene::FUN_1009f490(LegoFloat p_time, Matrix4& p_matrix)
{
MxMatrix localb0;
MxMatrix local4c;
Vector3 local5c(localb0[0]);
Vector3 local68(localb0[1]);
Vector3 local54(localb0[2]);
Vector3 localb8(localb0[3]);
Mx3DPointFloat localcc;
localb0.SetIdentity();
LegoU32 local60;
if (m_unk0x08 != 0) {
local60 = GetUnknown0x18();
LegoAnimNodeData::GetTranslation(m_unk0x08, m_unk0x0c, p_time, localb0, local60);
SetUnknown0x18(local60);
localcc = localb8;
localb8.Clear();
}
if (m_unk0x00 != 0) {
local60 = GetUnknown0x1c();
LegoAnimNodeData::GetTranslation(m_unk0x00, m_unk0x04, p_time, localb0, local60);
SetUnknown0x1c(local60);
}
local54 = localcc;
((Vector3&) local54).Sub(localb8);
if (local54.Unitize() == 0) {
local5c.EqualsCross(&local68, &local54);
if (local5c.Unitize() == 0) {
local68.EqualsCross(&local54, &local5c);
localcc = p_matrix[3];
((Vector3&) localcc).Add(localb0[3]);
p_matrix[3][0] = p_matrix[3][1] = p_matrix[3][2] = localb0[3][0] = localb0[3][1] = localb0[3][2] = 0;
if (m_unk0x10 != 0) {
LegoU32 locald0 = -1;
LegoU32 locald8;
locald0 = GetUnknown0x20();
LegoU32 localdc =
LegoAnimNodeData::FindKeys(p_time, m_unk0x10, m_unk0x14, sizeof(*m_unk0x14), locald8, locald0);
SetUnknown0x20(locald0);
switch (localdc) {
case 1:
p_matrix.RotateZ(m_unk0x14[locald8].GetZ());
break;
case 2:
// Seems to be unused
LegoFloat z = LegoAnimNodeData::Interpolate(
p_time,
m_unk0x14[locald8],
m_unk0x14[locald8].GetZ(),
m_unk0x14[locald8 + 1],
m_unk0x14[locald8 + 1].GetZ()
);
p_matrix.RotateZ(m_unk0x14[locald8].GetZ());
break;
}
}
local4c = p_matrix;
p_matrix.Product(local4c.GetData(), localb0.GetData());
p_matrix[3][0] = localcc[0];
p_matrix[3][1] = localcc[1];
p_matrix[3][2] = localcc[2];
}
}
return SUCCESS;
}
// FUNCTION: LEGO1 0x1009f900
LegoAnimKey::LegoAnimKey()
{
m_flags = 0;
m_time = 0;
}
// FUNCTION: LEGO1 0x1009f910
LegoResult LegoAnimKey::Read(LegoStorage* p_storage)
{
LegoResult result;
LegoS32 und;
if ((result = p_storage->Read(&und, sizeof(und))) != SUCCESS) {
return result;
}
m_flags = (LegoU32) und >> 24;
m_time = und & 0xffffff;
return SUCCESS;
}
// FUNCTION: LEGO1 0x1009f990
LegoTranslationKey::LegoTranslationKey()
{
m_x = 0.0F;
m_y = 0.0F;
m_z = 0.0F;
}
// FUNCTION: LEGO1 0x1009f9b0
LegoResult LegoTranslationKey::Read(LegoStorage* p_storage)
{
LegoResult result;
if ((result = LegoAnimKey::Read(p_storage)) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_x, sizeof(m_x))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_y, sizeof(m_y))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_z, sizeof(m_z))) != SUCCESS) {
return result;
}
if (m_x > 1e-05F || m_x < -1e-05F || m_y > 1e-05F || m_y < -1e-05F || m_z > 1e-05F || m_z < -1e-05F) {
m_flags |= c_bit1;
}
return SUCCESS;
}
// FUNCTION: LEGO1 0x1009faa0
// FUNCTION: BETA10 0x1017e2b3
LegoRotationKey::LegoRotationKey()
{
m_angle = 1.0F;
m_x = 0.0F;
m_y = 0.0F;
m_z = 0.0F;
}
// FUNCTION: LEGO1 0x1009fac0
LegoResult LegoRotationKey::Read(LegoStorage* p_storage)
{
LegoResult result;
if ((result = LegoAnimKey::Read(p_storage)) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_angle, sizeof(m_angle))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_x, sizeof(m_x))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_y, sizeof(m_y))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_z, sizeof(m_z))) != SUCCESS) {
return result;
}
if (m_angle != 1.0F) {
m_flags |= c_bit1;
}
return SUCCESS;
}
// FUNCTION: LEGO1 0x1009fba0
LegoScaleKey::LegoScaleKey()
{
m_x = 1.0F;
m_y = 1.0F;
m_z = 1.0F;
}
// FUNCTION: LEGO1 0x1009fbc0
LegoResult LegoScaleKey::Read(LegoStorage* p_storage)
{
LegoResult result;
if ((result = LegoAnimKey::Read(p_storage)) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_x, sizeof(m_x))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_y, sizeof(m_y))) != SUCCESS) {
return result;
}
if ((result = p_storage->Read(&m_z, sizeof(m_z))) != SUCCESS) {
return result;
}
if (m_x > 1.00001 || m_x < 0.99999 || m_y > 1.00001 || m_y < 0.99999 || m_z > 1.00001 || m_z < 0.99999) {
m_flags |= c_bit1;
}
return SUCCESS;
}
// FUNCTION: LEGO1 0x1009fcf0
// FUNCTION: BETA10 0x1017e71a
LegoAnimNodeData::LegoAnimNodeData()
{
m_numTranslationKeys = 0;
m_numRotationKeys = 0;
m_numScaleKeys = 0;
m_numMorphKeys = 0;
m_name = NULL;
m_translationKeys = NULL;
m_unk0x20 = 0;
m_rotationKeys = NULL;
m_unk0x22 = 0;
m_scaleKeys = NULL;
m_morphKeys = NULL;
m_translationIndex = 0;
m_rotationIndex = 0;
m_scaleIndex = 0;
m_morphIndex = 0;
}
// FUNCTION: LEGO1 0x1009fda0
LegoAnimNodeData::~LegoAnimNodeData()
{
if (m_name) {
delete[] m_name;
}
if (m_translationKeys) {
delete[] m_translationKeys;
}
if (m_rotationKeys) {
delete[] m_rotationKeys;
}
if (m_scaleKeys) {
delete[] m_scaleKeys;
}
if (m_morphKeys) {
delete[] m_morphKeys;
}
}
// FUNCTION: LEGO1 0x1009fe60
LegoResult LegoAnimNodeData::Read(LegoStorage* p_storage)
{
LegoResult result;
LegoU32 length;
if ((result = p_storage->Read(&length, sizeof(length))) != SUCCESS) {
return result;
}
if (m_name) {
delete[] m_name;
m_name = NULL;
}
if (length) {
m_name = new LegoChar[length + 1];
if ((result = p_storage->Read(m_name, length)) != SUCCESS) {
return result;
}
m_name[length] = '\0';
}
LegoU32 i;
if ((result = p_storage->Read(&m_numTranslationKeys, sizeof(m_numTranslationKeys))) != SUCCESS) {
return result;
}
if (m_translationKeys) {
delete[] m_translationKeys;
m_translationKeys = NULL;
}
if (m_numTranslationKeys) {
m_translationKeys = new LegoTranslationKey[m_numTranslationKeys];
for (i = 0; i < m_numTranslationKeys; i++) {
if ((result = m_translationKeys[i].Read(p_storage)) != SUCCESS) {
return result;
}
}
}
if ((result = p_storage->Read(&m_numRotationKeys, sizeof(m_numRotationKeys))) != SUCCESS) {
return result;
}
if (m_rotationKeys) {
delete[] m_rotationKeys;
m_rotationKeys = NULL;
}
if (m_numRotationKeys) {
m_rotationKeys = new LegoRotationKey[m_numRotationKeys];
for (i = 0; i < m_numRotationKeys; i++) {
if ((result = m_rotationKeys[i].Read(p_storage)) != SUCCESS) {
return result;
}
}
}
if ((result = p_storage->Read(&m_numScaleKeys, sizeof(m_numScaleKeys))) != SUCCESS) {
return result;
}
if (m_scaleKeys) {
delete[] m_scaleKeys;
m_scaleKeys = NULL;
}
if (m_numScaleKeys) {
m_scaleKeys = new LegoScaleKey[m_numScaleKeys];
for (i = 0; i < m_numScaleKeys; i++) {
if ((result = m_scaleKeys[i].Read(p_storage)) != SUCCESS) {
return result;
}
}
}
if ((result = p_storage->Read(&m_numMorphKeys, sizeof(m_numMorphKeys))) != SUCCESS) {
return result;
}
if (m_morphKeys) {
delete[] m_morphKeys;
m_morphKeys = NULL;
}
if (m_numMorphKeys) {
m_morphKeys = new LegoMorphKey[m_numMorphKeys];
for (i = 0; i < m_numMorphKeys; i++) {
if ((result = m_morphKeys[i].Read(p_storage)) != SUCCESS) {
return result;
}
}
}
return SUCCESS;
}
// STUB: LEGO1 0x100a01e0
LegoResult LegoAnimNodeData::Write(LegoStorage* p_storage)
{
// TODO
return SUCCESS;
}
// STUB: LEGO1 0x100a0360
// STUB: BETA10 0x1017f1e5
void LegoAnimNodeData::FUN_100a0360(LegoChar* p_param)
{
// TODO
}
// FUNCTION: LEGO1 0x100a03c0
LegoResult LegoAnimNodeData::CreateLocalTransform(LegoFloat p_time, Matrix4& p_matrix)
{
LegoU32 index;
if (m_scaleKeys != NULL) {
index = GetScaleIndex();
GetScale(m_numScaleKeys, m_scaleKeys, p_time, p_matrix, index);
SetScaleIndex(index);
if (m_rotationKeys != NULL) {
MxMatrix a, b;
a.SetIdentity();
index = GetRotationIndex();
GetRotation(m_numRotationKeys, m_rotationKeys, p_time, a, index);
SetRotationIndex(index);
b = p_matrix;
p_matrix.Product(b, a);
}
}
else if (m_rotationKeys != NULL) {
index = GetRotationIndex();
GetRotation(m_numRotationKeys, m_rotationKeys, p_time, p_matrix, index);
SetRotationIndex(index);
}
if (m_translationKeys != NULL) {
index = GetTranslationIndex();
GetTranslation(m_numTranslationKeys, m_translationKeys, p_time, p_matrix, index);
SetTranslationIndex(index);
}
return SUCCESS;
}
// FUNCTION: LEGO1 0x100a0600
inline void LegoAnimNodeData::GetTranslation(
LegoU16 p_numTranslationKeys,
LegoTranslationKey* p_translationKeys,
LegoFloat p_time,
Matrix4& p_matrix,
LegoU32& p_old_index
)
{
LegoU32 i, n;
LegoFloat x, y, z;
n = FindKeys(
p_time,
p_numTranslationKeys & USHRT_MAX,
p_translationKeys,
sizeof(*p_translationKeys),
i,
p_old_index
);
switch (n) {
case 0:
return;
case 1:
if (!p_translationKeys[i].TestBit1()) {
return;
}
x = p_translationKeys[i].GetX();
y = p_translationKeys[i].GetY();
z = p_translationKeys[i].GetZ();
break;
case 2:
if (!p_translationKeys[i].TestBit1() && !p_translationKeys[i + 1].TestBit1()) {
return;
}
x = Interpolate(
p_time,
p_translationKeys[i],
p_translationKeys[i].GetX(),
p_translationKeys[i + 1],
p_translationKeys[i + 1].GetX()
);
y = Interpolate(
p_time,
p_translationKeys[i],
p_translationKeys[i].GetY(),
p_translationKeys[i + 1],
p_translationKeys[i + 1].GetY()
);
z = Interpolate(
p_time,
p_translationKeys[i],
p_translationKeys[i].GetZ(),
p_translationKeys[i + 1],
p_translationKeys[i + 1].GetZ()
);
break;
}
p_matrix.TranslateBy(x, y, z);
}
// FUNCTION: LEGO1 0x100a06f0
/*inline*/ void LegoAnimNodeData::GetRotation(
LegoU16 p_numRotationKeys,
LegoRotationKey* p_rotationKeys,
LegoFloat p_time,
Matrix4& p_matrix,
LegoU32& p_old_index
)
{
LegoU32 i, n;
n = FindKeys(p_time, p_numRotationKeys & USHRT_MAX, p_rotationKeys, sizeof(*p_rotationKeys), i, p_old_index);
switch (n) {
case 0:
return;
case 1:
if (p_rotationKeys[i].TestBit1()) {
p_matrix.FromQuaternion(Mx4DPointFloat(
p_rotationKeys[i].GetX(),
p_rotationKeys[i].GetY(),
p_rotationKeys[i].GetZ(),
p_rotationKeys[i].GetAngle()
));
}
break;
case 2:
Mx4DPointFloat a;
UnknownMx4DPointFloat b;
if (p_rotationKeys[i].TestBit1() || p_rotationKeys[i + 1].TestBit1()) {
a[0] = p_rotationKeys[i].GetX();
a[1] = p_rotationKeys[i].GetY();
a[2] = p_rotationKeys[i].GetZ();
a[3] = p_rotationKeys[i].GetAngle();
if (p_rotationKeys[i + 1].TestBit3()) {
p_matrix.FromQuaternion(a);
return;
}
Mx4DPointFloat c;
if (p_rotationKeys[i + 1].TestBit2()) {
c[0] = -p_rotationKeys[i + 1].GetX();
c[1] = -p_rotationKeys[i + 1].GetY();
c[2] = -p_rotationKeys[i + 1].GetZ();
c[3] = -p_rotationKeys[i + 1].GetAngle();
}
else {
c[0] = p_rotationKeys[i + 1].GetX();
c[1] = p_rotationKeys[i + 1].GetY();
c[2] = p_rotationKeys[i + 1].GetZ();
c[3] = p_rotationKeys[i + 1].GetAngle();
}
b.Unknown4(a);
b.Unknown5(c);
b.Unknown6(
p_matrix,
(p_time - p_rotationKeys[i].GetTime()) / (p_rotationKeys[i + 1].GetTime() - p_rotationKeys[i].GetTime())
);
}
}
}
inline void LegoAnimNodeData::GetScale(
LegoU16 p_numScaleKeys,
LegoScaleKey* p_scaleKeys,
LegoFloat p_time,
Matrix4& p_matrix,
LegoU32& p_old_index
)
{
LegoU32 i, n;
LegoFloat x, y, z;
n = FindKeys(p_time, p_numScaleKeys & USHRT_MAX, p_scaleKeys, sizeof(*p_scaleKeys), i, p_old_index);
switch (n) {
case 0:
return;
case 1:
x = p_scaleKeys[i].GetX();
y = p_scaleKeys[i].GetY();
z = p_scaleKeys[i].GetZ();
break;
case 2:
x = Interpolate(p_time, p_scaleKeys[i], p_scaleKeys[i].GetX(), p_scaleKeys[i + 1], p_scaleKeys[i + 1].GetX());
y = Interpolate(p_time, p_scaleKeys[i], p_scaleKeys[i].GetY(), p_scaleKeys[i + 1], p_scaleKeys[i + 1].GetY());
z = Interpolate(p_time, p_scaleKeys[i], p_scaleKeys[i].GetZ(), p_scaleKeys[i + 1], p_scaleKeys[i + 1].GetZ());
break;
}
p_matrix.Scale(x, y, z);
}
// FUNCTION: LEGO1 0x100a0990
LegoBool LegoAnimNodeData::FUN_100a0990(LegoFloat p_time)
{
LegoU32 i, n;
LegoU32 index = GetMorphIndex();
LegoBool result;
n = FindKeys(p_time, m_numMorphKeys, m_morphKeys, sizeof(*m_morphKeys), i, index);
SetMorphIndex(index);
switch (n) {
case 0:
result = TRUE;
break;
case 1:
case 2:
result = m_morphKeys[i].GetUnknown0x08();
break;
}
return result;
}
// FUNCTION: LEGO1 0x100a0a00
LegoU32 LegoAnimNodeData::FindKeys(
LegoFloat p_time,
LegoU32 p_numKeys,
LegoAnimKey* p_keys,
LegoU32 p_size,
LegoU32& p_new_index,
LegoU32& p_old_index
)
{
LegoU32 numKeys;
if (p_numKeys == 0) {
numKeys = 0;
}
else if (p_time < GetKey(0, p_keys, p_size).GetTime()) {
numKeys = 0;
}
else if (p_time > GetKey(p_numKeys - 1, p_keys, p_size).GetTime()) {
p_new_index = p_numKeys - 1;
numKeys = 1;
}
else {
if (GetKey(p_old_index, p_keys, p_size).GetTime() <= p_time) {
for (p_new_index = p_old_index;
p_new_index < p_numKeys - 1 && p_time >= GetKey(p_new_index + 1, p_keys, p_size).GetTime();
p_new_index++) {
}
}
else {
for (p_new_index = 0;
p_new_index < p_numKeys - 1 && p_time >= GetKey(p_new_index + 1, p_keys, p_size).GetTime();
p_new_index++) {
}
}
p_old_index = p_new_index;
if (p_time == GetKey(p_new_index, p_keys, p_size).GetTime()) {
numKeys = 1;
}
else if (p_new_index < p_numKeys - 1) {
numKeys = 2;
}
else {
numKeys = 0;
}
}
return numKeys;
}
// FUNCTION: LEGO1 0x100a0b00
inline LegoFloat LegoAnimNodeData::Interpolate(
LegoFloat p_time,
LegoAnimKey& p_key1,
LegoFloat p_value1,
LegoAnimKey& p_key2,
LegoFloat p_value2
)
{
return p_value1 + (p_value2 - p_value1) * (p_time - p_key1.GetTime()) / (p_key2.GetTime() - p_key1.GetTime());
}
inline LegoAnimKey& LegoAnimNodeData::GetKey(LegoU32 p_i, LegoAnimKey* p_keys, LegoU32 p_size)
{
return *((LegoAnimKey*) (((LegoU8*) p_keys) + (p_i * p_size)));
}
// FUNCTION: LEGO1 0x100a0b30
LegoAnim::LegoAnim()
{
m_duration = 0;
m_actors = NULL;
m_numActors = 0;
m_camAnim = NULL;
}
// FUNCTION: LEGO1 0x100a0bc0
LegoAnim::~LegoAnim()
{
if (m_actors != NULL) {
for (LegoU32 i = 0; i < m_numActors; i++) {
delete[] m_actors[i].m_name;
}
delete[] m_actors;
}
if (m_camAnim != NULL) {
delete m_camAnim;
}
}
// FUNCTION: LEGO1 0x100a0c70
LegoResult LegoAnim::Read(LegoStorage* p_storage, LegoS32 p_parseScene)
{
LegoResult result = FAILURE;
LegoU32 length, i;
if (p_storage->Read(&length, sizeof(length)) != SUCCESS) {
goto done;
}
m_actors = new LegoAnimActorEntry[length];
m_numActors = 0;
for (i = 0; i < length; i++) {
LegoU32 length;
if (p_storage->Read(&length, sizeof(length)) != SUCCESS) {
goto done;
}
if (length) {
m_actors[i].m_name = new LegoChar[length + 1];
if (p_storage->Read(m_actors[i].m_name, length) != SUCCESS) {
goto done;
}
m_actors[i].m_name[length] = '\0';
if (p_storage->Read(&m_actors[i].m_unk0x04, sizeof(m_actors[i].m_unk0x04)) != SUCCESS) {
goto done;
}
}
m_numActors++;
}
if ((result = p_storage->Read(&m_duration, sizeof(m_duration))) != SUCCESS) {
goto done;
}
if (p_parseScene) {
m_camAnim = new LegoAnimScene();
result = m_camAnim->Read(p_storage);
if (result != SUCCESS) {
goto done;
}
}
result = LegoTree::Read(p_storage);
done:
if (result != SUCCESS && m_actors != NULL) {
for (i = 0; i < m_numActors; i++) {
delete[] m_actors[i].m_name;
}
m_numActors = 0;
delete[] m_actors;
m_actors = NULL;
}
return result;
}
// STUB: LEGO1 0x100a0e30
LegoResult LegoAnim::Write(LegoStorage* p_storage)
{
// TODO
return SUCCESS;
}
// FUNCTION: LEGO1 0x100a0f20
const LegoChar* LegoAnim::GetActorName(LegoU32 p_index)
{
if (p_index < m_numActors) {
return m_actors[p_index].m_name;
}
return NULL;
}
// FUNCTION: LEGO1 0x100a0f40
undefined4 LegoAnim::GetActorUnknown0x04(LegoU32 p_index)
{
if (p_index < m_numActors) {
return m_actors[p_index].m_unk0x04;
}
return NULL;
}
// FUNCTION: LEGO1 0x100a0f60
LegoMorphKey::LegoMorphKey()
{
m_unk0x08 = 0;
}
// FUNCTION: LEGO1 0x100a0f70
LegoResult LegoMorphKey::Read(LegoStorage* p_storage)
{
LegoResult result;
if ((result = LegoAnimKey::Read(p_storage)) != SUCCESS) {
return result;
}
result = p_storage->Read(&m_unk0x08, sizeof(m_unk0x08));
return result == SUCCESS ? SUCCESS : result;
}