2015-02-23 20:06:10 +00:00
|
|
|
#ifndef COMPONENTS_NIFOSG_CONTROLLER_H
|
|
|
|
#define COMPONENTS_NIFOSG_CONTROLLER_H
|
|
|
|
|
|
|
|
#include <components/nif/niffile.hpp>
|
|
|
|
#include <components/nif/nifkey.hpp>
|
|
|
|
#include <components/nif/controller.hpp>
|
|
|
|
#include <components/nif/data.hpp>
|
|
|
|
|
2015-04-14 15:29:12 +00:00
|
|
|
#include <components/sceneutil/controller.hpp>
|
|
|
|
#include <components/sceneutil/statesetupdater.hpp>
|
2015-04-14 14:41:06 +00:00
|
|
|
|
2015-02-23 20:06:10 +00:00
|
|
|
#include <set> //UVController
|
|
|
|
|
2015-02-23 23:02:10 +00:00
|
|
|
// FlipController
|
2015-03-28 01:20:20 +00:00
|
|
|
#include <osg/Texture2D>
|
2015-02-23 23:02:10 +00:00
|
|
|
#include <osg/ref_ptr>
|
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
#include <osg/StateSet>
|
|
|
|
#include <osg/NodeCallback>
|
|
|
|
#include <osg/Drawable>
|
2015-02-23 20:06:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
namespace osg
|
|
|
|
{
|
|
|
|
class Node;
|
|
|
|
class StateSet;
|
2020-05-12 13:37:00 +00:00
|
|
|
class Material;
|
2015-02-23 20:06:10 +00:00
|
|
|
}
|
|
|
|
|
2015-03-19 02:01:11 +00:00
|
|
|
namespace osgParticle
|
|
|
|
{
|
|
|
|
class Emitter;
|
|
|
|
}
|
|
|
|
|
2015-02-23 20:06:10 +00:00
|
|
|
namespace NifOsg
|
|
|
|
{
|
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
// interpolation of keyframes
|
2019-12-29 12:53:44 +00:00
|
|
|
template <typename MapT>
|
2015-02-23 20:06:10 +00:00
|
|
|
class ValueInterpolator
|
|
|
|
{
|
2019-12-29 12:53:44 +00:00
|
|
|
typename MapT::MapType::const_iterator retrieveKey(float time) const
|
2015-11-30 19:45:32 +00:00
|
|
|
{
|
2019-12-29 12:53:44 +00:00
|
|
|
// 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);
|
2015-11-30 19:45:32 +00:00
|
|
|
}
|
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
public:
|
|
|
|
using ValueT = typename MapT::ValueType;
|
|
|
|
|
|
|
|
ValueInterpolator() = default;
|
|
|
|
|
2017-05-05 17:26:09 +00:00
|
|
|
ValueInterpolator(std::shared_ptr<const MapT> keys, ValueT defaultVal = ValueT())
|
2015-11-30 19:45:32 +00:00
|
|
|
: mKeys(keys)
|
|
|
|
, mDefaultVal(defaultVal)
|
|
|
|
{
|
|
|
|
if (keys)
|
|
|
|
{
|
|
|
|
mLastLowKey = mKeys->mKeys.end();
|
|
|
|
mLastHighKey = mKeys->mKeys.end();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-01 15:18:19 +00:00
|
|
|
ValueT interpKey(float time) const
|
2015-03-19 16:00:16 +00:00
|
|
|
{
|
2015-11-30 19:45:32 +00:00
|
|
|
if (empty())
|
|
|
|
return mDefaultVal;
|
|
|
|
|
|
|
|
const typename MapT::MapType & keys = mKeys->mKeys;
|
2015-03-19 16:00:16 +00:00
|
|
|
|
|
|
|
if(time <= keys.begin()->first)
|
|
|
|
return keys.begin()->second.mValue;
|
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
typename MapT::MapType::const_iterator it = retrieveKey(time);
|
2015-11-30 19:45:32 +00:00
|
|
|
|
|
|
|
// now do the actual interpolation
|
2015-03-19 16:00:16 +00:00
|
|
|
if (it != keys.end())
|
|
|
|
{
|
2015-11-30 19:45:32 +00:00
|
|
|
// cache for next time
|
|
|
|
mLastHighKey = it;
|
2019-12-29 12:53:44 +00:00
|
|
|
mLastLowKey = --it;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
float a = (time - mLastLowKey->first) / (mLastHighKey->first - mLastLowKey->first);
|
2015-11-30 19:45:32 +00:00
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
return interpolate(mLastLowKey->second, mLastHighKey->second, a, mKeys->mInterpolationType);
|
2015-03-19 16:00:16 +00:00
|
|
|
}
|
2019-12-29 12:53:44 +00:00
|
|
|
|
|
|
|
return keys.rbegin()->second.mValue;
|
2015-03-19 16:00:16 +00:00
|
|
|
}
|
2015-11-30 19:45:32 +00:00
|
|
|
|
|
|
|
bool empty() const
|
|
|
|
{
|
|
|
|
return !mKeys || mKeys->mKeys.empty();
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
template <typename ValueType>
|
2019-12-29 12:53:44 +00:00
|
|
|
ValueType interpolate(const Nif::KeyT<ValueType>& a, const Nif::KeyT<ValueType>& b, float fraction, unsigned int type) const
|
2015-11-30 19:45:32 +00:00
|
|
|
{
|
2019-12-29 12:53:44 +00:00
|
|
|
switch (type)
|
|
|
|
{
|
2019-12-31 10:38:57 +00:00
|
|
|
case Nif::InterpolationType_Constant:
|
2019-12-29 12:53:44 +00:00
|
|
|
return fraction > 0.5f ? b.mValue : a.mValue;
|
|
|
|
default:
|
|
|
|
return a.mValue + ((b.mValue - a.mValue) * fraction);
|
|
|
|
}
|
2015-11-30 19:45:32 +00:00
|
|
|
}
|
2019-12-29 12:53:44 +00:00
|
|
|
osg::Quat interpolate(const Nif::KeyT<osg::Quat>& a, const Nif::KeyT<osg::Quat>& b, float fraction, unsigned int type) const
|
2015-11-30 19:45:32 +00:00
|
|
|
{
|
2019-12-29 12:53:44 +00:00
|
|
|
switch (type)
|
|
|
|
{
|
2019-12-31 10:38:57 +00:00
|
|
|
case Nif::InterpolationType_Constant:
|
2019-12-29 12:53:44 +00:00
|
|
|
return fraction > 0.5f ? b.mValue : a.mValue;
|
|
|
|
default:
|
|
|
|
{
|
|
|
|
osg::Quat result;
|
|
|
|
result.slerp(fraction, a.mValue, b.mValue);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
2015-11-30 19:45:32 +00:00
|
|
|
}
|
2019-12-29 12:53:44 +00:00
|
|
|
|
|
|
|
mutable typename MapT::MapType::const_iterator mLastLowKey;
|
|
|
|
mutable typename MapT::MapType::const_iterator mLastHighKey;
|
|
|
|
|
|
|
|
std::shared_ptr<const MapT> mKeys;
|
|
|
|
|
|
|
|
ValueT mDefaultVal = ValueT();
|
2015-11-30 19:45:32 +00:00
|
|
|
};
|
|
|
|
|
2019-12-29 12:53:44 +00:00
|
|
|
using QuaternionInterpolator = ValueInterpolator<Nif::QuaternionKeyMap>;
|
|
|
|
using FloatInterpolator = ValueInterpolator<Nif::FloatKeyMap>;
|
|
|
|
using Vec3Interpolator = ValueInterpolator<Nif::Vector3KeyMap>;
|
|
|
|
using Vec4Interpolator = ValueInterpolator<Nif::Vector4KeyMap>;
|
2015-11-30 19:45:32 +00:00
|
|
|
|
2015-04-14 15:29:12 +00:00
|
|
|
class ControllerFunction : public SceneUtil::ControllerFunction
|
2015-02-23 20:06:10 +00:00
|
|
|
{
|
|
|
|
private:
|
|
|
|
float mFrequency;
|
|
|
|
float mPhase;
|
|
|
|
float mStartTime;
|
|
|
|
float mStopTime;
|
2015-03-25 17:12:04 +00:00
|
|
|
enum ExtrapolationMode
|
|
|
|
{
|
|
|
|
Cycle = 0,
|
|
|
|
Reverse = 1,
|
|
|
|
Constant = 2
|
|
|
|
};
|
|
|
|
ExtrapolationMode mExtrapolationMode;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
|
|
|
public:
|
2015-03-25 17:12:04 +00:00
|
|
|
ControllerFunction(const Nif::Controller *ctrl);
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-04-22 21:12:58 +00:00
|
|
|
float calculate(float value) const;
|
2015-04-18 23:57:52 +00:00
|
|
|
|
|
|
|
virtual float getMaximum() const;
|
2015-02-23 20:06:10 +00:00
|
|
|
};
|
|
|
|
|
2017-09-01 20:56:09 +00:00
|
|
|
/// Must be set on a SceneUtil::MorphGeometry.
|
2015-11-30 19:45:32 +00:00
|
|
|
class GeomMorpherController : public osg::Drawable::UpdateCallback, public SceneUtil::Controller
|
2015-02-23 20:06:10 +00:00
|
|
|
{
|
|
|
|
public:
|
2015-03-20 18:51:54 +00:00
|
|
|
GeomMorpherController(const Nif::NiMorphData* data);
|
|
|
|
GeomMorpherController();
|
|
|
|
GeomMorpherController(const GeomMorpherController& copy, const osg::CopyOp& copyop);
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
META_Object(NifOsg, GeomMorpherController)
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
virtual void update(osg::NodeVisitor* nv, osg::Drawable* drawable);
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-02-23 20:21:19 +00:00
|
|
|
private:
|
2015-11-30 19:45:32 +00:00
|
|
|
std::vector<FloatInterpolator> mKeyFrames;
|
2015-02-23 20:06:10 +00:00
|
|
|
};
|
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
class KeyframeController : public osg::NodeCallback, public SceneUtil::Controller
|
2015-02-23 20:06:10 +00:00
|
|
|
{
|
2015-03-23 00:31:16 +00:00
|
|
|
public:
|
|
|
|
KeyframeController(const Nif::NiKeyframeData *data);
|
|
|
|
KeyframeController();
|
|
|
|
KeyframeController(const KeyframeController& copy, const osg::CopyOp& copyop);
|
|
|
|
|
|
|
|
META_Object(NifOsg, KeyframeController)
|
|
|
|
|
|
|
|
virtual osg::Vec3f getTranslation(float time) const;
|
|
|
|
|
|
|
|
virtual void operator() (osg::Node*, osg::NodeVisitor*);
|
|
|
|
|
2015-02-23 20:21:19 +00:00
|
|
|
private:
|
2015-11-30 19:45:32 +00:00
|
|
|
QuaternionInterpolator mRotations;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
FloatInterpolator mXRotations;
|
|
|
|
FloatInterpolator mYRotations;
|
|
|
|
FloatInterpolator mZRotations;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
Vec3Interpolator mTranslations;
|
|
|
|
FloatInterpolator mScales;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-02-23 20:21:19 +00:00
|
|
|
osg::Quat getXYZRotation(float time) const;
|
2015-03-23 00:31:16 +00:00
|
|
|
};
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
class UVController : public SceneUtil::StateSetUpdater, public SceneUtil::Controller
|
2015-02-23 20:06:10 +00:00
|
|
|
{
|
2015-03-20 18:51:54 +00:00
|
|
|
public:
|
|
|
|
UVController();
|
2015-11-09 17:57:17 +00:00
|
|
|
UVController(const UVController&,const osg::CopyOp&);
|
2017-04-20 11:36:14 +00:00
|
|
|
UVController(const Nif::NiUVData *data, const std::set<int>& textureUnits);
|
2015-03-20 18:51:54 +00:00
|
|
|
|
|
|
|
META_Object(NifOsg,UVController)
|
|
|
|
|
2015-04-14 14:41:06 +00:00
|
|
|
virtual void setDefaults(osg::StateSet* stateset);
|
|
|
|
virtual void apply(osg::StateSet *stateset, osg::NodeVisitor *nv);
|
2015-03-20 18:51:54 +00:00
|
|
|
|
2015-02-23 20:21:19 +00:00
|
|
|
private:
|
2015-11-30 19:45:32 +00:00
|
|
|
FloatInterpolator mUTrans;
|
|
|
|
FloatInterpolator mVTrans;
|
|
|
|
FloatInterpolator mUScale;
|
|
|
|
FloatInterpolator mVScale;
|
2015-02-23 20:21:19 +00:00
|
|
|
std::set<int> mTextureUnits;
|
2015-02-23 20:06:10 +00:00
|
|
|
};
|
|
|
|
|
2015-04-14 15:29:12 +00:00
|
|
|
class VisController : public osg::NodeCallback, public SceneUtil::Controller
|
2015-02-23 20:06:10 +00:00
|
|
|
{
|
2015-02-23 20:21:19 +00:00
|
|
|
private:
|
|
|
|
std::vector<Nif::NiVisData::VisData> mData;
|
2020-04-22 11:34:19 +00:00
|
|
|
unsigned int mMask;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-02-23 20:21:19 +00:00
|
|
|
bool calculate(float time) const;
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-02-23 20:21:19 +00:00
|
|
|
public:
|
2020-04-22 11:34:19 +00:00
|
|
|
VisController(const Nif::NiVisData *data, unsigned int mask);
|
2015-03-20 18:51:54 +00:00
|
|
|
VisController();
|
|
|
|
VisController(const VisController& copy, const osg::CopyOp& copyop);
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
META_Object(NifOsg, VisController)
|
2015-02-23 20:06:10 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
|
2015-02-23 20:06:10 +00:00
|
|
|
};
|
|
|
|
|
2019-02-20 18:55:40 +00:00
|
|
|
class RollController : public osg::NodeCallback, public SceneUtil::Controller
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
FloatInterpolator mData;
|
|
|
|
double mStartingTime;
|
|
|
|
|
|
|
|
public:
|
|
|
|
RollController(const Nif::NiFloatData *data);
|
|
|
|
RollController();
|
|
|
|
RollController(const RollController& copy, const osg::CopyOp& copyop);
|
|
|
|
|
|
|
|
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
|
|
|
|
|
|
|
|
META_Object(NifOsg, RollController)
|
|
|
|
};
|
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
class AlphaController : public SceneUtil::StateSetUpdater, public SceneUtil::Controller
|
2015-02-23 20:44:01 +00:00
|
|
|
{
|
|
|
|
private:
|
2015-11-30 19:45:32 +00:00
|
|
|
FloatInterpolator mData;
|
2020-05-12 13:37:00 +00:00
|
|
|
osg::ref_ptr<const osg::Material> mBaseMaterial;
|
2015-02-23 20:44:01 +00:00
|
|
|
public:
|
2020-05-12 13:37:00 +00:00
|
|
|
AlphaController(const Nif::NiFloatData *data, const osg::Material* baseMaterial);
|
2015-03-20 18:51:54 +00:00
|
|
|
AlphaController();
|
|
|
|
AlphaController(const AlphaController& copy, const osg::CopyOp& copyop);
|
|
|
|
|
2015-07-30 21:51:37 +00:00
|
|
|
virtual void setDefaults(osg::StateSet* stateset);
|
|
|
|
|
2015-04-14 14:41:06 +00:00
|
|
|
virtual void apply(osg::StateSet* stateset, osg::NodeVisitor* nv);
|
2015-02-23 20:44:01 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
META_Object(NifOsg, AlphaController)
|
2015-02-23 20:44:01 +00:00
|
|
|
};
|
|
|
|
|
2015-11-30 19:45:32 +00:00
|
|
|
class MaterialColorController : public SceneUtil::StateSetUpdater, public SceneUtil::Controller
|
2015-02-23 20:44:01 +00:00
|
|
|
{
|
|
|
|
public:
|
2019-09-12 19:15:41 +00:00
|
|
|
enum TargetColor
|
|
|
|
{
|
|
|
|
Ambient = 0,
|
|
|
|
Diffuse = 1,
|
|
|
|
Specular = 2,
|
|
|
|
Emissive = 3
|
|
|
|
};
|
2020-05-12 13:37:00 +00:00
|
|
|
MaterialColorController(const Nif::NiPosData *data, TargetColor color, const osg::Material* baseMaterial);
|
2015-03-20 18:51:54 +00:00
|
|
|
MaterialColorController();
|
|
|
|
MaterialColorController(const MaterialColorController& copy, const osg::CopyOp& copyop);
|
2015-02-23 20:44:01 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
META_Object(NifOsg, MaterialColorController)
|
|
|
|
|
2015-07-30 22:01:55 +00:00
|
|
|
virtual void setDefaults(osg::StateSet* stateset);
|
|
|
|
|
2015-04-14 14:41:06 +00:00
|
|
|
virtual void apply(osg::StateSet* stateset, osg::NodeVisitor* nv);
|
2019-09-12 19:15:41 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
Vec3Interpolator mData;
|
2019-11-13 10:47:29 +00:00
|
|
|
TargetColor mTargetColor = Ambient;
|
2020-05-12 13:37:00 +00:00
|
|
|
osg::ref_ptr<const osg::Material> mBaseMaterial;
|
2015-02-23 20:44:01 +00:00
|
|
|
};
|
|
|
|
|
2015-04-14 15:29:12 +00:00
|
|
|
class FlipController : public SceneUtil::StateSetUpdater, public SceneUtil::Controller
|
2015-02-23 23:02:10 +00:00
|
|
|
{
|
|
|
|
private:
|
|
|
|
int mTexSlot;
|
|
|
|
float mDelta;
|
2015-03-28 01:20:20 +00:00
|
|
|
std::vector<osg::ref_ptr<osg::Texture2D> > mTextures;
|
2015-02-23 23:02:10 +00:00
|
|
|
|
|
|
|
public:
|
2017-04-20 11:36:14 +00:00
|
|
|
FlipController(const Nif::NiFlipController* ctrl, const std::vector<osg::ref_ptr<osg::Texture2D> >& textures);
|
|
|
|
FlipController(int texSlot, float delta, const std::vector<osg::ref_ptr<osg::Texture2D> >& textures);
|
2015-03-20 18:51:54 +00:00
|
|
|
FlipController();
|
|
|
|
FlipController(const FlipController& copy, const osg::CopyOp& copyop);
|
|
|
|
|
|
|
|
META_Object(NifOsg, FlipController)
|
2015-02-23 23:02:10 +00:00
|
|
|
|
2016-02-05 17:51:51 +00:00
|
|
|
std::vector<osg::ref_ptr<osg::Texture2D> >& getTextures() { return mTextures; }
|
|
|
|
|
2015-04-14 14:41:06 +00:00
|
|
|
virtual void apply(osg::StateSet *stateset, osg::NodeVisitor *nv);
|
2015-02-23 23:02:10 +00:00
|
|
|
};
|
|
|
|
|
2015-04-14 15:29:12 +00:00
|
|
|
class ParticleSystemController : public osg::NodeCallback, public SceneUtil::Controller
|
2015-03-19 02:01:11 +00:00
|
|
|
{
|
|
|
|
public:
|
2015-03-20 18:51:54 +00:00
|
|
|
ParticleSystemController(const Nif::NiParticleSystemController* ctrl);
|
|
|
|
ParticleSystemController();
|
|
|
|
ParticleSystemController(const ParticleSystemController& copy, const osg::CopyOp& copyop);
|
|
|
|
|
|
|
|
META_Object(NifOsg, ParticleSystemController)
|
2015-03-19 02:01:11 +00:00
|
|
|
|
2015-03-20 18:51:54 +00:00
|
|
|
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
|
2015-03-19 02:01:11 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
float mEmitStart;
|
|
|
|
float mEmitStop;
|
|
|
|
};
|
|
|
|
|
2019-10-16 20:21:32 +00:00
|
|
|
class PathController : public osg::NodeCallback, public SceneUtil::Controller
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
PathController(const Nif::NiPathController* ctrl);
|
|
|
|
PathController() = default;
|
|
|
|
PathController(const PathController& copy, const osg::CopyOp& copyop);
|
|
|
|
|
|
|
|
META_Object(NifOsg, PathController)
|
|
|
|
|
|
|
|
virtual void operator() (osg::Node*, osg::NodeVisitor*);
|
|
|
|
|
|
|
|
private:
|
|
|
|
Vec3Interpolator mPath;
|
|
|
|
FloatInterpolator mPercent;
|
|
|
|
int mFlags{0};
|
|
|
|
|
|
|
|
float getPercent(float time) const;
|
|
|
|
};
|
|
|
|
|
2015-02-23 20:06:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|