mirror of
				https://github.com/OpenMW/openmw.git
				synced 2025-10-25 17:56:37 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			285 lines
		
	
	
	
		
			9.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			285 lines
		
	
	
	
		
			9.1 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #ifndef OPENMW_COMPONENTS_NIFOSG_PARTICLE_H
 | |
| #define OPENMW_COMPONENTS_NIFOSG_PARTICLE_H
 | |
| 
 | |
| #include <optional>
 | |
| 
 | |
| #include <osgParticle/Counter>
 | |
| #include <osgParticle/Emitter>
 | |
| #include <osgParticle/Operator>
 | |
| #include <osgParticle/Particle>
 | |
| #include <osgParticle/Placer>
 | |
| #include <osgParticle/Shooter>
 | |
| 
 | |
| #include <components/nif/particle.hpp> // NiGravity::ForceType
 | |
| 
 | |
| #include <components/sceneutil/nodecallback.hpp>
 | |
| 
 | |
| #include "controller.hpp" // ValueInterpolator
 | |
| 
 | |
| namespace Nif
 | |
| {
 | |
|     struct NiColorData;
 | |
| }
 | |
| 
 | |
| namespace NifOsg
 | |
| {
 | |
| 
 | |
|     // Subclass ParticleSystem to support a limit on the number of active particles.
 | |
|     class ParticleSystem : public osgParticle::ParticleSystem
 | |
|     {
 | |
|     public:
 | |
|         ParticleSystem();
 | |
|         ParticleSystem(const ParticleSystem& copy, const osg::CopyOp& copyop);
 | |
| 
 | |
|         META_Object(NifOsg, ParticleSystem)
 | |
| 
 | |
|         osgParticle::Particle* createParticle(const osgParticle::Particle* ptemplate) override;
 | |
| 
 | |
|         void setQuota(int quota);
 | |
| 
 | |
|         void drawImplementation(osg::RenderInfo& renderInfo) const override;
 | |
| 
 | |
|     private:
 | |
|         int mQuota;
 | |
|         osg::ref_ptr<osg::Vec3Array> mNormalArray;
 | |
|     };
 | |
| 
 | |
|     // HACK: Particle doesn't allow setting the initial age, but we need this for loading the particle system state
 | |
|     class ParticleAgeSetter : public osgParticle::Particle
 | |
|     {
 | |
|     public:
 | |
|         ParticleAgeSetter(float age)
 | |
|             : Particle()
 | |
|         {
 | |
|             _t0 = age;
 | |
|         }
 | |
|     };
 | |
| 
 | |
|     // Node callback used to set the inverse of the parent's world matrix on the MatrixTransform
 | |
|     // that the callback is attached to. Used for certain particle systems,
 | |
|     // so that the particles do not move with the node they are attached to.
 | |
|     class InverseWorldMatrix : public SceneUtil::NodeCallback<InverseWorldMatrix, osg::MatrixTransform*>
 | |
|     {
 | |
|     public:
 | |
|         InverseWorldMatrix() {}
 | |
|         InverseWorldMatrix(const InverseWorldMatrix& copy, const osg::CopyOp& copyop)
 | |
|             : osg::Object(copy, copyop)
 | |
|             , SceneUtil::NodeCallback<InverseWorldMatrix, osg::MatrixTransform*>(copy, copyop)
 | |
|         {
 | |
|         }
 | |
| 
 | |
|         META_Object(NifOsg, InverseWorldMatrix)
 | |
| 
 | |
|         void operator()(osg::MatrixTransform* node, osg::NodeVisitor* nv);
 | |
|     };
 | |
| 
 | |
|     class ParticleShooter : public osgParticle::Shooter
 | |
|     {
 | |
|     public:
 | |
|         ParticleShooter(float minSpeed, float maxSpeed, float horizontalDir, float horizontalAngle, float verticalDir,
 | |
|             float verticalAngle, float lifetime, float lifetimeRandom);
 | |
|         ParticleShooter();
 | |
|         ParticleShooter(const ParticleShooter& copy, const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY);
 | |
| 
 | |
|         ParticleShooter& operator=(const ParticleShooter&) = delete;
 | |
| 
 | |
|         META_Object(NifOsg, ParticleShooter)
 | |
| 
 | |
|         void shoot(osgParticle::Particle* particle) const override;
 | |
| 
 | |
|     private:
 | |
|         float mMinSpeed;
 | |
|         float mMaxSpeed;
 | |
|         float mHorizontalDir;
 | |
|         float mHorizontalAngle;
 | |
|         float mVerticalDir;
 | |
|         float mVerticalAngle;
 | |
|         float mLifetime;
 | |
|         float mLifetimeRandom;
 | |
|     };
 | |
| 
 | |
|     class PlanarCollider : public osgParticle::Operator
 | |
|     {
 | |
|     public:
 | |
|         PlanarCollider(const Nif::NiPlanarCollider* collider);
 | |
|         PlanarCollider() = default;
 | |
|         PlanarCollider(const PlanarCollider& copy, const osg::CopyOp& copyop);
 | |
| 
 | |
|         META_Object(NifOsg, PlanarCollider)
 | |
| 
 | |
|         void beginOperate(osgParticle::Program* program) override;
 | |
|         void operate(osgParticle::Particle* particle, double dt) override;
 | |
| 
 | |
|     private:
 | |
|         float mBounceFactor{ 0.f };
 | |
|         osg::Vec2f mExtents;
 | |
|         osg::Vec3f mPosition, mPositionInParticleSpace;
 | |
|         osg::Vec3f mXVector, mXVectorInParticleSpace;
 | |
|         osg::Vec3f mYVector, mYVectorInParticleSpace;
 | |
|         osg::Plane mPlane, mPlaneInParticleSpace;
 | |
|     };
 | |
| 
 | |
|     class SphericalCollider : public osgParticle::Operator
 | |
|     {
 | |
|     public:
 | |
|         SphericalCollider(const Nif::NiSphericalCollider* collider);
 | |
|         SphericalCollider();
 | |
|         SphericalCollider(const SphericalCollider& copy, const osg::CopyOp& copyop);
 | |
| 
 | |
|         META_Object(NifOsg, SphericalCollider)
 | |
| 
 | |
|         void beginOperate(osgParticle::Program* program) override;
 | |
|         void operate(osgParticle::Particle* particle, double dt) override;
 | |
| 
 | |
|     private:
 | |
|         float mBounceFactor;
 | |
|         osg::BoundingSphere mSphere;
 | |
|         osg::BoundingSphere mSphereInParticleSpace;
 | |
|     };
 | |
| 
 | |
|     class GrowFadeAffector : public osgParticle::Operator
 | |
|     {
 | |
|     public:
 | |
|         GrowFadeAffector(float growTime, float fadeTime);
 | |
|         GrowFadeAffector();
 | |
|         GrowFadeAffector(const GrowFadeAffector& copy, const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY);
 | |
| 
 | |
|         GrowFadeAffector& operator=(const GrowFadeAffector&) = delete;
 | |
| 
 | |
|         META_Object(NifOsg, GrowFadeAffector)
 | |
| 
 | |
|         void beginOperate(osgParticle::Program* program) override;
 | |
|         void operate(osgParticle::Particle* particle, double dt) override;
 | |
| 
 | |
|     private:
 | |
|         float mGrowTime;
 | |
|         float mFadeTime;
 | |
| 
 | |
|         float mCachedDefaultSize;
 | |
|     };
 | |
| 
 | |
|     class ParticleColorAffector : public osgParticle::Operator
 | |
|     {
 | |
|     public:
 | |
|         ParticleColorAffector(const Nif::NiColorData* clrdata);
 | |
|         ParticleColorAffector();
 | |
|         ParticleColorAffector(const ParticleColorAffector& copy, const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY);
 | |
| 
 | |
|         ParticleColorAffector& operator=(const ParticleColorAffector&) = delete;
 | |
| 
 | |
|         META_Object(NifOsg, ParticleColorAffector)
 | |
| 
 | |
|         void operate(osgParticle::Particle* particle, double dt) override;
 | |
| 
 | |
|     private:
 | |
|         Vec4Interpolator mData;
 | |
|     };
 | |
| 
 | |
|     class GravityAffector : public osgParticle::Operator
 | |
|     {
 | |
|     public:
 | |
|         GravityAffector(const Nif::NiGravity* gravity);
 | |
|         GravityAffector() = default;
 | |
|         GravityAffector(const GravityAffector& copy, const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY);
 | |
| 
 | |
|         GravityAffector& operator=(const GravityAffector&) = delete;
 | |
| 
 | |
|         META_Object(NifOsg, GravityAffector)
 | |
| 
 | |
|         void operate(osgParticle::Particle* particle, double dt) override;
 | |
|         void beginOperate(osgParticle::Program*) override;
 | |
| 
 | |
|     private:
 | |
|         float mForce{ 0.f };
 | |
|         Nif::ForceType mType{ Nif::ForceType::Wind };
 | |
|         osg::Vec3f mPosition;
 | |
|         osg::Vec3f mDirection;
 | |
|         float mDecay{ 0.f };
 | |
|         osg::Vec3f mCachedWorldPosition;
 | |
|         osg::Vec3f mCachedWorldDirection;
 | |
|     };
 | |
| 
 | |
|     class ParticleBomb : public osgParticle::Operator
 | |
|     {
 | |
|     public:
 | |
|         ParticleBomb(const Nif::NiParticleBomb* bomb);
 | |
|         ParticleBomb() = default;
 | |
|         ParticleBomb(const ParticleBomb& copy, const osg::CopyOp& copyop = osg::CopyOp::SHALLOW_COPY);
 | |
| 
 | |
|         ParticleBomb& operator=(const ParticleBomb&) = delete;
 | |
| 
 | |
|         META_Object(NifOsg, ParticleBomb)
 | |
| 
 | |
|         void operate(osgParticle::Particle* particle, double dt) override;
 | |
|         void beginOperate(osgParticle::Program*) override;
 | |
| 
 | |
|     private:
 | |
|         float mRange{ 0.f };
 | |
|         float mStrength{ 0.f };
 | |
|         Nif::DecayType mDecayType{ Nif::DecayType::None };
 | |
|         Nif::SymmetryType mSymmetryType{ Nif::SymmetryType::Spherical };
 | |
|         osg::Vec3f mPosition;
 | |
|         osg::Vec3f mDirection;
 | |
|         osg::Vec3f mCachedWorldPosition;
 | |
|         osg::Vec3f mCachedWorldDirection;
 | |
|     };
 | |
| 
 | |
|     // NodeVisitor to find a Group node with the given record index, stored in the node's user data container.
 | |
|     // Alternatively, returns the node's parent Group if that node is not a Group (i.e. a leaf node).
 | |
|     class FindGroupByRecIndex : public osg::NodeVisitor
 | |
|     {
 | |
|     public:
 | |
|         FindGroupByRecIndex(unsigned int recIndex);
 | |
| 
 | |
|         void apply(osg::Node& node) override;
 | |
| 
 | |
|         // Technically not required as the default implementation would trickle down to apply(Node&) anyway,
 | |
|         // but we'll shortcut instead to avoid the chain of virtual function calls
 | |
|         void apply(osg::MatrixTransform& node) override;
 | |
|         void apply(osg::Geometry& node) override;
 | |
| 
 | |
|         void applyNode(osg::Node& searchNode);
 | |
| 
 | |
|         osg::Group* mFound;
 | |
|         osg::NodePath mFoundPath;
 | |
| 
 | |
|     private:
 | |
|         unsigned int mRecIndex;
 | |
|     };
 | |
| 
 | |
|     // Subclass emitter to support randomly choosing one of the child node's transforms for the emit position of new
 | |
|     // particles.
 | |
|     class Emitter : public osgParticle::Emitter
 | |
|     {
 | |
|     public:
 | |
|         Emitter(const std::vector<int>& targets);
 | |
|         Emitter();
 | |
|         Emitter(const Emitter& copy, const osg::CopyOp& copyop);
 | |
| 
 | |
|         META_Object(NifOsg, Emitter)
 | |
| 
 | |
|         void emitParticles(double dt) override;
 | |
| 
 | |
|         void setShooter(osgParticle::Shooter* shooter) { mShooter = shooter; }
 | |
|         void setPlacer(osgParticle::Placer* placer) { mPlacer = placer; }
 | |
|         void setCounter(osgParticle::Counter* counter) { mCounter = counter; }
 | |
|         void setGeometryEmitterTarget(std::optional<int> recIndex) { mGeometryEmitterTarget = recIndex; }
 | |
|         void setFlags(int flags) { mFlags = flags; }
 | |
| 
 | |
|     private:
 | |
|         // NIF Record indices
 | |
|         std::vector<int> mTargets;
 | |
| 
 | |
|         osg::ref_ptr<osgParticle::Placer> mPlacer;
 | |
|         osg::ref_ptr<osgParticle::Shooter> mShooter;
 | |
|         osg::ref_ptr<osgParticle::Counter> mCounter;
 | |
| 
 | |
|         int mFlags;
 | |
| 
 | |
|         std::optional<int> mGeometryEmitterTarget;
 | |
|         osg::observer_ptr<osg::Vec3Array> mCachedGeometryEmitter;
 | |
|     };
 | |
| 
 | |
| }
 | |
| 
 | |
| #endif
 |