TDME2 1.9.121
Body.cpp
Go to the documentation of this file.
1#include <string>
2#include <vector>
3
4#include <ext/reactphysics3d/src/body/Body.h>
5#include <ext/reactphysics3d/src/body/CollisionBody.h>
6#include <ext/reactphysics3d/src/collision/narrowphase/DefaultCollisionDispatch.h>
7#include <ext/reactphysics3d/src/collision/narrowphase/NarrowPhaseAlgorithm.h>
8#include <ext/reactphysics3d/src/collision/shapes/CollisionShape.h>
9#include <ext/reactphysics3d/src/collision/CollisionCallback.h>
10#include <ext/reactphysics3d/src/collision/NarrowPhaseInfo.h>
11#include <ext/reactphysics3d/src/collision/ProxyShape.h>
12#include <ext/reactphysics3d/src/constraint/ContactPoint.h>
13#include <ext/reactphysics3d/src/engine/CollisionWorld.h>
14#include <ext/reactphysics3d/src/engine/DynamicsWorld.h>
15#include <ext/reactphysics3d/src/mathematics/Transform.h>
16#include <ext/reactphysics3d/src/mathematics/Vector3.h>
17#include <ext/reactphysics3d/src/memory/MemoryAllocator.h>
18
19#include <tdme/tdme.h>
31#include <tdme/math/Math.h>
32#include <tdme/math/Matrix4x4.h>
34#include <tdme/math/Vector3.h>
36
37using std::string;
38using std::to_string;
39using std::vector;
40
57
58constexpr uint16_t Body::TYPEIDS_ALL;
59constexpr uint16_t Body::TYPEID_STATIC;
60constexpr uint16_t Body::TYPEID_DYNAMIC;
61
62Body::Body(World* world, const string& id, int type, bool enabled, uint16_t collisionTypeId, const Transformations& transformations, float restitution, float friction, float mass, const Vector3& inertiaTensor, const vector<BoundingVolume*> boundingVolumes)
63{
64 this->world = world;
65 this->id = id;
66 this->rootId = id;
67 this->inertiaTensor = inertiaTensor;
68 this->type = type;
69 this->mass = mass;
70 this->collideTypeIds = ~0;
71 this->collisionTypeId = collisionTypeId;
72 switch (type) {
73 case TYPE_STATIC:
74 this->rigidBody = this->world->world.createRigidBody(reactphysics3d::Transform());
75 this->rigidBody->setType(reactphysics3d::BodyType::STATIC);
77 break;
78 case TYPE_DYNAMIC:
79 this->rigidBody = this->world->world.createRigidBody(reactphysics3d::Transform());
80 this->rigidBody->setType(reactphysics3d::BodyType::DYNAMIC);
82 break;
83 case TYPE_KINEMATIC:
84 this->rigidBody = this->world->world.createRigidBody(reactphysics3d::Transform());
85 this->rigidBody->setType(reactphysics3d::BodyType::KINEMATIC);
87 break;
88 case TYPE_COLLISION:
89 this->rigidBody = nullptr;
90 this->collisionBody = this->world->world.createCollisionBody(reactphysics3d::Transform());
91 break;
92 default:
93 Console::println("Body::Body(): unsupported type: " + to_string(type) + ": using collision body");
94 this->rigidBody = nullptr;
95 this->collisionBody = this->world->world.createCollisionBody(reactphysics3d::Transform());
96 break;
97 }
98 if (rigidBody != nullptr) {
99 rigidBody->getMaterial().setFrictionCoefficient(friction);
100 rigidBody->getMaterial().setBounciness(restitution);
101 rigidBody->setMass(mass);
102 }
103 collisionBody->setUserData(this);
104 for (auto boundingVolume: boundingVolumes) {
105 this->boundingVolumes.push_back(dynamic_cast<TerrainMesh*>(boundingVolume) != nullptr?boundingVolume:boundingVolume->clone());
106 }
108 setEnabled(enabled);
109}
110
112 for (auto boundingVolume: boundingVolumes) {
113 if (dynamic_cast<TerrainMesh*>(boundingVolume) != nullptr && cloned == true) continue;
114 delete boundingVolume;
115 }
116}
117
119 return Vector3(0.0f, 0.0f, 0.0f);
120}
121
122Matrix4x4 Body::computeInverseInertiaMatrix(BoundingBox* boundingBox, float mass, float scaleXAxis, float scaleYAxis, float scaleZAxis)
123{
124 auto width = boundingBox->getDimensions().getX();
125 auto height = boundingBox->getDimensions().getY();
126 auto depth = boundingBox->getDimensions().getZ();
127 return
128 (Matrix4x4(
129 scaleXAxis > Math::EPSILON && mass > Math::EPSILON?1.0f / (scaleXAxis * 1.0f / 12.0f * mass * (height * height + depth * depth)):0.0f,
130 0.0f,
131 0.0f,
132 0.0f,
133 0.0f,
134 scaleYAxis > Math::EPSILON && mass > Math::EPSILON?1.0f / (scaleYAxis * 1.0f / 12.0f * mass * (width * width + depth * depth)):0.0f,
135 0.0f,
136 0.0f,
137 0.0f,
138 0.0f,
139 scaleZAxis > Math::EPSILON && mass > Math::EPSILON?1.0f / (scaleZAxis * 1.0f / 12.0f * mass * (width * width + height * height)):0.0f,
140 0.0f,
141 0.0f,
142 0.0f,
143 0.0f,
144 1.0f
145 ));
146}
147
149 return cloned;
150}
151
152void Body::setCloned(bool cloned) {
153 this->cloned = cloned;
154}
155
156const string& Body::getId()
157{
158 return id;
159}
160
161const string& Body::getRootId()
162{
163 return rootId;
164}
165
166void Body::setRootId(const string& rootId) {
167 this->rootId = rootId;
168}
169
170int32_t Body::getType() {
171 return type;
172}
173
175{
176 return collisionTypeId;
177}
178
179void Body::setCollisionTypeId(uint16_t typeId)
180{
181 this->collisionTypeId = typeId;
182 for (auto proxyShape: proxyShapes) {
183 proxyShape->setCollisionCategoryBits(typeId);
184 }
185}
186
188{
189 return collideTypeIds;
190}
191
192void Body::setCollisionTypeIds(uint16_t collisionTypeIds)
193{
194 this->collideTypeIds = collisionTypeIds;
195 for (auto proxyShape: proxyShapes) {
196 proxyShape->setCollideWithMaskBits(collisionTypeIds);
197 }
198}
199
201{
202 return collisionBody->isActive();
203}
204
205void Body::setEnabled(bool enabled)
206{
207 collisionBody->setIsActive(enabled);
208 if (enabled == true) collisionBody->setIsSleeping(false);
209}
210
212{
213 return collisionBody->isSleeping();
214}
215
216void Body::setSleeping(bool sleeping) {
217 collisionBody->setIsSleeping(sleeping);
218}
219
220vector<BoundingVolume*>& Body::getBoundingVolumes() {
221 return boundingVolumes;
222}
223
225 // remove proxy shapes
226 for (auto proxyShape: proxyShapes) {
227 if (rigidBody != nullptr) {
228 rigidBody->removeCollisionShape(proxyShape);
229 } else {
230 collisionBody->removeCollisionShape(proxyShape);
231 }
232 }
233 proxyShapes.clear();
234
235 // set up scale
236 for (auto boundingVolume: boundingVolumes) {
237 // scale bounding volume and recreate it if nessessary
238 if (boundingVolume->getScale().equals(transformations.getScale()) == false) {
239 boundingVolume->setScale(transformations.getScale());
240 }
241 }
242
243 // reset proxy shapes with mass
244 if (mass > Math::EPSILON) {
245 // determine total volume
246 float volumeTotal = 0.0f;
247 for (auto boundingVolume: boundingVolumes) {
248 volumeTotal+=
249 boundingVolume->boundingBoxTransformed.getDimensions().getX() *
250 boundingVolume->boundingBoxTransformed.getDimensions().getY() *
251 boundingVolume->boundingBoxTransformed.getDimensions().getZ();
252 }
253 // add bounding volumes with mass
254 for (auto boundingVolume: boundingVolumes) {
255 reactphysics3d::ProxyShape* proxyShape = nullptr;
256 float volumeBoundingVolume =
257 boundingVolume->boundingBoxTransformed.getDimensions().getX() *
258 boundingVolume->boundingBoxTransformed.getDimensions().getY() *
259 boundingVolume->boundingBoxTransformed.getDimensions().getZ();
260 if (rigidBody != nullptr) {
261 proxyShape = rigidBody->addCollisionShape(boundingVolume->collisionShape, boundingVolume->collisionShapeLocalTransform, mass / volumeTotal * volumeBoundingVolume);
262 } else {
263 proxyShape = collisionBody->addCollisionShape(boundingVolume->collisionShape, boundingVolume->collisionShapeLocalTransform);
264 }
265 proxyShapes.push_back(proxyShape);
266 }
267 } else {
268 // add bounding volumes without mass
269 for (auto boundingVolume: boundingVolumes) {
270 reactphysics3d::ProxyShape* proxyShape = nullptr;
271 if (rigidBody != nullptr) {
272 proxyShape = rigidBody->addCollisionShape(boundingVolume->collisionShape, boundingVolume->collisionShapeLocalTransform, 0.0f);
273 } else {
274 proxyShape = collisionBody->addCollisionShape(boundingVolume->collisionShape, boundingVolume->collisionShapeLocalTransform);
275 }
276 proxyShapes.push_back(proxyShape);
277 }
278 }
279
280 // set up collisioin type ids and type id
281 for (auto proxyShape: proxyShapes) {
282 proxyShape->setCollideWithMaskBits(collideTypeIds);
283 proxyShape->setCollisionCategoryBits(collisionTypeId);
284 }
285
286 // set up inverse inertia tensor local
287 if (rigidBody != nullptr) {
288 // set inverse inertia tensor local
289 // TODO: transform with identity transform, this is a bit sub optimal but currently required to get rigid body dimensions
290 // this method is only used when setting TDME2 transform, so no need to transform back
291 reactphysics3d::Transform transform;
292 collisionBody->setTransform(transform);
293
294 auto boundingBoxTransformed = computeBoundingBoxTransformed();
295 auto& inverseInertiaMatrixArray = computeInverseInertiaMatrix(&boundingBoxTransformed, mass, inertiaTensor.getX(), inertiaTensor.getY(), inertiaTensor.getZ()).getArray();
296 rigidBody->setInverseInertiaTensorLocal(
297 reactphysics3d::Matrix3x3(
298 inverseInertiaMatrixArray[0],
299 inverseInertiaMatrixArray[1],
300 inverseInertiaMatrixArray[2],
301 inverseInertiaMatrixArray[4],
302 inverseInertiaMatrixArray[5],
303 inverseInertiaMatrixArray[6],
304 inverseInertiaMatrixArray[8],
305 inverseInertiaMatrixArray[9],
306 inverseInertiaMatrixArray[10]
307 )
308 );
309 };
310}
311
313 auto aabb = rigidBody->getAABB();
314 return BoundingBox(
315 Vector3(
316 aabb.getMin().x,
317 aabb.getMin().y,
318 aabb.getMin().z
319 ),
320 Vector3(
321 aabb.getMax().x,
322 aabb.getMax().y,
323 aabb.getMax().z
324 )
325 );
326}
327
329{
330 if (rigidBody == nullptr) {
331 Console::println("Body::getFriction(): no rigid body attached");
332 return 0.0f;
333 }
334 return rigidBody->getMaterial().getFrictionCoefficient();
335}
336
337void Body::setFriction(float friction)
338{
339 if (rigidBody == nullptr) {
340 Console::println("Body::setFriction(): no rigid body attached");
341 return;
342 }
343 rigidBody->getMaterial().setFrictionCoefficient(friction);
344}
345
347{
348 if (rigidBody == nullptr) {
349 Console::println("Body::getRestitution(): no rigid body attached");
350 return 0.0f;
351 }
352 return rigidBody->getMaterial().getBounciness();
353}
354
355void Body::setRestitution(float restitution)
356{
357 if (rigidBody == nullptr) {
358 Console::println("Body::setRestitution(): no rigid body attached");
359 return;
360 }
361 rigidBody->getMaterial().setBounciness(restitution);
362}
363
365{
366 return mass;
367}
368
369void Body::setMass(float mass)
370{
371 this->mass = mass;
372 if (rigidBody == nullptr) {
373 Console::println("Body::setMass(): no rigid body attached");
374 return;
375 }
376 rigidBody->setMass(mass);
377}
378
380{
381 if (rigidBody == nullptr) {
382 Console::println("Body::getLinearVelocity(): no rigid body attached");
383 return Vector3();
384 }
385 return Vector3(
386 rigidBody->getLinearVelocity().x,
387 rigidBody->getLinearVelocity().y,
388 rigidBody->getLinearVelocity().z
389 );
390}
391
392void Body::setLinearVelocity(const Vector3& linearVelocity) {
393 if (rigidBody == nullptr) {
394 Console::println("Body::setLinearVelocity(): no rigid body attached");
395 return;
396 }
397 rigidBody->setLinearVelocity(reactphysics3d::Vector3(linearVelocity.getX(), linearVelocity.getY(), linearVelocity.getZ()));
398}
399
401{
402 if (rigidBody == nullptr) {
403 Console::println("Body::getAngularVelocity(): no rigid body attached");
404 return Vector3();
405 }
406 return Vector3(
407 rigidBody->getAngularVelocity().x,
408 rigidBody->getAngularVelocity().y,
409 rigidBody->getAngularVelocity().z
410 );
411}
412
413void Body::setAngularVelocity(const Vector3& angularVelocity) {
414 if (rigidBody == nullptr) {
415 Console::println("Body::setAngularVelocity(): no rigid body attached");
416 return;
417 }
418 rigidBody->setAngularVelocity(reactphysics3d::Vector3(angularVelocity.getX(), angularVelocity.getY(), angularVelocity.getZ()));
419}
420
422{
423 if (rigidBody == nullptr) {
424 Console::println("Body::getLinearDamping(): no rigid body attached");
425 return 0.0f;
426 }
427 return rigidBody->getLinearDamping();
428}
429
430void Body::setLinearDamping(float linearDamping)
431{
432 if (rigidBody == nullptr) {
433 Console::println("Body::setLinearDamping(): no rigid body attached");
434 return;
435 }
436 rigidBody->setLinearDamping(linearDamping);
437}
438
440{
441 if (rigidBody == nullptr) {
442 Console::println("Body::getAngularDamping(): no rigid body attached");
443 return 0.0f;
444 }
445 return rigidBody->getAngularDamping();
446}
447
448void Body::setAngularDamping(float angularDamping)
449{
450 if (rigidBody == nullptr) {
451 Console::println("Body::setAngularDamping(): no rigid body attached");
452 return;
453 }
454 rigidBody->setAngularDamping(angularDamping);
455}
456
458 return transformations;
459}
460
461void Body::fromTransformations(const Transformations& transformations)
462{
463 // store engine transformations
464 this->transformations.fromTransformations(transformations);
465
466 // reset proxy shapes if bounding volumes do not match proxy shapes or if scaling has changed
470 }
471
472 /*
473 // TODO: center of mass ~ pivot
474 // set center of mass which is basically center of bv for now
475 body->setCenterOfMassLocal(boundingVolume->collisionShapeLocalTransform.getPosition());
476 // find final position, not sure yet if its working 100%, but still works with some tests
477 auto centerOfMassWorld = transform * boundingVolume->collisionShapeLocalTransform.getPosition();
478 transform.setPosition(
479 transform.getPosition() +
480 transform.getPosition() -
481 centerOfMassWorld +
482 (
483 reactphysics3d::Vector3(
484 boundingVolume->collisionShapeLocalTranslation.getX(),
485 boundingVolume->collisionShapeLocalTranslation.getY(),
486 boundingVolume->collisionShapeLocalTranslation.getZ()
487 ) * scaleVectorTransformed
488 )
489 );
490 */
491
492 // set transform
493 collisionBody->setTransform(
494 reactphysics3d::Transform(
495 reactphysics3d::Vector3(
496 this->transformations.getTranslation().getX(),
497 this->transformations.getTranslation().getY(),
498 this->transformations.getTranslation().getZ()
499 ),
500 reactphysics3d::Quaternion(
501 this->transformations.getRotationsQuaternion().getX(),
502 this->transformations.getRotationsQuaternion().getY(),
503 this->transformations.getRotationsQuaternion().getZ(),
504 this->transformations.getRotationsQuaternion().getW()
505 )
506 )
507 );
508}
509
510void Body::addForce(const Vector3& forceOrigin, const Vector3& force)
511{
512 if (rigidBody == nullptr) {
513 Console::println("Body::addForce(): no rigid body attached");
514 return;
515 }
516 rigidBody->applyForce(
517 reactphysics3d::Vector3(force.getX(), force.getY(), force.getZ()),
518 reactphysics3d::Vector3(forceOrigin.getX(), forceOrigin.getY(), forceOrigin.getZ())
519 );
520}
521
522void Body::addForce(const Vector3& force)
523{
524 if (rigidBody == nullptr) {
525 Console::println("Body::addForce(): no rigid body attached");
526 return;
527 }
528 rigidBody->applyForceToCenterOfMass(
529 reactphysics3d::Vector3(force.getX(), force.getY(), force.getZ())
530 );
531}
532
533void Body::addTorque(const Vector3& torque)
534{
535 if (rigidBody == nullptr) {
536 Console::println("Body::addTorque(): no rigid body attached");
537 return;
538 }
539 rigidBody->applyTorque(
540 reactphysics3d::Vector3(torque.getX(), torque.getY(), torque.getZ())
541 );
542}
543
545{
546 collisionListener.push_back(listener);
547}
548
550{
551 collisionListener.erase(remove(collisionListener.begin(), collisionListener.end(), listener), collisionListener.end());
552}
553
554void Body::fireOnCollision(Body* other, CollisionResponse& collisionResponse)
555{
556 for (auto listener: collisionListener) {
557 listener->onCollision(this, other, collisionResponse);
558 }
559}
560
561void Body::fireOnCollisionBegin(Body* other, CollisionResponse& collisionResponse)
562{
563 for (auto listener: collisionListener) {
564 listener->onCollisionBegin(this, other, collisionResponse);
565 }
566}
567
569{
570 for (auto listener: collisionListener) {
571 listener->onCollisionEnd(this, other);
572 }
573}
Rotation representation.
Definition: Rotation.h:18
Transformations which contain scale, rotations and translation.
const Vector3 & getScale() const
virtual void fromTransformations(const Transformations &transformations)
Set up this transformations from given transformations.
const Vector3 & getTranslation() const
Dynamic rigid/static rigid/collision body class.
Definition: Body.h:43
vector< BoundingVolume * > boundingVolumes
Definition: Body.h:75
uint16_t getCollisionTypeIds()
Definition: Body.cpp:187
void setCloned(bool cloned)
Set cloned.
Definition: Body.cpp:152
vector< CollisionListener * > collisionListener
Definition: Body.h:77
void addTorque(const Vector3 &torque)
Add torque.
Definition: Body.cpp:533
reactphysics3d::RigidBody * rigidBody
Definition: Body.h:63
void fireOnCollisionBegin(Body *other, CollisionResponse &collisionResponse)
Fire on collision begin.
Definition: Body.cpp:561
void setMass(float mass)
Set up mass.
Definition: Body.cpp:369
void resetProxyShapes()
Reset proxy shapes.
Definition: Body.cpp:224
void setAngularVelocity(const Vector3 &angularVelocity)
Set angular velocity.
Definition: Body.cpp:413
vector< BoundingVolume * > & getBoundingVolumes()
Definition: Body.cpp:220
Transformations transformations
Definition: Body.h:73
BoundingBox computeBoundingBoxTransformed()
Compute bounding box transformed.
Definition: Body.cpp:312
void setEnabled(bool enabled)
Set up if rigid body is enabled.
Definition: Body.cpp:205
static Matrix4x4 computeInverseInertiaMatrix(BoundingBox *boundingBox, float mass, float scaleXAxis, float scaleYAxis, float scaleZAxis)
Computes the inverse inertia matrix.
Definition: Body.cpp:122
void setLinearVelocity(const Vector3 &linearVelocity)
Set linear velocity.
Definition: Body.cpp:392
void setFriction(float friction)
Set up friction.
Definition: Body.cpp:337
void setRestitution(float restitution)
Set up restitution.
Definition: Body.cpp:355
void setSleeping(bool sleeping)
Set sleeping.
Definition: Body.cpp:216
uint16_t collideTypeIds
Definition: Body.h:70
reactphysics3d::CollisionBody * collisionBody
Definition: Body.h:64
void setCollisionTypeId(uint16_t typeId)
Set collision type id.
Definition: Body.cpp:179
void addForce(const Vector3 &forceOrigin, const Vector3 &force)
Add force.
Definition: Body.cpp:510
static constexpr int32_t TYPE_DYNAMIC
Definition: Body.h:48
int getType()
Return type, see TYPE_*.
Definition: Body.cpp:170
void setLinearDamping(float linearDamping)
Set linear damping.
Definition: Body.cpp:430
void removeCollisionListener(CollisionListener *listener)
Remove a collision listener to this rigid body.
Definition: Body.cpp:549
void fireOnCollisionEnd(Body *other)
Fire on collision end.
Definition: Body.cpp:568
void setAngularDamping(float angularDamping)
Set angular damping.
Definition: Body.cpp:448
static constexpr int32_t TYPE_KINEMATIC
Definition: Body.h:49
const Transformations & getTransformations()
Definition: Body.cpp:457
const Vector3 getAngularVelocity()
Definition: Body.cpp:400
const string & getRootId()
Definition: Body.cpp:161
vector< reactphysics3d::ProxyShape * > proxyShapes
Definition: Body.h:76
static const Vector3 getNoRotationInertiaTensor()
Definition: Body.cpp:118
void setCollisionTypeIds(uint16_t collisionTypeIds)
Set up collision type ids.
Definition: Body.cpp:192
Vector3 transformationsScale
Definition: Body.h:74
uint16_t collisionTypeId
Definition: Body.h:71
void fromTransformations(const Transformations &transformations)
Synchronizes this rigid body with transformations.
Definition: Body.cpp:461
const string & getId()
Definition: Body.cpp:156
void addCollisionListener(CollisionListener *listener)
Add a collision listener to this rigid body.
Definition: Body.cpp:544
uint16_t getCollisionTypeId()
Definition: Body.cpp:174
void fireOnCollision(Body *other, CollisionResponse &collisionResponse)
Fire on collision.
Definition: Body.cpp:554
static constexpr int32_t TYPE_STATIC
Definition: Body.h:47
const Vector3 getLinearVelocity()
Definition: Body.cpp:379
static constexpr int32_t TYPE_COLLISION
Definition: Body.h:50
void setRootId(const string &rootId)
Set root id.
Definition: Body.cpp:166
Dynamic physics world class.
Definition: World.h:38
reactphysics3d::DynamicsWorld world
Definition: World.h:47
Axis aligned bounding box used for frustum, this is not directly connectable with physics engine.
Definition: BoundingBox.h:25
const Vector3 & getDimensions() const
Definition: BoundingBox.h:127
Capsule physics primitive.
Definition: Capsule.h:18
Oriented bounding box physics primitive.
Terrain mesh physics primitive.
Definition: TerrainMesh.h:28
BoundingVolume * clone() const override
Clones this bounding volume.
Standard math functions.
Definition: Math.h:21
4x4 3D Matrix class
Definition: Matrix4x4.h:24
array< float, 16 > & getArray() const
Returns array data.
Definition: Matrix4x4.h:616
Quaternion class.
Definition: Quaternion.h:22
3D vector 3 class
Definition: Vector3.h:22
float getY() const
Definition: Vector3.h:119
float getX() const
Definition: Vector3.h:103
float getZ() const
Definition: Vector3.h:136
bool equals(const Vector3 &v) const
Compares this vector with given vector.
Definition: Vector3.h:381
Vector3 & set(float x, float y, float z)
Set up vector.
Definition: Vector3.h:73
Console class.
Definition: Console.h:26