2015-08-11 23:32:02 +00:00
|
|
|
#ifndef _DNAMP1_ANIM_HPP_
|
|
|
|
#define _DNAMP1_ANIM_HPP_
|
|
|
|
|
|
|
|
#include "DNAMP1.hpp"
|
|
|
|
#include "../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"
|
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
|
|
|
|
{
|
|
|
|
Delete expl;
|
|
|
|
|
|
|
|
struct IANIM : BigDNA
|
|
|
|
{
|
|
|
|
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;
|
|
|
|
UniqueID32 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
|
|
|
|
{
|
|
|
|
DECL_EXPLICIT_DNA
|
|
|
|
ANIM0() : IANIM(0) {}
|
|
|
|
|
|
|
|
struct Header : BigDNA
|
|
|
|
{
|
|
|
|
DECL_DNA
|
|
|
|
Value<float> duration;
|
|
|
|
Value<atUint32> unk0;
|
|
|
|
Value<float> interval;
|
|
|
|
Value<atUint32> unk1;
|
|
|
|
Value<atUint32> keyCount;
|
2015-08-14 03:00:51 +00:00
|
|
|
Value<atUint32> unk2;
|
|
|
|
Value<atUint32> boneSlotCount;
|
2015-08-11 23:32:02 +00:00
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
struct ANIM2 : IANIM
|
|
|
|
{
|
|
|
|
DECL_EXPLICIT_DNA
|
2016-08-26 22:23:59 +00:00
|
|
|
ANIM2(bool pc) : IANIM(pc ? 3 : 2) {}
|
2015-08-11 23:32:02 +00:00
|
|
|
|
|
|
|
struct Header : BigDNA
|
|
|
|
{
|
|
|
|
DECL_DNA
|
|
|
|
Value<atUint32> scratchSize;
|
|
|
|
UniqueID32 evnt;
|
2016-08-26 22:23:59 +00:00
|
|
|
Value<atUint32> unk0 = 1;
|
2015-08-11 23:32:02 +00:00
|
|
|
Value<float> duration;
|
|
|
|
Value<float> interval;
|
2016-08-26 22:23:59 +00:00
|
|
|
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
|
|
|
{
|
2015-08-14 03:00:51 +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();
|
2015-08-14 03:00:51 +00:00
|
|
|
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
|
|
|
{
|
2015-08-14 03:00:51 +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);
|
2015-08-14 03:00:51 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
}
|
2015-10-18 04:08:45 +00:00
|
|
|
size_t binarySize(size_t __isz) const
|
|
|
|
{
|
|
|
|
__isz += 17;
|
|
|
|
if (keyCount2)
|
|
|
|
__isz += 9;
|
|
|
|
return __isz;
|
|
|
|
}
|
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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
size_t binarySize(size_t __isz) const
|
|
|
|
{
|
|
|
|
__isz += 24;
|
|
|
|
if (keyCount2)
|
|
|
|
__isz += 12;
|
|
|
|
return __isz;
|
|
|
|
}
|
|
|
|
};
|
2015-08-11 23:32:02 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
std::unique_ptr<IANIM> m_anim;
|
2016-03-04 23:04:53 +00:00
|
|
|
void read(athena::io::IStreamReader& reader)
|
2015-08-11 23:32:02 +00:00
|
|
|
{
|
2015-08-14 03:00:51 +00:00
|
|
|
atUint32 version = reader.readUint32Big();
|
2015-08-11 23:32:02 +00:00
|
|
|
switch (version)
|
|
|
|
{
|
|
|
|
case 0:
|
|
|
|
m_anim.reset(new struct ANIM0);
|
|
|
|
m_anim->read(reader);
|
|
|
|
break;
|
|
|
|
case 2:
|
2016-08-26 22:23:59 +00:00
|
|
|
m_anim.reset(new struct ANIM2(false));
|
|
|
|
m_anim->read(reader);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
m_anim.reset(new struct ANIM2(true));
|
2015-08-11 23:32:02 +00:00
|
|
|
m_anim->read(reader);
|
|
|
|
break;
|
|
|
|
default:
|
2016-03-04 23:04:53 +00:00
|
|
|
Log.report(logvisor::Error, "unrecognized ANIM version");
|
2015-08-11 23:32:02 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-03-04 23:04:53 +00:00
|
|
|
void write(athena::io::IStreamWriter& writer) const
|
2015-08-11 23:32:02 +00:00
|
|
|
{
|
2015-08-14 03:00:51 +00:00
|
|
|
writer.writeUint32Big(m_anim->m_version);
|
2015-08-11 23:32:02 +00:00
|
|
|
m_anim->write(writer);
|
|
|
|
}
|
2015-08-13 07:29:00 +00:00
|
|
|
|
2015-10-18 04:08:45 +00:00
|
|
|
size_t binarySize(size_t __isz) const
|
|
|
|
{
|
|
|
|
return m_anim->binarySize(__isz + 4);
|
|
|
|
}
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
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,
|
2016-08-26 22:23:59 +00:00
|
|
|
const DNAANIM::RigInverter<CINF>& rig,
|
|
|
|
bool pc);
|
2015-08-11 23:32:02 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // _DNAMP1_ANIM_HPP_
|