#ifndef _DNACOMMON_ANIMBITSTREAM_HPP_ #define _DNACOMMON_ANIMBITSTREAM_HPP_ #include "DNACommon.hpp" namespace Retro { namespace DNAANIM { union Value { atVec3f v3; atVec4f v4; Value(atVec3f v) : v3(v) {} Value(atVec4f v) : v4(v) {} Value(float x, float y, float z) { v3.vec[0] = x; v3.vec[1] = y; v3.vec[2] = z; v4.vec[3] = 0.0; } Value(float w, float x, float y, float z) { v4.vec[0] = w; v4.vec[1] = x; v4.vec[2] = y; v4.vec[3] = z; } }; struct QuantizedValue { atInt16 v[4]; atInt16& operator[] (size_t idx) {return v[idx];} const atInt16& operator[] (size_t idx) const {return v[idx];} }; struct QuantizedRot { QuantizedValue v; bool w; }; struct Channel { enum Type { ROTATION, TRANSLATION, SCALE, KF_HEAD, ROTATION_MP3 } type; QuantizedValue i = {}; atUint8 q[4] = {}; }; size_t ComputeBitstreamSize(size_t keyFrameCount, const std::vector& channels); class BitstreamReader { size_t m_bitCur; atInt16 dequantize(const atUint8* data, atUint8 q); bool dequantizeBit(const atUint8* data); public: std::vector> read(const atUint8* data, size_t keyFrameCount, const std::vector& channels, atUint32 rotDiv, float transMult); }; class BitstreamWriter { size_t m_bitCur; void quantize(atUint8* data, atUint8 q, atInt16 val); void quantizeBit(atUint8* data, bool val); public: std::unique_ptr write(const std::vector>& chanKeys, size_t keyFrameCount, std::vector& channels, atUint32& rotDivOut, float& transMultOut, size_t& sizeOut); }; } } #endif // _DNACOMMON_ANIMBITSTREAM_HPP_