#include "particles.hpp" #include #include #include #include #include #include #include #include #include #include #include /* FIXME: "Nif" isn't really an appropriate emitter name. */ class NifEmitter : public Ogre::ParticleEmitter { public: std::string mSkelBaseName; Ogre::Bone* mBone; Ogre::ParticleSystem* getPartSys() { return mParent; } class CmdSkelBase : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { assert(false && "Unimplemented"); return ""; } void doSet(void *target, const Ogre::String &val) { NifEmitter* emitter = static_cast(target); emitter->mSkelBaseName = val; } }; class CmdBone : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { assert(false && "Unimplemented"); return ""; } void doSet(void *target, const Ogre::String &val) { NifEmitter* emitter = static_cast(target); assert(!emitter->mSkelBaseName.empty() && "Base entity needs to be set first"); Ogre::ParticleSystem* partsys = emitter->getPartSys(); Ogre::Entity* ent = partsys->getParentSceneNode()->getCreator()->getEntity(emitter->mSkelBaseName); Ogre::Bone* bone = ent->getSkeleton()->getBone(val); assert(bone); emitter->mBone = bone; } }; /** Command object for the emitter width (see Ogre::ParamCommand).*/ class CmdWidth : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { return Ogre::StringConverter::toString(static_cast(target)->getWidth()); } void doSet(void *target, const Ogre::String &val) { static_cast(target)->setWidth(Ogre::StringConverter::parseReal(val)); } }; /** Command object for the emitter height (see Ogre::ParamCommand).*/ class CmdHeight : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { return Ogre::StringConverter::toString(static_cast(target)->getHeight()); } void doSet(void *target, const Ogre::String &val) { static_cast(target)->setHeight(Ogre::StringConverter::parseReal(val)); } }; /** Command object for the emitter depth (see Ogre::ParamCommand).*/ class CmdDepth : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { return Ogre::StringConverter::toString(static_cast(target)->getDepth()); } void doSet(void *target, const Ogre::String &val) { static_cast(target)->setDepth(Ogre::StringConverter::parseReal(val)); } }; /** Command object for the emitter vertical_direction (see Ogre::ParamCommand).*/ class CmdVerticalDir : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const NifEmitter *self = static_cast(target); return Ogre::StringConverter::toString(self->getVerticalDirection().valueDegrees()); } void doSet(void *target, const Ogre::String &val) { NifEmitter *self = static_cast(target); self->setVerticalDirection(Ogre::Degree(Ogre::StringConverter::parseReal(val))); } }; /** Command object for the emitter vertical_angle (see Ogre::ParamCommand).*/ class CmdVerticalAngle : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const NifEmitter *self = static_cast(target); return Ogre::StringConverter::toString(self->getVerticalAngle().valueDegrees()); } void doSet(void *target, const Ogre::String &val) { NifEmitter *self = static_cast(target); self->setVerticalAngle(Ogre::Degree(Ogre::StringConverter::parseReal(val))); } }; /** Command object for the emitter horizontal_direction (see Ogre::ParamCommand).*/ class CmdHorizontalDir : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const NifEmitter *self = static_cast(target); return Ogre::StringConverter::toString(self->getHorizontalDirection().valueDegrees()); } void doSet(void *target, const Ogre::String &val) { NifEmitter *self = static_cast(target); self->setHorizontalDirection(Ogre::Degree(Ogre::StringConverter::parseReal(val))); } }; /** Command object for the emitter horizontal_angle (see Ogre::ParamCommand).*/ class CmdHorizontalAngle : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const NifEmitter *self = static_cast(target); return Ogre::StringConverter::toString(self->getHorizontalAngle().valueDegrees()); } void doSet(void *target, const Ogre::String &val) { NifEmitter *self = static_cast(target); self->setHorizontalAngle(Ogre::Degree(Ogre::StringConverter::parseReal(val))); } }; NifEmitter(Ogre::ParticleSystem *psys) : Ogre::ParticleEmitter(psys) , mBone(NULL) { initDefaults("Nif"); } /** See Ogre::ParticleEmitter. */ unsigned short _getEmissionCount(Ogre::Real timeElapsed) { // Use basic constant emission return genConstantEmissionCount(timeElapsed); } /** See Ogre::ParticleEmitter. */ void _initParticle(Ogre::Particle *particle) { assert (mBone && "No node set"); Ogre::Vector3 xOff, yOff, zOff; // Call superclass ParticleEmitter::_initParticle(particle); xOff = Ogre::Math::SymmetricRandom() * mXRange; yOff = Ogre::Math::SymmetricRandom() * mYRange; zOff = Ogre::Math::SymmetricRandom() * mZRange; #if OGRE_VERSION >= (1 << 16 | 10 << 8 | 0) Ogre::Vector3& position = particle->mPosition; Ogre::Vector3& direction = particle->mDirection; Ogre::ColourValue& colour = particle->mColour; Ogre::Real& totalTimeToLive = particle->mTotalTimeToLive; Ogre::Real& timeToLive = particle->mTimeToLive; #else Ogre::Vector3& position = particle->position; Ogre::Vector3& direction = particle->direction; Ogre::ColourValue& colour = particle->colour; Ogre::Real& totalTimeToLive = particle->totalTimeToLive; Ogre::Real& timeToLive = particle->timeToLive; #endif position = mBone->_getDerivedPosition() + xOff + yOff + zOff; // Generate complex data by reference genEmissionColour(colour); // NOTE: We do not use mDirection/mAngle for the initial direction. Ogre::Radian hdir = mHorizontalDir + mHorizontalAngle*Ogre::Math::SymmetricRandom(); Ogre::Radian vdir = mVerticalDir + mVerticalAngle*Ogre::Math::SymmetricRandom(); direction = (mBone->_getDerivedOrientation() * Ogre::Quaternion(hdir, Ogre::Vector3::UNIT_Z) * Ogre::Quaternion(vdir, Ogre::Vector3::UNIT_X)) * Ogre::Vector3::UNIT_Z; genEmissionVelocity(direction); // Generate simpler data timeToLive = totalTimeToLive = genEmissionTTL(); } /** Overloaded to update the trans. matrix */ void setDirection(const Ogre::Vector3 &dir) { ParticleEmitter::setDirection(dir); genAreaAxes(); } /** Sets the size of the area from which particles are emitted. @param size Vector describing the size of the area. The area extends around the center point by half the x, y and z components of this vector. The box is aligned such that it's local Z axis points along it's direction (see setDirection) */ void setSize(const Ogre::Vector3 &size) { mSize = size; genAreaAxes(); } /** Sets the size of the area from which particles are emitted. @param x,y,z Individual axis lengths describing the size of the area. The area extends around the center point by half the x, y and z components of this vector. The box is aligned such that it's local Z axis points along it's direction (see setDirection) */ void setSize(Ogre::Real x, Ogre::Real y, Ogre::Real z) { mSize.x = x; mSize.y = y; mSize.z = z; genAreaAxes(); } /** Sets the width (local x size) of the emitter. */ void setWidth(Ogre::Real width) { mSize.x = width; genAreaAxes(); } /** Gets the width (local x size) of the emitter. */ Ogre::Real getWidth(void) const { return mSize.x; } /** Sets the height (local y size) of the emitter. */ void setHeight(Ogre::Real height) { mSize.y = height; genAreaAxes(); } /** Gets the height (local y size) of the emitter. */ Ogre::Real getHeight(void) const { return mSize.y; } /** Sets the depth (local y size) of the emitter. */ void setDepth(Ogre::Real depth) { mSize.z = depth; genAreaAxes(); } /** Gets the depth (local y size) of the emitter. */ Ogre::Real getDepth(void) const { return mSize.z; } void setVerticalDirection(Ogre::Radian vdir) { mVerticalDir = vdir; } Ogre::Radian getVerticalDirection(void) const { return mVerticalDir; } void setVerticalAngle(Ogre::Radian vangle) { mVerticalAngle = vangle; } Ogre::Radian getVerticalAngle(void) const { return mVerticalAngle; } void setHorizontalDirection(Ogre::Radian hdir) { mHorizontalDir = hdir; } Ogre::Radian getHorizontalDirection(void) const { return mHorizontalDir; } void setHorizontalAngle(Ogre::Radian hangle) { mHorizontalAngle = hangle; } Ogre::Radian getHorizontalAngle(void) const { return mHorizontalAngle; } protected: /// Size of the area Ogre::Vector3 mSize; Ogre::Radian mVerticalDir; Ogre::Radian mVerticalAngle; Ogre::Radian mHorizontalDir; Ogre::Radian mHorizontalAngle; /// Local axes, not normalised, their magnitude reflects area size Ogre::Vector3 mXRange, mYRange, mZRange; /// Internal method for generating the area axes void genAreaAxes(void) { Ogre::Vector3 mLeft = mUp.crossProduct(mDirection); mXRange = mLeft * (mSize.x * 0.5f); mYRange = mUp * (mSize.y * 0.5f); mZRange = mDirection * (mSize.z * 0.5f); } /** Internal for initializing some defaults and parameters @return True if custom parameters need initialising */ bool initDefaults(const Ogre::String &t) { // Defaults mDirection = Ogre::Vector3::UNIT_Z; mUp = Ogre::Vector3::UNIT_Y; setSize(100.0f, 100.0f, 100.0f); mType = t; // Set up parameters if(createParamDictionary(mType + "Emitter")) { addBaseParameters(); Ogre::ParamDictionary *dict = getParamDictionary(); // Custom params dict->addParameter(Ogre::ParameterDef("width", "Width of the shape in world coordinates.", Ogre::PT_REAL), &msWidthCmd); dict->addParameter(Ogre::ParameterDef("height", "Height of the shape in world coordinates.", Ogre::PT_REAL), &msHeightCmd); dict->addParameter(Ogre::ParameterDef("depth", "Depth of the shape in world coordinates.", Ogre::PT_REAL), &msDepthCmd); dict->addParameter(Ogre::ParameterDef("vertical_direction", "Vertical direction of emitted particles (in degrees).", Ogre::PT_REAL), &msVerticalDirCmd); dict->addParameter(Ogre::ParameterDef("vertical_angle", "Vertical direction variance of emitted particles (in degrees).", Ogre::PT_REAL), &msVerticalAngleCmd); dict->addParameter(Ogre::ParameterDef("horizontal_direction", "Horizontal direction of emitted particles (in degrees).", Ogre::PT_REAL), &msHorizontalDirCmd); dict->addParameter(Ogre::ParameterDef("horizontal_angle", "Horizontal direction variance of emitted particles (in degrees).", Ogre::PT_REAL), &msHorizontalAngleCmd); dict->addParameter(Ogre::ParameterDef("bone", "The bone where the particles should be spawned", Ogre::PT_STRING), &msBoneCmd); dict->addParameter(Ogre::ParameterDef("skelbase", "The name of the entity containing the bone (see 'bone' parameter)", Ogre::PT_STRING), &msSkelBaseCmd); return true; } return false; } /// Command objects static CmdWidth msWidthCmd; static CmdHeight msHeightCmd; static CmdDepth msDepthCmd; static CmdVerticalDir msVerticalDirCmd; static CmdVerticalAngle msVerticalAngleCmd; static CmdHorizontalDir msHorizontalDirCmd; static CmdHorizontalAngle msHorizontalAngleCmd; static CmdBone msBoneCmd; static CmdSkelBase msSkelBaseCmd; }; NifEmitter::CmdWidth NifEmitter::msWidthCmd; NifEmitter::CmdHeight NifEmitter::msHeightCmd; NifEmitter::CmdDepth NifEmitter::msDepthCmd; NifEmitter::CmdVerticalDir NifEmitter::msVerticalDirCmd; NifEmitter::CmdVerticalAngle NifEmitter::msVerticalAngleCmd; NifEmitter::CmdHorizontalDir NifEmitter::msHorizontalDirCmd; NifEmitter::CmdHorizontalAngle NifEmitter::msHorizontalAngleCmd; NifEmitter::CmdBone NifEmitter::msBoneCmd; NifEmitter::CmdSkelBase NifEmitter::msSkelBaseCmd; Ogre::ParticleEmitter* NifEmitterFactory::createEmitter(Ogre::ParticleSystem *psys) { Ogre::ParticleEmitter *emitter = OGRE_NEW NifEmitter(psys); mEmitters.push_back(emitter); return emitter; } class GrowFadeAffector : public Ogre::ParticleAffector { public: /** Command object for grow_time (see Ogre::ParamCommand).*/ class CmdGrowTime : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const GrowFadeAffector *self = static_cast(target); return Ogre::StringConverter::toString(self->getGrowTime()); } void doSet(void *target, const Ogre::String &val) { GrowFadeAffector *self = static_cast(target); self->setGrowTime(Ogre::StringConverter::parseReal(val)); } }; /** Command object for fade_time (see Ogre::ParamCommand).*/ class CmdFadeTime : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const GrowFadeAffector *self = static_cast(target); return Ogre::StringConverter::toString(self->getFadeTime()); } void doSet(void *target, const Ogre::String &val) { GrowFadeAffector *self = static_cast(target); self->setFadeTime(Ogre::StringConverter::parseReal(val)); } }; /** Default constructor. */ GrowFadeAffector(Ogre::ParticleSystem *psys) : ParticleAffector(psys) { mGrowTime = 0.0f; mFadeTime = 0.0f; mType = "GrowFade"; // Init parameters if(createParamDictionary("GrowFadeAffector")) { Ogre::ParamDictionary *dict = getParamDictionary(); Ogre::String grow_title("grow_time"); Ogre::String fade_title("fade_time"); Ogre::String grow_descr("Time from begin to reach full size."); Ogre::String fade_descr("Time from end to shrink."); dict->addParameter(Ogre::ParameterDef(grow_title, grow_descr, Ogre::PT_REAL), &msGrowCmd); dict->addParameter(Ogre::ParameterDef(fade_title, fade_descr, Ogre::PT_REAL), &msFadeCmd); } } /** See Ogre::ParticleAffector. */ void _initParticle(Ogre::Particle *particle) { #if OGRE_VERSION >= (1 << 16 | 10 << 8 | 0) const Ogre::Real life_time = particle->mTotalTimeToLive; Ogre::Real particle_time = particle->mTimeToLive; #else const Ogre::Real life_time = particle->totalTimeToLive; Ogre::Real particle_time = particle->timeToLive; #endif Ogre::Real width = mParent->getDefaultWidth(); Ogre::Real height = mParent->getDefaultHeight(); if(life_time-particle_time < mGrowTime) { Ogre::Real scale = (life_time-particle_time) / mGrowTime; width *= scale; height *= scale; } if(particle_time < mFadeTime) { Ogre::Real scale = particle_time / mFadeTime; width *= scale; height *= scale; } particle->setDimensions(width, height); } /** See Ogre::ParticleAffector. */ void _affectParticles(Ogre::ParticleSystem *psys, Ogre::Real timeElapsed) { Ogre::ParticleIterator pi = psys->_getIterator(); while (!pi.end()) { Ogre::Particle *p = pi.getNext(); #if OGRE_VERSION >= (1 << 16 | 10 << 8 | 0) const Ogre::Real life_time = p->mTotalTimeToLive; Ogre::Real particle_time = p->mTimeToLive; #else const Ogre::Real life_time = p->totalTimeToLive; Ogre::Real particle_time = p->timeToLive; #endif Ogre::Real width = mParent->getDefaultWidth(); Ogre::Real height = mParent->getDefaultHeight(); if(life_time-particle_time < mGrowTime) { Ogre::Real scale = (life_time-particle_time) / mGrowTime; width *= scale; height *= scale; } if(particle_time < mFadeTime) { Ogre::Real scale = particle_time / mFadeTime; width *= scale; height *= scale; } p->setDimensions(width, height); } } void setGrowTime(Ogre::Real time) { mGrowTime = time; } Ogre::Real getGrowTime() const { return mGrowTime; } void setFadeTime(Ogre::Real time) { mFadeTime = time; } Ogre::Real getFadeTime() const { return mFadeTime; } static CmdGrowTime msGrowCmd; static CmdFadeTime msFadeCmd; protected: Ogre::Real mGrowTime; Ogre::Real mFadeTime; }; GrowFadeAffector::CmdGrowTime GrowFadeAffector::msGrowCmd; GrowFadeAffector::CmdFadeTime GrowFadeAffector::msFadeCmd; Ogre::ParticleAffector *GrowFadeAffectorFactory::createAffector(Ogre::ParticleSystem *psys) { Ogre::ParticleAffector *p = new GrowFadeAffector(psys); mAffectors.push_back(p); return p; } class GravityAffector : public Ogre::ParticleAffector { enum ForceType { Type_Wind, Type_Point }; public: std::string mSkelBaseName; Ogre::Bone* mBone; Ogre::ParticleSystem* getPartSys() { return mParent; } class CmdSkelBase : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { assert(false && "Unimplemented"); return ""; } void doSet(void *target, const Ogre::String &val) { GravityAffector* affector = static_cast(target); affector->mSkelBaseName = val; } }; class CmdBone : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { assert(false && "Unimplemented"); return ""; } void doSet(void *target, const Ogre::String &val) { GravityAffector* affector = static_cast(target); assert(!affector->mSkelBaseName.empty() && "Base entity needs to be set first"); Ogre::ParticleSystem* partsys = affector->getPartSys(); Ogre::Entity* ent = partsys->getParentSceneNode()->getCreator()->getEntity(affector->mSkelBaseName); Ogre::Bone* bone = ent->getSkeleton()->getBone(val); assert(bone); affector->mBone = bone; } }; /** Command object for force (see Ogre::ParamCommand).*/ class CmdForce : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const GravityAffector *self = static_cast(target); return Ogre::StringConverter::toString(self->getForce()); } void doSet(void *target, const Ogre::String &val) { GravityAffector *self = static_cast(target); self->setForce(Ogre::StringConverter::parseReal(val)); } }; /** Command object for force_type (see Ogre::ParamCommand).*/ class CmdForceType : public Ogre::ParamCommand { static ForceType getTypeFromString(const Ogre::String &type) { if(type == "wind") return Type_Wind; if(type == "point") return Type_Point; OGRE_EXCEPT(Ogre::Exception::ERR_INVALIDPARAMS, "Invalid force type string: "+type, "CmdForceType::getTypeFromString"); } static Ogre::String getStringFromType(ForceType type) { switch(type) { case Type_Wind: return "wind"; case Type_Point: return "point"; } OGRE_EXCEPT(Ogre::Exception::ERR_INVALIDPARAMS, "Invalid force type enum: "+Ogre::StringConverter::toString(type), "CmdForceType::getStringFromType"); } public: Ogre::String doGet(const void *target) const { const GravityAffector *self = static_cast(target); return getStringFromType(self->getForceType()); } void doSet(void *target, const Ogre::String &val) { GravityAffector *self = static_cast(target); self->setForceType(getTypeFromString(val)); } }; /** Command object for direction (see Ogre::ParamCommand).*/ class CmdDirection : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const GravityAffector *self = static_cast(target); return Ogre::StringConverter::toString(self->getDirection()); } void doSet(void *target, const Ogre::String &val) { GravityAffector *self = static_cast(target); self->setDirection(Ogre::StringConverter::parseVector3(val)); } }; /** Command object for position (see Ogre::ParamCommand).*/ class CmdPosition : public Ogre::ParamCommand { public: Ogre::String doGet(const void *target) const { const GravityAffector *self = static_cast(target); return Ogre::StringConverter::toString(self->getPosition()); } void doSet(void *target, const Ogre::String &val) { GravityAffector *self = static_cast(target); self->setPosition(Ogre::StringConverter::parseVector3(val)); } }; /** Default constructor. */ GravityAffector(Ogre::ParticleSystem *psys) : ParticleAffector(psys) , mForce(0.0f) , mForceType(Type_Wind) , mPosition(0.0f) , mDirection(0.0f) , mBone(NULL) { mType = "Gravity"; // Init parameters if(createParamDictionary("GravityAffector")) { Ogre::ParamDictionary *dict = getParamDictionary(); Ogre::String force_title("force"); Ogre::String force_descr("Amount of force applied to particles."); Ogre::String force_type_title("force_type"); Ogre::String force_type_descr("Type of force applied to particles (point or wind)."); Ogre::String direction_title("direction"); Ogre::String direction_descr("Direction of wind forces."); Ogre::String position_title("position"); Ogre::String position_descr("Position of point forces."); dict->addParameter(Ogre::ParameterDef(force_title, force_descr, Ogre::PT_REAL), &msForceCmd); dict->addParameter(Ogre::ParameterDef(force_type_title, force_type_descr, Ogre::PT_STRING), &msForceTypeCmd); dict->addParameter(Ogre::ParameterDef(direction_title, direction_descr, Ogre::PT_VECTOR3), &msDirectionCmd); dict->addParameter(Ogre::ParameterDef(position_title, position_descr, Ogre::PT_VECTOR3), &msPositionCmd); dict->addParameter(Ogre::ParameterDef("bone", "The bone where the particles should be spawned", Ogre::PT_STRING), &msBoneCmd); dict->addParameter(Ogre::ParameterDef("skelbase", "The name of the entity containing the bone (see 'bone' parameter)", Ogre::PT_STRING), &msSkelBaseCmd); } } /** See Ogre::ParticleAffector. */ void _affectParticles(Ogre::ParticleSystem *psys, Ogre::Real timeElapsed) { switch(mForceType) { case Type_Wind: applyWindForce(psys, timeElapsed); break; case Type_Point: applyPointForce(psys, timeElapsed); break; } } void setForce(Ogre::Real force) { mForce = force; } Ogre::Real getForce() const { return mForce; } void setForceType(ForceType type) { mForceType = type; } ForceType getForceType() const { return mForceType; } void setDirection(const Ogre::Vector3 &dir) { mDirection = dir; } const Ogre::Vector3 &getDirection() const { return mDirection; } void setPosition(const Ogre::Vector3 &pos) { mPosition = pos; } const Ogre::Vector3 &getPosition() const { return mPosition; } static CmdForce msForceCmd; static CmdForceType msForceTypeCmd; static CmdDirection msDirectionCmd; static CmdPosition msPositionCmd; static CmdBone msBoneCmd; static CmdSkelBase msSkelBaseCmd; protected: void applyWindForce(Ogre::ParticleSystem *psys, Ogre::Real timeElapsed) { const Ogre::Vector3 vec = mBone->_getDerivedOrientation() * mDirection * mForce * timeElapsed; Ogre::ParticleIterator pi = psys->_getIterator(); while (!pi.end()) { Ogre::Particle *p = pi.getNext(); #if OGRE_VERSION >= (1 << 16 | 10 << 8 | 0) p->mDirection += vec; #else p->direction += vec; #endif } } void applyPointForce(Ogre::ParticleSystem *psys, Ogre::Real timeElapsed) { const Ogre::Real force = mForce * timeElapsed; Ogre::ParticleIterator pi = psys->_getIterator(); while (!pi.end()) { Ogre::Particle *p = pi.getNext(); #if OGRE_VERSION >= (1 << 16 | 10 << 8 | 0) Ogre::Vector3 position = p->mPosition; #else Ogre::Vector3 position = p->position; #endif const Ogre::Vector3 vec = ( (mBone->_getDerivedOrientation() * mPosition + mBone->_getDerivedPosition()) - position).normalisedCopy() * force; #if OGRE_VERSION >= (1 << 16 | 10 << 8 | 0) p->mDirection += vec; #else p->direction += vec; #endif } } float mForce; ForceType mForceType; Ogre::Vector3 mPosition; Ogre::Vector3 mDirection; }; GravityAffector::CmdForce GravityAffector::msForceCmd; GravityAffector::CmdForceType GravityAffector::msForceTypeCmd; GravityAffector::CmdDirection GravityAffector::msDirectionCmd; GravityAffector::CmdPosition GravityAffector::msPositionCmd; GravityAffector::CmdBone GravityAffector::msBoneCmd; GravityAffector::CmdSkelBase GravityAffector::msSkelBaseCmd; Ogre::ParticleAffector *GravityAffectorFactory::createAffector(Ogre::ParticleSystem *psys) { Ogre::ParticleAffector *p = new GravityAffector(psys); mAffectors.push_back(p); return p; }