TDME2 1.9.121
ObjectParticleSystemInternal.cpp
Go to the documentation of this file.
2
3#include <algorithm>
4#include <string>
5#include <vector>
6
7#include <tdme/tdme.h>
16#include <tdme/engine/Engine.h>
17#include <tdme/engine/Entity.h>
19#include <tdme/engine/Timing.h>
21#include <tdme/math/Math.h>
22#include <tdme/math/Matrix4x4.h>
23#include <tdme/math/Vector3.h>
24
25using std::remove;
26using std::string;
27using std::to_string;
28using std::vector;
29
47
48ObjectParticleSystemInternal::ObjectParticleSystemInternal(const string& id, Model* model, const Vector3& scale, bool autoEmit, bool contributesShadows, bool receivesShadows, int32_t maxCount, ParticleEmitter* emitter)
49{
50 this->id = id;
51 this->enabled = true;
52 this->model = model;
53 this->objectScale = scale;
54 this->autoEmit = autoEmit;
55 this->contributesShadows = contributesShadows;
56 this->receivesShadows = receivesShadows;
57 particles.resize(maxCount);
58 objects.resize(maxCount);
59 for (auto i = 0; i < objects.size(); i++) {
60 objects[i] = new Object3D(
61 "tdme.opse." +
62 id +
63 ":" +
64 to_string(i),
65 model
66 );
67 objects[i]->setEnabled(false);
68 objects[i]->setScale(objectScale);
69 objects[i]->setContributesShadows(contributesShadows);
70 objects[i]->setReceivesShadows(receivesShadows);
71 objects[i]->setPickable(false);
72 }
73 this->boundingBox = new BoundingBox();
75 this->emitter = emitter;
76 this->effectColorMul.set(1.0f, 1.0f, 1.0f, 1.0f);
77 this->effectColorAdd.set(0.0f, 0.0f, 0.0f, 0.0f);
78 this->pickable = false;
79 this->particlesToSpawnRemainder = 0.0f;
80}
81
83 delete emitter;
84 for (auto i = 0; i < objects.size(); i++) delete objects[i];
85}
86
88{
89 return id;
90}
91
93{
94 this->engine = engine;
95 for (auto i = 0; i < objects.size(); i++) {
96 objects[i]->setEngine(engine);
97 }
98}
99
101{
102 this->renderer = renderer;
103 for (auto i = 0; i < objects.size(); i++) {
104 objects[i]->setRenderer(renderer);
105 }
106}
107
109{
112}
113
115{
118}
119
121{
122 // particles to spawn
123 auto particlesToSpawn = 0;
124 {
125 auto particlesToSpawnWithFraction = emitter->getCount() * engine->getTiming()->getDeltaTime() / 1000.0f;
126 if (autoEmit == true) {
127 particlesToSpawn = static_cast< int32_t >(particlesToSpawnWithFraction);
128 particlesToSpawnRemainder += particlesToSpawnWithFraction - particlesToSpawn;
129 if (particlesToSpawnRemainder > 1.0f) {
130 particlesToSpawn++;
132 }
133 } else {
134 particlesToSpawn = emitter->getCount();
135 }
136 }
137 // skip if nothing to spawn
138 if (particlesToSpawn == 0) return 0;
139 // spawn
140 auto particlesSpawned = 0;
141 for (auto i = 0; i < particles.size(); i++) {
142 auto& particle = particles[i];
143 if (particle.active == true) continue;
144 // emit particle
145 emitter->emit(&particle);
146 // enable object
147 auto object = objects[i];
148 object->setTranslation(particle.position);
149 object->update();
150 object->setEnabled(true);
151 object->setEffectColorAdd(effectColorAdd);
152 object->setEffectColorMul(effectColorMul);
153 enabledObjects.push_back(object);
154 // all particles spawned?
155 particlesSpawned++;
156 if (particlesSpawned == particlesToSpawn) break;
157 }
158 // done
159 return particlesSpawned;
160}
161
163{
164 //
165 Vector3 center;
166 auto& localTransformationsMatrix = localTransformations.getTransformationsMatrix();
167 localTransformationsMatrix.getTranslation(center);
168 center.add(emitter->getCenter());
169 // transformations
171 //
172 Vector3 point;
173 Vector3 velocityForTime;
174 auto first = true;
175 auto& bbMinXYZ = boundingBoxTransformed.getMin().getArray();
176 auto& bbMaxXYZ = boundingBoxTransformed.getMax().getArray();
177 auto timeDelta = engine->getTiming()->getDeltaTime();
178 for (auto i = 0; i < particles.size(); i++) {
179 auto& particle = particles[i];
180 if (particle.active == false)
181 continue;
182
183 auto object = objects[i];
184 // life time
185 particle.lifeTimeCurrent += timeDelta;
186 if (particle.lifeTimeCurrent >= particle.lifeTimeMax) {
187 particle.active = false;
188 object->setEnabled(false);
189 enabledObjects.erase(remove(enabledObjects.begin(), enabledObjects.end(), object), enabledObjects.end());
190 continue;
191 }
192 // add gravity if our particle have a noticable mass
193 if (particle.mass > Math::EPSILON)
194 particle.velocity.sub(Vector3(0.0f, 0.5f * Math::g * static_cast<float>(timeDelta) / 1000.0f, 0.0f));
195 // TODO:
196 // maybe take air resistance into account like a huge paper needs more time to fall than a sphere of paper
197 // or heat for smoke or fire, whereas having no mass for those particles works around this problem for now
198 // translation
199 particle.position.add(velocityForTime.set(particle.velocity).scale(static_cast<float>(timeDelta) / 1000.0f));
200 // update up effect colors
201 object->setEffectColorAdd(effectColorAdd);
202 object->setEffectColorMul(effectColorMul);
203 // translation
204 point = localTransformationsMatrix.multiply(particle.position);
205 point.add(center);
206 // transform particle according to its transformations
207 point = transformationsMatrix.multiply(point);
208 // apply to object
209 object->setTranslation(point);
210 object->update();
211 if (first == true) {
212 boundingBoxTransformed.getMin().set(object->getBoundingBoxTransformed()->getMin());
213 boundingBoxTransformed.getMax().set(object->getBoundingBoxTransformed()->getMax());
214 first = false;
215 } else {
216 auto& objBbMinXYZ = object->getBoundingBoxTransformed()->getMin().getArray();
217 auto& objBbMaxXYZ = object->getBoundingBoxTransformed()->getMax().getArray();
218 if (objBbMinXYZ[0] < bbMinXYZ[0]) bbMinXYZ[0] = objBbMinXYZ[0];
219 if (objBbMinXYZ[1] < bbMinXYZ[1]) bbMinXYZ[1] = objBbMinXYZ[1];
220 if (objBbMinXYZ[2] < bbMinXYZ[2]) bbMinXYZ[2] = objBbMinXYZ[2];
221 if (objBbMaxXYZ[0] > bbMaxXYZ[0]) bbMaxXYZ[0] = objBbMaxXYZ[0];
222 if (objBbMaxXYZ[1] > bbMaxXYZ[1]) bbMaxXYZ[1] = objBbMaxXYZ[1];
223 if (objBbMaxXYZ[2] > bbMaxXYZ[2]) bbMaxXYZ[2] = objBbMaxXYZ[2];
224 }
225 }
226 // compute bounding boxes
229}
230
232{
233 for (auto i = 0; i < objects.size(); i++) {
234 objects[i]->dispose();
235 }
236}
237
Engine main class.
Definition: Engine.h:122
Timing * getTiming()
Definition: Engine.h:900
TDME engine entity.
Definition: Entity.h:31
Object 3D to be used with engine class.
Definition: Object3D.h:60
Timing class.
Definition: Timing.h:17
int64_t getDeltaTime()
Gets the time passed between last and current frame.
Definition: Timing.h:83
Transformations which contain scale, rotations and translation.
const Matrix4x4 & getTransformationsMatrix() const
virtual void fromTransformations(const Transformations &transformations)
Set up this transformations from given transformations.
virtual void update()
Computes transformation matrix.
Color 4 base definition class.
Definition: Color4Base.h:19
void set(const array< float, 4 > &color)
Set up color.
Definition: Color4Base.h:68
Color 4 definition.
Definition: Color4.h:20
Representation of a 3d model.
Definition: Model.h:32
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
void fromBoundingVolumeWithTransformations(BoundingBox *original, const Transformations &transformations)
Create bounding volume from given original(of same type) with applied transformations.
Definition: BoundingBox.cpp:79
void update()
Updates this bounding box.
void fromTransformations(const Transformations &transformations) override
Set up this transformations from given transformations.
int32_t emitParticles() override
Adds particles to this particle entity at given position.
Standard math functions.
Definition: Math.h:21
4x4 3D Matrix class
Definition: Matrix4x4.h:24
void getTranslation(Vector3 &translation) const
Get translation.
Definition: Matrix4x4.h:261
Vector3 multiply(const Vector3 &v) const
Multiplies a vector3 with this matrix into destination vector.
Definition: Matrix4x4.h:351
3D vector 3 class
Definition: Vector3.h:22
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Vector3 & add(const Vector3 &v)
Adds a vector.
Definition: Vector3.h:301
Vector3 & scale(float scale)
Scale this vector.
Definition: Vector3.h:349
array< float, 3 > & getArray() const
Definition: Vector3.h:171
virtual void emit(Particle *particle)=0
Emits particles.