|
|
@ -113,24 +113,24 @@ public:
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
/// Get a given record
|
|
|
|
/// Get a given record
|
|
|
|
Record *getRecord(size_t index)
|
|
|
|
Record *getRecord(size_t index) const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Record *res = records.at(index);
|
|
|
|
Record *res = records.at(index);
|
|
|
|
assert(res != NULL);
|
|
|
|
assert(res != NULL);
|
|
|
|
return res;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/// Number of records
|
|
|
|
/// Number of records
|
|
|
|
size_t numRecords() { return records.size(); }
|
|
|
|
size_t numRecords() const { return records.size(); }
|
|
|
|
|
|
|
|
|
|
|
|
/// Get a given root
|
|
|
|
/// Get a given root
|
|
|
|
Record *getRoot(size_t index=0)
|
|
|
|
Record *getRoot(size_t index=0) const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Record *res = roots.at(index);
|
|
|
|
Record *res = roots.at(index);
|
|
|
|
assert(res != NULL);
|
|
|
|
assert(res != NULL);
|
|
|
|
return res;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
/// Number of roots
|
|
|
|
/// Number of roots
|
|
|
|
size_t numRoots() { return roots.size(); }
|
|
|
|
size_t numRoots() const { return roots.size(); }
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
@ -163,45 +163,33 @@ struct KeyListT {
|
|
|
|
|
|
|
|
|
|
|
|
void read(NIFStream *nif, bool force=false)
|
|
|
|
void read(NIFStream *nif, bool force=false)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
assert(nif);
|
|
|
|
size_t count = nif->getInt();
|
|
|
|
size_t count = nif->getInt();
|
|
|
|
if(count == 0 && !force)
|
|
|
|
if(count == 0 && !force)
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
|
|
mInterpolationType = nif->getInt();
|
|
|
|
mInterpolationType = nif->getInt();
|
|
|
|
mKeys.resize(count);
|
|
|
|
mKeys.resize(count);
|
|
|
|
|
|
|
|
NIFStream &nifReference = *nif;
|
|
|
|
if(mInterpolationType == sLinearInterpolation)
|
|
|
|
if(mInterpolationType == sLinearInterpolation)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
for(size_t i = 0;i < count;i++)
|
|
|
|
for(size_t i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
KeyT<T> &key = mKeys[i];
|
|
|
|
readTimeAndValue(nifReference, mKeys[i]);
|
|
|
|
key.mTime = nif->getFloat();
|
|
|
|
|
|
|
|
key.mValue = (nif->*getValue)();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(mInterpolationType == sQuadraticInterpolation)
|
|
|
|
else if(mInterpolationType == sQuadraticInterpolation)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
for(size_t i = 0;i < count;i++)
|
|
|
|
for(size_t i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
KeyT<T> &key = mKeys[i];
|
|
|
|
readQuadratic(nifReference, mKeys[i]);
|
|
|
|
key.mTime = nif->getFloat();
|
|
|
|
|
|
|
|
key.mValue = (nif->*getValue)();
|
|
|
|
|
|
|
|
if( typeid(Ogre::Quaternion) != typeid(T) )
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
key.mForwardValue = (nif->*getValue)();
|
|
|
|
|
|
|
|
key.mBackwardValue = (nif->*getValue)();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if(mInterpolationType == sTBCInterpolation)
|
|
|
|
else if(mInterpolationType == sTBCInterpolation)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
for(size_t i = 0;i < count;i++)
|
|
|
|
for(size_t i = 0;i < count;i++)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
KeyT<T> &key = mKeys[i];
|
|
|
|
readTBC(nifReference, mKeys[i]);
|
|
|
|
key.mTime = nif->getFloat();
|
|
|
|
|
|
|
|
key.mValue = (nif->*getValue)();
|
|
|
|
|
|
|
|
key.mTension = nif->getFloat();
|
|
|
|
|
|
|
|
key.mBias = nif->getFloat();
|
|
|
|
|
|
|
|
key.mContinuity = nif->getFloat();
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//\FIXME This now reads the correct amount of data in the file, but doesn't actually do anything with it.
|
|
|
|
//\FIXME This now reads the correct amount of data in the file, but doesn't actually do anything with it.
|
|
|
@ -212,38 +200,71 @@ struct KeyListT {
|
|
|
|
nif->file->fail("count should always be '1' for XYZ_ROTATION_KEY. Retrieved Value: "+Ogre::StringConverter::toString(count));
|
|
|
|
nif->file->fail("count should always be '1' for XYZ_ROTATION_KEY. Retrieved Value: "+Ogre::StringConverter::toString(count));
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
readXYZ(nifReference);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (mInterpolationType == 0)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (count != 0)
|
|
|
|
|
|
|
|
nif->file->fail("Interpolation type 0 doesn't work with keys");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
nif->file->fail("Unhandled interpolation type: "+Ogre::StringConverter::toString(mInterpolationType));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
|
|
|
static void readTimeAndValue(NIFStream &nif, KeyT<T> &key)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
key.mTime = nif.getFloat();
|
|
|
|
|
|
|
|
key.mValue = (nif.*getValue)();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void readQuadratic(NIFStream &nif, KeyT<Ogre::Quaternion> &key)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
readTimeAndValue(nif, key);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
template <typename U>
|
|
|
|
|
|
|
|
static void readQuadratic(NIFStream &nif, KeyT<U> &key)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
readTimeAndValue(nif, key);
|
|
|
|
|
|
|
|
key.mForwardValue = (nif.*getValue)();
|
|
|
|
|
|
|
|
key.mBackwardValue = (nif.*getValue)();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void readTBC(NIFStream &nif, KeyT<T> &key)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
readTimeAndValue(nif, key);
|
|
|
|
|
|
|
|
key.mTension = nif.getFloat();
|
|
|
|
|
|
|
|
key.mBias = nif.getFloat();
|
|
|
|
|
|
|
|
key.mContinuity = nif.getFloat();
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
static void readXYZ(NIFStream &nif)
|
|
|
|
|
|
|
|
{
|
|
|
|
//KeyGroup (see http://niftools.sourceforge.net/doc/nif/NiKeyframeData.html)
|
|
|
|
//KeyGroup (see http://niftools.sourceforge.net/doc/nif/NiKeyframeData.html)
|
|
|
|
//Chomp unknown and possibly unused float
|
|
|
|
//Chomp unknown and possibly unused float
|
|
|
|
nif->getFloat();
|
|
|
|
nif.getFloat();
|
|
|
|
for(size_t i=0;i<3;++i)
|
|
|
|
for(size_t i=0;i<3;++i)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
unsigned int numKeys = nif->getInt();
|
|
|
|
const unsigned int numKeys = nif.getInt();
|
|
|
|
if(numKeys != 0)
|
|
|
|
if(numKeys != 0)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
int interpolationTypeAgain = nif->getInt();
|
|
|
|
const int interpolationTypeAgain = nif.getInt();
|
|
|
|
if( interpolationTypeAgain != sLinearInterpolation)
|
|
|
|
if( interpolationTypeAgain != sLinearInterpolation)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
nif->file->fail("XYZ_ROTATION_KEY's KeyGroup keyType must be '1' (Linear Interpolation). Retrieved Value: "+Ogre::StringConverter::toString(interpolationTypeAgain));
|
|
|
|
nif.file->fail("XYZ_ROTATION_KEY's KeyGroup keyType must be '1' (Linear Interpolation). Retrieved Value: "+Ogre::StringConverter::toString(interpolationTypeAgain));
|
|
|
|
return;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
for(size_t j = 0;j < numKeys;j++)
|
|
|
|
for(size_t j = 0;j < numKeys;++j)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
//For now just chomp these
|
|
|
|
//For now just chomp these
|
|
|
|
nif->getFloat();
|
|
|
|
nif.getFloat();
|
|
|
|
nif->getFloat();
|
|
|
|
nif.getFloat();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
nif->file->warn("XYZ_ROTATION_KEY read, but not used!");
|
|
|
|
nif.file->warn("XYZ_ROTATION_KEY read, but not used!");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (mInterpolationType == 0)
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
if (count != 0)
|
|
|
|
|
|
|
|
nif->file->fail("Interpolation type 0 doesn't work with keys");
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else
|
|
|
|
|
|
|
|
nif->file->fail("Unhandled interpolation type: "+Ogre::StringConverter::toString(mInterpolationType));
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
};
|
|
|
|
};
|
|
|
|
typedef KeyListT<float,&NIFStream::getFloat> FloatKeyList;
|
|
|
|
typedef KeyListT<float,&NIFStream::getFloat> FloatKeyList;
|
|
|
|
typedef KeyListT<Ogre::Vector3,&NIFStream::getVector3> Vector3KeyList;
|
|
|
|
typedef KeyListT<Ogre::Vector3,&NIFStream::getVector3> Vector3KeyList;
|
|
|
|