You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
openmw-tes3mp/components/nifosg/controller.hpp

306 lines
9.2 KiB
C++

#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>
#include <components/nifcache/nifcache.hpp>
#include <boost/shared_ptr.hpp>
#include <set> //UVController
// FlipController
#include <osg/Texture2D>
#include <osg/ref_ptr>
#include <osg/Timer>
#include <osg/StateSet>
#include <osg/NodeCallback>
#include <osg/Drawable>
namespace osg
{
class Node;
class StateSet;
}
namespace osgParticle
{
class Emitter;
}
namespace osgAnimation
{
class MorphGeometry;
}
namespace NifOsg
{
// FIXME: Should not be here. We might also want to use this for non-NIF model formats
class ValueInterpolator
{
protected:
template <typename T>
T interpKey (const std::map< float, Nif::KeyT<T> >& keys, float time, T defaultValue = T()) const
{
if (keys.size() == 0)
return defaultValue;
if(time <= keys.begin()->first)
return keys.begin()->second.mValue;
typename std::map< float, Nif::KeyT<T> >::const_iterator it = keys.lower_bound(time);
if (it != keys.end())
{
float aTime = it->first;
const Nif::KeyT<T>* aKey = &it->second;
assert (it != keys.begin()); // Shouldn't happen, was checked at beginning of this function
typename std::map< float, Nif::KeyT<T> >::const_iterator last = --it;
float aLastTime = last->first;
const Nif::KeyT<T>* aLastKey = &last->second;
float a = (time - aLastTime) / (aTime - aLastTime);
return aLastKey->mValue + ((aKey->mValue - aLastKey->mValue) * a);
}
else
return keys.rbegin()->second.mValue;
}
};
// FIXME: Should not be here. We might also want to use this for non-NIF model formats
class ControllerFunction
{
private:
float mFrequency;
float mPhase;
float mStartTime;
float mStopTime;
enum ExtrapolationMode
{
Cycle = 0,
Reverse = 1,
Constant = 2
};
ExtrapolationMode mExtrapolationMode;
public:
ControllerFunction(const Nif::Controller *ctrl);
float calculate(float value);
};
class ControllerSource
{
public:
virtual float getValue(osg::NodeVisitor* nv) = 0;
};
class FrameTimeSource : public ControllerSource
{
public:
FrameTimeSource();
virtual float getValue(osg::NodeVisitor* nv);
};
class Controller
{
public:
Controller();
bool hasInput() const;
float getInputValue(osg::NodeVisitor* nv);
boost::shared_ptr<ControllerSource> mSource;
// The source value gets passed through this function before it's passed on to the DestValue.
boost::shared_ptr<ControllerFunction> mFunction;
};
class GeomMorpherController : public osg::Drawable::UpdateCallback, public Controller, public ValueInterpolator
{
public:
GeomMorpherController(const Nif::NiMorphData* data);
GeomMorpherController();
GeomMorpherController(const GeomMorpherController& copy, const osg::CopyOp& copyop);
META_Object(NifOsg, GeomMorpherController)
virtual void update(osg::NodeVisitor* nv, osg::Drawable* drawable);
private:
std::vector<Nif::FloatKeyMapPtr> mKeyFrames;
};
class KeyframeController : public osg::NodeCallback, public Controller, public ValueInterpolator
{
10 years ago
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*);
private:
Nif::QuaternionKeyMapPtr mRotations;
Nif::FloatKeyMapPtr mXRotations;
Nif::FloatKeyMapPtr mYRotations;
Nif::FloatKeyMapPtr mZRotations;
Nif::Vector3KeyMapPtr mTranslations;
Nif::FloatKeyMapPtr mScales;
using ValueInterpolator::interpKey;
osg::Quat interpKey(const Nif::QuaternionKeyMap::MapType &keys, float time);
osg::Quat getXYZRotation(float time) const;
10 years ago
};
// Specialization that can be enabled/disabled. Used for multiple animation sources support, i.e. .kf files.
10 years ago
// A SourcedKeyframeController is disabled by default and should be manually enabled when playing an animation from
// the relevant animation source.
class SourcedKeyframeController : public KeyframeController
{
public:
SourcedKeyframeController(const Nif::NiKeyframeData* data);
10 years ago
SourcedKeyframeController();
SourcedKeyframeController(const SourcedKeyframeController& copy, const osg::CopyOp& copyop);
10 years ago
META_Object(NifOsg, SourcedKeyframeController)
virtual void operator() (osg::Node*, osg::NodeVisitor*);
10 years ago
void setEnabled(bool enabled);
private:
bool mEnabled;
};
class StateSetController
{
protected:
// Clones a StateSet to make it "writable". This is to prevent race conditions when the OSG draw thread of the last frame
// queues up a StateSet that we want to modify. Note, we could also set the StateSet to DYNAMIC data variance but that would
// undo all the benefits of the threading model - having the cull and draw traversals run in parallel can yield up to 200% framerates.
// If the StateSet allocations per frame are proving too much of an overhead we could "reuse" StateSets from previous frames,
// kind of like a double buffering scheme.
osg::StateSet* getWritableStateSet(osg::Node* node);
};
// Note we're using NodeCallback instead of StateSet::Callback because the StateSet callback doesn't support nesting
class UVController : public osg::NodeCallback, public Controller, public StateSetController, public ValueInterpolator
{
public:
UVController();
UVController(const UVController&,const osg::CopyOp& = osg::CopyOp::SHALLOW_COPY);
UVController(const Nif::NiUVData *data, std::set<int> textureUnits);
META_Object(NifOsg,UVController)
virtual void operator() (osg::Node*, osg::NodeVisitor*);
private:
Nif::FloatKeyMapPtr mUTrans;
Nif::FloatKeyMapPtr mVTrans;
Nif::FloatKeyMapPtr mUScale;
Nif::FloatKeyMapPtr mVScale;
std::set<int> mTextureUnits;
};
class VisController : public osg::NodeCallback, public Controller
{
private:
std::vector<Nif::NiVisData::VisData> mData;
bool calculate(float time) const;
public:
VisController(const Nif::NiVisData *data);
VisController();
VisController(const VisController& copy, const osg::CopyOp& copyop);
META_Object(NifOsg, VisController)
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
};
class AlphaController : public osg::NodeCallback, public Controller, public StateSetController, public ValueInterpolator
{
private:
Nif::FloatKeyMapPtr mData;
public:
AlphaController(const Nif::NiFloatData *data);
AlphaController();
AlphaController(const AlphaController& copy, const osg::CopyOp& copyop);
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
META_Object(NifOsg, AlphaController)
};
class MaterialColorController : public osg::NodeCallback, public Controller, public StateSetController, public ValueInterpolator
{
private:
Nif::Vector3KeyMapPtr mData;
public:
MaterialColorController(const Nif::NiPosData *data);
MaterialColorController();
MaterialColorController(const MaterialColorController& copy, const osg::CopyOp& copyop);
META_Object(NifOsg, MaterialColorController)
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
};
class FlipController : public osg::NodeCallback, public Controller, public StateSetController
{
private:
int mTexSlot;
float mDelta;
std::vector<osg::ref_ptr<osg::Texture2D> > mTextures;
public:
FlipController(const Nif::NiFlipController* ctrl, std::vector<osg::ref_ptr<osg::Texture2D> > textures);
FlipController();
FlipController(const FlipController& copy, const osg::CopyOp& copyop);
META_Object(NifOsg, FlipController)
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
};
class ParticleSystemController : public osg::NodeCallback, public Controller
{
public:
ParticleSystemController(const Nif::NiParticleSystemController* ctrl);
ParticleSystemController();
ParticleSystemController(const ParticleSystemController& copy, const osg::CopyOp& copyop);
META_Object(NifOsg, ParticleSystemController)
virtual void operator() (osg::Node* node, osg::NodeVisitor* nv);
private:
float mEmitStart;
float mEmitStop;
};
}
#endif