|
|
|
@ -35,17 +35,33 @@ namespace NifOsg
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
// interpolation of keyframes
|
|
|
|
|
template <typename MapT, typename InterpolationFunc>
|
|
|
|
|
template <typename MapT>
|
|
|
|
|
class ValueInterpolator
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
typedef typename MapT::ValueType ValueT;
|
|
|
|
|
|
|
|
|
|
ValueInterpolator()
|
|
|
|
|
: mDefaultVal(ValueT())
|
|
|
|
|
typename MapT::MapType::const_iterator retrieveKey(float time) const
|
|
|
|
|
{
|
|
|
|
|
// retrieve the current position in the map, optimized for the most common case
|
|
|
|
|
// where time moves linearly along the keyframe track
|
|
|
|
|
if (mLastHighKey != mKeys->mKeys.end())
|
|
|
|
|
{
|
|
|
|
|
if (time > mLastHighKey->first)
|
|
|
|
|
{
|
|
|
|
|
// try if we're there by incrementing one
|
|
|
|
|
++mLastLowKey;
|
|
|
|
|
++mLastHighKey;
|
|
|
|
|
}
|
|
|
|
|
if (mLastHighKey != mKeys->mKeys.end() && time >= mLastLowKey->first && time <= mLastHighKey->first)
|
|
|
|
|
return mLastHighKey;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return mKeys->mKeys.lower_bound(time);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
using ValueT = typename MapT::ValueType;
|
|
|
|
|
|
|
|
|
|
ValueInterpolator() = default;
|
|
|
|
|
|
|
|
|
|
ValueInterpolator(std::shared_ptr<const MapT> keys, ValueT defaultVal = ValueT())
|
|
|
|
|
: mKeys(keys)
|
|
|
|
|
, mDefaultVal(defaultVal)
|
|
|
|
@ -67,43 +83,20 @@ namespace NifOsg
|
|
|
|
|
if(time <= keys.begin()->first)
|
|
|
|
|
return keys.begin()->second.mValue;
|
|
|
|
|
|
|
|
|
|
// retrieve the current position in the map, optimized for the most common case
|
|
|
|
|
// where time moves linearly along the keyframe track
|
|
|
|
|
typename MapT::MapType::const_iterator it = mLastHighKey;
|
|
|
|
|
if (mLastHighKey != keys.end())
|
|
|
|
|
{
|
|
|
|
|
if (time > mLastHighKey->first)
|
|
|
|
|
{
|
|
|
|
|
// try if we're there by incrementing one
|
|
|
|
|
++mLastLowKey;
|
|
|
|
|
++mLastHighKey;
|
|
|
|
|
it = mLastHighKey;
|
|
|
|
|
}
|
|
|
|
|
if (mLastHighKey == keys.end() || (time < mLastLowKey->first || time > mLastHighKey->first))
|
|
|
|
|
it = keys.lower_bound(time); // still not there, reorient by performing lower_bound check on the whole map
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
it = keys.lower_bound(time);
|
|
|
|
|
typename MapT::MapType::const_iterator it = retrieveKey(time);
|
|
|
|
|
|
|
|
|
|
// now do the actual interpolation
|
|
|
|
|
if (it != keys.end())
|
|
|
|
|
{
|
|
|
|
|
float aTime = it->first;
|
|
|
|
|
const typename MapT::KeyType* aKey = &it->second;
|
|
|
|
|
|
|
|
|
|
// cache for next time
|
|
|
|
|
mLastHighKey = it;
|
|
|
|
|
mLastLowKey = --it;
|
|
|
|
|
|
|
|
|
|
typename MapT::MapType::const_iterator last = --it;
|
|
|
|
|
mLastLowKey = last;
|
|
|
|
|
float aLastTime = last->first;
|
|
|
|
|
const typename MapT::KeyType* aLastKey = &last->second;
|
|
|
|
|
float a = (time - mLastLowKey->first) / (mLastHighKey->first - mLastLowKey->first);
|
|
|
|
|
|
|
|
|
|
float a = (time - aLastTime) / (aTime - aLastTime);
|
|
|
|
|
|
|
|
|
|
return InterpolationFunc()(aLastKey->mValue, aKey->mValue, a);
|
|
|
|
|
return interpolate(mLastLowKey->second, mLastHighKey->second, a, mKeys->mInterpolationType);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
|
|
|
|
|
return keys.rbegin()->second.mValue;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -113,36 +106,44 @@ namespace NifOsg
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
mutable typename MapT::MapType::const_iterator mLastLowKey;
|
|
|
|
|
mutable typename MapT::MapType::const_iterator mLastHighKey;
|
|
|
|
|
|
|
|
|
|
std::shared_ptr<const MapT> mKeys;
|
|
|
|
|
|
|
|
|
|
ValueT mDefaultVal;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct LerpFunc
|
|
|
|
|
{
|
|
|
|
|
template <typename ValueType>
|
|
|
|
|
inline ValueType operator()(const ValueType& a, const ValueType& b, float fraction)
|
|
|
|
|
ValueType interpolate(const Nif::KeyT<ValueType>& a, const Nif::KeyT<ValueType>& b, float fraction, unsigned int type) const
|
|
|
|
|
{
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
return a + ((b - a) * fraction);
|
|
|
|
|
case 5:
|
|
|
|
|
return fraction > 0.5f ? b.mValue : a.mValue;
|
|
|
|
|
default:
|
|
|
|
|
return a.mValue + ((b.mValue - a.mValue) * fraction);
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
struct QuaternionSlerpFunc
|
|
|
|
|
}
|
|
|
|
|
osg::Quat interpolate(const Nif::KeyT<osg::Quat>& a, const Nif::KeyT<osg::Quat>& b, float fraction, unsigned int type) const
|
|
|
|
|
{
|
|
|
|
|
inline osg::Quat operator()(const osg::Quat& a, const osg::Quat& b, float fraction)
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case 5:
|
|
|
|
|
return fraction > 0.5f ? b.mValue : a.mValue;
|
|
|
|
|
default:
|
|
|
|
|
{
|
|
|
|
|
osg::Quat result;
|
|
|
|
|
result.slerp(fraction, a, b);
|
|
|
|
|
result.slerp(fraction, a.mValue, b.mValue);
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
mutable typename MapT::MapType::const_iterator mLastLowKey;
|
|
|
|
|
mutable typename MapT::MapType::const_iterator mLastHighKey;
|
|
|
|
|
|
|
|
|
|
std::shared_ptr<const MapT> mKeys;
|
|
|
|
|
|
|
|
|
|
ValueT mDefaultVal = ValueT();
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
typedef ValueInterpolator<Nif::QuaternionKeyMap, QuaternionSlerpFunc> QuaternionInterpolator;
|
|
|
|
|
typedef ValueInterpolator<Nif::FloatKeyMap, LerpFunc> FloatInterpolator;
|
|
|
|
|
typedef ValueInterpolator<Nif::Vector3KeyMap, LerpFunc> Vec3Interpolator;
|
|
|
|
|
using QuaternionInterpolator = ValueInterpolator<Nif::QuaternionKeyMap>;
|
|
|
|
|
using FloatInterpolator = ValueInterpolator<Nif::FloatKeyMap>;
|
|
|
|
|
using Vec3Interpolator = ValueInterpolator<Nif::Vector3KeyMap>;
|
|
|
|
|
using Vec4Interpolator = ValueInterpolator<Nif::Vector4KeyMap>;
|
|
|
|
|
|
|
|
|
|
class ControllerFunction : public SceneUtil::ControllerFunction
|
|
|
|
|
{
|
|
|
|
|