metaforce/DataSpec/DNAMP1/ANIM.hpp

247 lines
7.8 KiB
C++
Raw Normal View History

2018-10-07 03:42:33 +00:00
#pragma once
2015-08-11 23:32:02 +00:00
#include "DNAMP1.hpp"
2018-06-29 20:21:36 +00:00
#include "DataSpec/DNACommon/ANIM.hpp"
2016-04-07 03:40:25 +00:00
#include "DataSpec/DNACommon/RigInverter.hpp"
2015-08-13 07:29:00 +00:00
#include "CINF.hpp"
2018-10-28 01:22:55 +00:00
#include "EVNT.hpp"
#include "DataSpec/DNACommon/ANCS.hpp"
2015-08-11 23:32:02 +00:00
2017-12-29 08:08:12 +00:00
namespace DataSpec::DNAMP1
2015-08-11 23:32:02 +00:00
{
struct ANIM : BigDNA
{
2018-02-22 07:24:51 +00:00
AT_DECL_EXPLICIT_DNA
2015-08-11 23:32:02 +00:00
static UniqueID32 GetEVNTId(athena::io::IStreamReader& r);
2018-02-22 07:24:51 +00:00
struct IANIM : BigDNAV
2015-08-11 23:32:02 +00:00
{
Delete expl;
atUint32 m_version;
IANIM(atUint32 version) : m_version(version) {}
std::vector<std::pair<atUint32, bool>> bones;
2015-08-13 21:29:07 +00:00
std::vector<atUint32> frames;
2015-08-11 23:32:02 +00:00
std::vector<DNAANIM::Channel> channels;
std::vector<std::vector<DNAANIM::Value>> chanKeys;
float mainInterval = 0.0;
2018-04-02 04:27:24 +00:00
UniqueID32Zero evnt;
2016-09-06 05:52:51 +00:00
bool looping = false;
2015-08-13 07:29:00 +00:00
2017-12-29 08:08:12 +00:00
void sendANIMToBlender(hecl::blender::PyOutStream&, const DNAANIM::RigInverter<CINF>& rig) const;
2015-08-11 23:32:02 +00:00
};
struct ANIM0 : IANIM
{
2018-02-22 07:24:51 +00:00
AT_DECL_EXPLICIT_DNA
AT_DECL_DNAV
2015-08-11 23:32:02 +00:00
ANIM0() : IANIM(0) {}
struct Header : BigDNA
{
2018-02-22 07:24:51 +00:00
AT_DECL_DNA
2015-08-11 23:32:02 +00:00
Value<float> duration;
Value<atUint32> unk0;
Value<float> interval;
Value<atUint32> unk1;
Value<atUint32> keyCount;
Value<atUint32> unk2;
Value<atUint32> boneSlotCount;
2015-08-11 23:32:02 +00:00
};
};
struct ANIM2 : IANIM
{
2018-02-22 07:24:51 +00:00
AT_DECL_EXPLICIT_DNA
AT_DECL_DNAV
ANIM2(bool pc) : IANIM(pc ? 3 : 2) {}
2015-08-11 23:32:02 +00:00
struct Header : BigDNA
{
2018-02-22 07:24:51 +00:00
AT_DECL_DNA
2015-08-11 23:32:02 +00:00
Value<atUint32> scratchSize;
2018-04-02 04:27:24 +00:00
UniqueID32Zero evnt;
Value<atUint32> unk0 = 1;
2015-08-11 23:32:02 +00:00
Value<float> duration;
Value<float> interval;
Value<atUint32> rootBoneId = 3;
2016-08-27 21:16:44 +00:00
Value<atUint32> looping = 0;
2015-08-11 23:32:02 +00:00
Value<atUint32> rotDiv;
Value<float> translationMult;
Value<atUint32> boneChannelCount;
Value<atUint32> unk3;
Value<atUint32> keyBitmapBitCount;
};
struct ChannelDesc : BigDNA
{
Delete expl;
Value<atUint32> id = 0;
Value<atUint16> keyCount1 = 0;
2016-08-29 02:28:55 +00:00
Value<atInt16> initRX = 0;
2015-08-11 23:32:02 +00:00
Value<atUint8> qRX = 0;
2016-08-29 02:28:55 +00:00
Value<atInt16> initRY = 0;
2015-08-11 23:32:02 +00:00
Value<atUint8> qRY = 0;
2016-08-29 02:28:55 +00:00
Value<atInt16> initRZ = 0;
2015-08-11 23:32:02 +00:00
Value<atUint8> qRZ = 0;
Value<atUint16> keyCount2 = 0;
2016-08-29 02:28:55 +00:00
Value<atInt16> initTX = 0;
2015-08-11 23:32:02 +00:00
Value<atUint8> qTX = 0;
2016-08-29 02:28:55 +00:00
Value<atInt16> initTY = 0;
2015-08-11 23:32:02 +00:00
Value<atUint8> qTY = 0;
2016-08-29 02:28:55 +00:00
Value<atInt16> initTZ = 0;
2015-08-11 23:32:02 +00:00
Value<atUint8> qTZ = 0;
2016-03-04 23:04:53 +00:00
void read(athena::io::IStreamReader& reader)
2015-08-11 23:32:02 +00:00
{
id = reader.readUint32Big();
keyCount1 = reader.readUint16Big();
2016-08-29 02:28:55 +00:00
initRX = reader.readInt16Big();
2015-08-11 23:32:02 +00:00
qRX = reader.readUByte();
2016-08-29 02:28:55 +00:00
initRY = reader.readInt16Big();
2015-08-11 23:32:02 +00:00
qRY = reader.readUByte();
2016-08-29 02:28:55 +00:00
initRZ = reader.readInt16Big();
2015-08-11 23:32:02 +00:00
qRZ = reader.readUByte();
keyCount2 = reader.readUint16Big();
2015-08-11 23:32:02 +00:00
if (keyCount2)
{
2016-08-29 02:28:55 +00:00
initTX = reader.readInt16Big();
2015-08-11 23:32:02 +00:00
qTX = reader.readUByte();
2016-08-29 02:28:55 +00:00
initTY = reader.readInt16Big();
2015-08-11 23:32:02 +00:00
qTY = reader.readUByte();
2016-08-29 02:28:55 +00:00
initTZ = reader.readInt16Big();
2015-08-11 23:32:02 +00:00
qTZ = reader.readUByte();
}
}
2016-03-04 23:04:53 +00:00
void write(athena::io::IStreamWriter& writer) const
2015-08-11 23:32:02 +00:00
{
writer.writeUint32Big(id);
writer.writeUint16Big(keyCount1);
2016-08-29 02:28:55 +00:00
writer.writeInt16Big(initRX);
2015-08-11 23:32:02 +00:00
writer.writeUByte(qRX);
2016-08-29 02:28:55 +00:00
writer.writeInt16Big(initRY);
2015-08-11 23:32:02 +00:00
writer.writeUByte(qRY);
2016-08-29 02:28:55 +00:00
writer.writeInt16Big(initRZ);
2015-08-11 23:32:02 +00:00
writer.writeUByte(qRZ);
writer.writeUint16Big(keyCount2);
2015-08-11 23:32:02 +00:00
if (keyCount2)
{
2016-08-29 02:28:55 +00:00
writer.writeInt16Big(initTX);
2015-08-11 23:32:02 +00:00
writer.writeUByte(qTX);
2016-08-29 02:28:55 +00:00
writer.writeInt16Big(initTY);
2015-08-11 23:32:02 +00:00
writer.writeUByte(qTY);
2016-08-29 02:28:55 +00:00
writer.writeInt16Big(initTZ);
2015-08-11 23:32:02 +00:00
writer.writeUByte(qTZ);
}
}
2018-03-04 06:15:40 +00:00
void binarySize(size_t& __isz) const
{
__isz += 17;
if (keyCount2)
__isz += 9;
}
2015-08-11 23:32:02 +00:00
};
2016-08-27 01:02:12 +00:00
struct ChannelDescPC : BigDNA
{
Delete expl;
Value<atUint32> id = 0;
Value<atUint32> keyCount1 = 0;
Value<atUint32> QinitRX = 0;
Value<atUint32> QinitRY = 0;
Value<atUint32> QinitRZ = 0;
Value<atUint32> keyCount2 = 0;
Value<atUint32> QinitTX = 0;
Value<atUint32> QinitTY = 0;
Value<atUint32> QinitTZ = 0;
void read(athena::io::IStreamReader& reader)
{
id = reader.readUint32Big();
keyCount1 = reader.readUint32Big();
QinitRX = reader.readUint32Big();
QinitRY = reader.readUint32Big();
QinitRZ = reader.readUint32Big();
keyCount2 = reader.readUint32Big();
if (keyCount2)
{
QinitTX = reader.readUint32Big();
QinitTY = reader.readUint32Big();
QinitTZ = reader.readUint32Big();
}
}
void write(athena::io::IStreamWriter& writer) const
{
writer.writeUint32Big(id);
writer.writeUint32Big(keyCount1);
writer.writeUint32Big(QinitRX);
writer.writeUint32Big(QinitRY);
writer.writeUint32Big(QinitRZ);
writer.writeUint32Big(keyCount2);
if (keyCount2)
{
writer.writeUint32Big(QinitTX);
writer.writeUint32Big(QinitTY);
writer.writeUint32Big(QinitTZ);
}
}
2018-03-04 06:15:40 +00:00
void binarySize(size_t& __isz) const
2016-08-27 01:02:12 +00:00
{
__isz += 24;
if (keyCount2)
__isz += 12;
}
};
2015-08-11 23:32:02 +00:00
};
std::unique_ptr<IANIM> m_anim;
2017-12-29 08:08:12 +00:00
void sendANIMToBlender(hecl::blender::PyOutStream& os, const DNAANIM::RigInverter<CINF>& rig, bool) const
2015-08-13 07:29:00 +00:00
{
2016-04-07 03:40:25 +00:00
m_anim->sendANIMToBlender(os, rig);
2015-08-13 07:29:00 +00:00
}
2016-09-06 05:52:51 +00:00
bool isLooping() const
{
if (!m_anim)
return false;
return m_anim->looping;
}
2018-10-28 01:22:55 +00:00
void extractEVNT(const DNAANCS::AnimationResInfo<UniqueID32>& animInfo,
const hecl::ProjectPath& outPath, PAKRouter<PAKBridge>& pakRouter, bool force) const
{
if (m_anim->evnt)
{
hecl::SystemStringConv sysStr(animInfo.name);
hecl::ProjectPath evntYamlPath = outPath.getWithExtension((hecl::SystemString(_SYS_STR(".")) +
sysStr.c_str() +
_SYS_STR(".evnt.yaml")).c_str(), true);
hecl::ProjectPath::Type evntYamlType = evntYamlPath.getPathType();
if (force || evntYamlType == hecl::ProjectPath::Type::None)
{
EVNT evnt;
if (pakRouter.lookupAndReadDNA(m_anim->evnt, evnt, true))
{
athena::io::FileWriter writer(evntYamlPath.getAbsolutePath());
athena::io::ToYAMLStream(evnt, writer);
}
}
}
}
2017-12-29 08:08:12 +00:00
using BlenderAction = hecl::blender::Action;
2016-04-09 23:19:17 +00:00
ANIM() = default;
2016-08-22 00:11:18 +00:00
ANIM(const BlenderAction& act,
const std::unordered_map<std::string, atInt32>& idMap,
const DNAANIM::RigInverter<CINF>& rig,
bool pc);
2015-08-11 23:32:02 +00:00
};
}