#ifndef COMPONENTS_NIFOSG_CONTROLLER_H #define COMPONENTS_NIFOSG_CONTROLLER_H #include #include #include #include #include #include #include //UVController // FlipController #include #include #include #include #include #include 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 T interpKey (const std::map< float, Nif::KeyT >& 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 >::const_iterator it = keys.lower_bound(time); if (it != keys.end()) { float aTime = it->first; const Nif::KeyT* aKey = &it->second; assert (it != keys.begin()); // Shouldn't happen, was checked at beginning of this function typename std::map< float, Nif::KeyT >::const_iterator last = --it; float aLastTime = last->first; const Nif::KeyT* 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 mSource; // The source value gets passed through this function before it's passed on to the DestValue. boost::shared_ptr 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 mKeyFrames; }; class KeyframeController : public osg::NodeCallback, public Controller, public ValueInterpolator { 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; }; // Specialization that can be enabled/disabled. Used for multiple animation sources support, i.e. .kf files. // 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); SourcedKeyframeController(); SourcedKeyframeController(const SourcedKeyframeController& copy, const osg::CopyOp& copyop); META_Object(NifOsg, SourcedKeyframeController) virtual void operator() (osg::Node*, osg::NodeVisitor*); 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 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 mTextureUnits; }; class VisController : public osg::NodeCallback, public Controller { private: std::vector 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 > mTextures; public: FlipController(const Nif::NiFlipController* ctrl, std::vector > 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