mirror of
https://github.com/OpenMW/openmw.git
synced 2025-10-24 15:56:36 +00:00
Note, I suspect Rng::rollClosedProbability() is not needed. The only difference between it and rollProbability() is that one time in 37k (on Windows), it will give an output of 1.0. On some versions of Linux, the value of 1.0 will occur about 1 time in 4 billion.
775 lines
27 KiB
C++
775 lines
27 KiB
C++
#include "particles.hpp"
|
|
|
|
#include <OgreStringConverter.h>
|
|
#include <OgreParticleSystem.h>
|
|
#include <OgreParticleEmitter.h>
|
|
#include <OgreParticleAffector.h>
|
|
#include <OgreParticle.h>
|
|
#include <OgreBone.h>
|
|
#include <OgreTagPoint.h>
|
|
#include <OgreEntity.h>
|
|
#include <OgreSkeletonInstance.h>
|
|
#include <OgreSceneNode.h>
|
|
#include <OgreSceneManager.h>
|
|
|
|
#include <openengine/misc/rng.hpp>
|
|
|
|
/* FIXME: "Nif" isn't really an appropriate emitter name. */
|
|
class NifEmitter : public Ogre::ParticleEmitter
|
|
{
|
|
public:
|
|
std::vector<Ogre::Bone*> mEmitterBones;
|
|
Ogre::Bone* mParticleBone;
|
|
|
|
Ogre::ParticleSystem* getPartSys() { return mParent; }
|
|
|
|
/** 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<const NifEmitter*>(target)->getWidth());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
static_cast<NifEmitter*>(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<const NifEmitter*>(target)->getHeight());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
static_cast<NifEmitter*>(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<const NifEmitter*>(target)->getDepth());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
static_cast<NifEmitter*>(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<const NifEmitter*>(target);
|
|
return Ogre::StringConverter::toString(self->getVerticalDirection().valueDegrees());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
NifEmitter *self = static_cast<NifEmitter*>(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<const NifEmitter*>(target);
|
|
return Ogre::StringConverter::toString(self->getVerticalAngle().valueDegrees());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
NifEmitter *self = static_cast<NifEmitter*>(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<const NifEmitter*>(target);
|
|
return Ogre::StringConverter::toString(self->getHorizontalDirection().valueDegrees());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
NifEmitter *self = static_cast<NifEmitter*>(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<const NifEmitter*>(target);
|
|
return Ogre::StringConverter::toString(self->getHorizontalAngle().valueDegrees());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
NifEmitter *self = static_cast<NifEmitter*>(target);
|
|
self->setHorizontalAngle(Ogre::Degree(Ogre::StringConverter::parseReal(val)));
|
|
}
|
|
};
|
|
|
|
|
|
NifEmitter(Ogre::ParticleSystem *psys)
|
|
: Ogre::ParticleEmitter(psys)
|
|
, mEmitterBones(Ogre::any_cast<NiNodeHolder>(psys->getUserObjectBindings().getUserAny()).mBones)
|
|
{
|
|
assert (!mEmitterBones.empty());
|
|
Ogre::TagPoint* tag = static_cast<Ogre::TagPoint*>(mParent->getParentNode());
|
|
mParticleBone = static_cast<Ogre::Bone*>(tag->getParent());
|
|
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)
|
|
{
|
|
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
|
|
|
|
Ogre::Node* emitterBone = mEmitterBones.at(OEngine::Misc::Rng::rollDice(mEmitterBones.size()));
|
|
|
|
position = xOff + yOff + zOff +
|
|
mParticleBone->_getDerivedOrientation().Inverse() * (emitterBone->_getDerivedPosition()
|
|
- mParticleBone->_getDerivedPosition());
|
|
|
|
// 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 = (mParticleBone->_getDerivedOrientation().Inverse()
|
|
* emitterBone->_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);
|
|
|
|
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;
|
|
};
|
|
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;
|
|
|
|
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<const GrowFadeAffector*>(target);
|
|
return Ogre::StringConverter::toString(self->getGrowTime());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
GrowFadeAffector *self = static_cast<GrowFadeAffector*>(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<const GrowFadeAffector*>(target);
|
|
return Ogre::StringConverter::toString(self->getFadeTime());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
GrowFadeAffector *self = static_cast<GrowFadeAffector*>(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;
|
|
assert (scale >= 0);
|
|
// HACK: don't allow zero-sized particles which can rarely cause an AABB assertion in Ogre to fail
|
|
scale = std::max(scale, 0.00001f);
|
|
width *= scale;
|
|
height *= scale;
|
|
}
|
|
if(particle_time < mFadeTime)
|
|
{
|
|
Ogre::Real scale = particle_time / mFadeTime;
|
|
assert (scale >= 0);
|
|
// HACK: don't allow zero-sized particles which can rarely cause an AABB assertion in Ogre to fail
|
|
scale = std::max(scale, 0.00001f);
|
|
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;
|
|
assert (scale >= 0);
|
|
// HACK: don't allow zero-sized particles which can rarely cause an AABB assertion in Ogre to fail
|
|
scale = std::max(scale, 0.00001f);
|
|
width *= scale;
|
|
height *= scale;
|
|
}
|
|
if(particle_time < mFadeTime)
|
|
{
|
|
Ogre::Real scale = particle_time / mFadeTime;
|
|
assert (scale >= 0);
|
|
// HACK: don't allow zero-sized particles which can rarely cause an AABB assertion in Ogre to fail
|
|
scale = std::max(scale, 0.00001f);
|
|
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:
|
|
Ogre::Bone* mEmitterBone;
|
|
Ogre::Bone* mParticleBone;
|
|
|
|
Ogre::ParticleSystem* getPartSys() { return mParent; }
|
|
|
|
/** 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<const GravityAffector*>(target);
|
|
return Ogre::StringConverter::toString(self->getForce());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
GravityAffector *self = static_cast<GravityAffector*>(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<const GravityAffector*>(target);
|
|
return getStringFromType(self->getForceType());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
GravityAffector *self = static_cast<GravityAffector*>(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<const GravityAffector*>(target);
|
|
return Ogre::StringConverter::toString(self->getDirection());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
GravityAffector *self = static_cast<GravityAffector*>(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<const GravityAffector*>(target);
|
|
return Ogre::StringConverter::toString(self->getPosition());
|
|
}
|
|
void doSet(void *target, const Ogre::String &val)
|
|
{
|
|
GravityAffector *self = static_cast<GravityAffector*>(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)
|
|
{
|
|
std::vector<Ogre::Bone*> bones = Ogre::any_cast<NiNodeHolder>(psys->getUserObjectBindings().getUserAny()).mBones;
|
|
assert (!bones.empty());
|
|
mEmitterBone = bones[0];
|
|
Ogre::TagPoint* tag = static_cast<Ogre::TagPoint*>(mParent->getParentNode());
|
|
mParticleBone = static_cast<Ogre::Bone*>(tag->getParent());
|
|
|
|
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);
|
|
}
|
|
}
|
|
|
|
/** 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;
|
|
|
|
protected:
|
|
void applyWindForce(Ogre::ParticleSystem *psys, Ogre::Real timeElapsed)
|
|
{
|
|
const Ogre::Vector3 vec = 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
|
|
|
|
Ogre::Vector3 vec = (mPosition - 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;
|
|
|
|
Ogre::ParticleAffector *GravityAffectorFactory::createAffector(Ogre::ParticleSystem *psys)
|
|
{
|
|
Ogre::ParticleAffector *p = new GravityAffector(psys);
|
|
mAffectors.push_back(p);
|
|
return p;
|
|
}
|