irrlicht/source/Irrlicht/CParticleSphereEmitter.cpp

106 lines
3.2 KiB
C++

// Copyright (C) 2002-2008 Nikolaus Gebhardt
// This file is part of the "Irrlicht Engine".
// For conditions of distribution and use, see copyright notice in irrlicht.h
#include "IrrCompileConfig.h"
#include "CParticleSphereEmitter.h"
#include "os.h"
#include <math.h>
namespace irr
{
namespace scene
{
//! constructor
CParticleSphereEmitter::CParticleSphereEmitter(
const core::vector3df& center, f32 radius,
const core::vector3df& direction, u32 minParticlesPerSecond,
u32 maxParticlesPerSecond, const video::SColor& minStartColor,
const video::SColor& maxStartColor, u32 lifeTimeMin, u32 lifeTimeMax,
s32 maxAngleDegrees)
: Center(center), Radius(radius), Direction(direction), MinParticlesPerSecond(minParticlesPerSecond),
MaxParticlesPerSecond(maxParticlesPerSecond),
MinStartColor(minStartColor), MaxStartColor(maxStartColor),
MinLifeTime(lifeTimeMin), MaxLifeTime(lifeTimeMax), Time(0), Emitted(0),
MaxAngleDegrees(maxAngleDegrees)
{
#ifdef _DEBUG
setDebugName("CParticleSphereEmitter");
#endif
}
//! Prepares an array with new particles to emitt into the system
//! and returns how much new particles there are.
s32 CParticleSphereEmitter::emitt(u32 now, u32 timeSinceLastCall, SParticle*& outArray)
{
Time += timeSinceLastCall;
u32 pps = (MaxParticlesPerSecond - MinParticlesPerSecond);
f32 perSecond = pps ? (f32)MinParticlesPerSecond + (os::Randomizer::rand() % pps) : MinParticlesPerSecond;
f32 everyWhatMillisecond = 1000.0f / perSecond;
if(Time > everyWhatMillisecond)
{
Particles.set_used(0);
u32 amount = (u32)((Time / everyWhatMillisecond) + 0.5f);
Time = 0;
SParticle p;
if(amount > MaxParticlesPerSecond*2)
amount = MaxParticlesPerSecond * 2;
for(u32 i=0; i<amount; ++i)
{
// Random distance from center
f32 distance = fmodf( (f32)os::Randomizer::rand(), Radius * 1000.0f ) * 0.001f;
// Random direction from center
p.pos.X = Center.X + distance;
p.pos.Y = Center.Y + distance;
p.pos.Z = Center.Z + distance;
p.pos.rotateXYBy( os::Randomizer::rand() % 360, Center );
p.pos.rotateYZBy( os::Randomizer::rand() % 360, Center );
p.pos.rotateXZBy( os::Randomizer::rand() % 360, Center );
p.startTime = now;
p.vector = Direction;
if(MaxAngleDegrees)
{
core::vector3df tgt = Direction;
tgt.rotateXYBy((os::Randomizer::rand()%(MaxAngleDegrees*2)) - MaxAngleDegrees, core::vector3df());
tgt.rotateYZBy((os::Randomizer::rand()%(MaxAngleDegrees*2)) - MaxAngleDegrees, core::vector3df());
tgt.rotateXZBy((os::Randomizer::rand()%(MaxAngleDegrees*2)) - MaxAngleDegrees, core::vector3df());
p.vector = tgt;
}
if(MaxLifeTime - MinLifeTime == 0)
p.endTime = now + MinLifeTime;
else
p.endTime = now + MinLifeTime + (os::Randomizer::rand() % (MaxLifeTime - MinLifeTime));
p.color = MinStartColor.getInterpolated(
MaxStartColor, (os::Randomizer::rand() % 100) / 100.0f);
p.startColor = p.color;
p.startVector = p.vector;
Particles.push_back(p);
}
outArray = Particles.pointer();
return Particles.size();
}
return 0;
}
} // end namespace scene
} // end namespace irr