1
// Copyright (C) 2002-2011 Nikolaus Gebhardt
2
// This file is part of the "Irrlicht Engine".
3
// For conditions of distribution and use, see copyright notice in irrlicht.h
5
#include "CParticleBoxEmitter.h"
7
#include "IAttributes.h"
16
CParticleBoxEmitter::CParticleBoxEmitter(
17
const core::aabbox3df& box, const core::vector3df& direction,
18
u32 minParticlesPerSecond, u32 maxParticlesPerSecond,
19
video::SColor minStartColor, video::SColor maxStartColor,
20
u32 lifeTimeMin, u32 lifeTimeMax, s32 maxAngleDegrees,
21
const core::dimension2df& minStartSize, const core::dimension2df& maxStartSize)
22
: Box(box), Direction(direction),
23
MaxStartSize(maxStartSize), MinStartSize(minStartSize),
24
MinParticlesPerSecond(minParticlesPerSecond),
25
MaxParticlesPerSecond(maxParticlesPerSecond),
26
MinStartColor(minStartColor), MaxStartColor(maxStartColor),
27
MinLifeTime(lifeTimeMin), MaxLifeTime(lifeTimeMax),
28
Time(0), Emitted(0), MaxAngleDegrees(maxAngleDegrees)
31
setDebugName("CParticleBoxEmitter");
36
//! Prepares an array with new particles to emitt into the system
37
//! and returns how much new particles there are.
38
s32 CParticleBoxEmitter::emitt(u32 now, u32 timeSinceLastCall, SParticle*& outArray)
40
Time += timeSinceLastCall;
42
const u32 pps = (MaxParticlesPerSecond - MinParticlesPerSecond);
43
const f32 perSecond = pps ? ((f32)MinParticlesPerSecond + os::Randomizer::frand() * pps) : MinParticlesPerSecond;
44
const f32 everyWhatMillisecond = 1000.0f / perSecond;
46
if (Time > everyWhatMillisecond)
48
Particles.set_used(0);
49
u32 amount = (u32)((Time / everyWhatMillisecond) + 0.5f);
52
const core::vector3df& extent = Box.getExtent();
54
if (amount > MaxParticlesPerSecond*2)
55
amount = MaxParticlesPerSecond * 2;
57
for (u32 i=0; i<amount; ++i)
59
p.pos.X = Box.MinEdge.X + os::Randomizer::frand() * extent.X;
60
p.pos.Y = Box.MinEdge.Y + os::Randomizer::frand() * extent.Y;
61
p.pos.Z = Box.MinEdge.Z + os::Randomizer::frand() * extent.Z;
68
core::vector3df tgt = Direction;
69
tgt.rotateXYBy(os::Randomizer::frand() * MaxAngleDegrees);
70
tgt.rotateYZBy(os::Randomizer::frand() * MaxAngleDegrees);
71
tgt.rotateXZBy(os::Randomizer::frand() * MaxAngleDegrees);
75
p.endTime = now + MinLifeTime;
76
if (MaxLifeTime != MinLifeTime)
77
p.endTime += os::Randomizer::rand() % (MaxLifeTime - MinLifeTime);
79
if (MinStartColor==MaxStartColor)
80
p.color=MinStartColor;
82
p.color = MinStartColor.getInterpolated(MaxStartColor, os::Randomizer::frand());
84
p.startColor = p.color;
85
p.startVector = p.vector;
87
if (MinStartSize==MaxStartSize)
88
p.startSize = MinStartSize;
90
p.startSize = MinStartSize.getInterpolated(MaxStartSize, os::Randomizer::frand());
93
Particles.push_back(p);
96
outArray = Particles.pointer();
98
return Particles.size();
105
//! Writes attributes of the object.
106
void CParticleBoxEmitter::serializeAttributes(io::IAttributes* out, io::SAttributeReadWriteOptions* options) const
108
core::vector3df b = Box.getExtent();
110
out->addVector3d("Box", b);
111
out->addVector3d("Direction", Direction);
112
out->addFloat("MinStartSizeWidth", MinStartSize.Width);
113
out->addFloat("MinStartSizeHeight", MinStartSize.Height);
114
out->addFloat("MaxStartSizeWidth", MaxStartSize.Width);
115
out->addFloat("MaxStartSizeHeight", MaxStartSize.Height);
116
out->addInt("MinParticlesPerSecond", MinParticlesPerSecond);
117
out->addInt("MaxParticlesPerSecond", MaxParticlesPerSecond);
118
out->addColor("MinStartColor", MinStartColor);
119
out->addColor("MaxStartColor", MaxStartColor);
120
out->addInt("MinLifeTime", MinLifeTime);
121
out->addInt("MaxLifeTime", MaxLifeTime);
122
out->addInt("MaxAngleDegrees", MaxAngleDegrees);
126
//! Reads attributes of the object.
127
void CParticleBoxEmitter::deserializeAttributes(io::IAttributes* in, io::SAttributeReadWriteOptions* options)
129
// read data and correct input values here
131
core::vector3df b = in->getAttributeAsVector3d("Box");
140
Box.MinEdge.X = -b.X;
141
Box.MinEdge.Y = -b.Y;
142
Box.MinEdge.Z = -b.Z;
147
Direction = in->getAttributeAsVector3d("Direction");
148
if (Direction.getLength() == 0)
149
Direction.set(0,0.01f,0);
152
idx = in->findAttribute("MinStartSizeWidth");
154
MinStartSize.Width = in->getAttributeAsFloat(idx);
155
idx = in->findAttribute("MinStartSizeHeight");
157
MinStartSize.Height = in->getAttributeAsFloat(idx);
158
idx = in->findAttribute("MaxStartSizeWidth");
160
MaxStartSize.Width = in->getAttributeAsFloat(idx);
161
idx = in->findAttribute("MaxStartSizeHeight");
163
MaxStartSize.Height = in->getAttributeAsFloat(idx);
165
MinParticlesPerSecond = in->getAttributeAsInt("MinParticlesPerSecond");
166
MaxParticlesPerSecond = in->getAttributeAsInt("MaxParticlesPerSecond");
168
MinParticlesPerSecond = core::max_(1u, MinParticlesPerSecond);
169
MaxParticlesPerSecond = core::max_(MaxParticlesPerSecond, 1u);
170
MaxParticlesPerSecond = core::min_(MaxParticlesPerSecond, 200u);
171
MinParticlesPerSecond = core::min_(MinParticlesPerSecond, MaxParticlesPerSecond);
173
MinStartColor = in->getAttributeAsColor("MinStartColor");
174
MaxStartColor = in->getAttributeAsColor("MaxStartColor");
175
MinLifeTime = in->getAttributeAsInt("MinLifeTime");
176
MaxLifeTime = in->getAttributeAsInt("MaxLifeTime");
177
MaxAngleDegrees = in->getAttributeAsInt("MaxAngleDegrees");
179
MinLifeTime = core::max_(0u, MinLifeTime);
180
MaxLifeTime = core::max_(MaxLifeTime, MinLifeTime);
181
MinLifeTime = core::min_(MinLifeTime, MaxLifeTime);
186
} // end namespace scene
187
} // end namespace irr