2007-05-27 12:01:53 -04:00
|
|
|
//
|
|
|
|
// SuperTuxKart - a fun racing game with go-kart
|
|
|
|
// Copyright (C) 2006 Joerg Henrichs
|
|
|
|
//
|
|
|
|
// This program is free software; you can redistribute it and/or
|
|
|
|
// modify it under the terms of the GNU General Public License
|
2008-06-12 20:53:52 -04:00
|
|
|
// as published by the Free Software Foundation; either version 3
|
2007-05-27 12:01:53 -04:00
|
|
|
// of the License, or (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
2007-11-03 09:13:26 -04:00
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty ofati
|
2007-05-27 12:01:53 -04:00
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program; if not, write to the Free Software
|
|
|
|
// Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
|
2008-11-06 20:05:52 -05:00
|
|
|
#include "physics/physics.hpp"
|
2008-09-19 02:07:29 -04:00
|
|
|
|
2011-09-30 02:29:00 -04:00
|
|
|
#include "animations/three_d_animation.hpp"
|
2013-06-29 18:57:18 -04:00
|
|
|
#include "karts/abstract_kart.hpp"
|
2012-03-19 16:21:11 -04:00
|
|
|
#include "karts/kart_properties.hpp"
|
2012-04-01 18:57:41 -04:00
|
|
|
#include "karts/rescue_animation.hpp"
|
2013-06-29 18:57:18 -04:00
|
|
|
#include "items/flyable.hpp"
|
|
|
|
#include "modes/world.hpp"
|
2011-01-17 16:21:47 -05:00
|
|
|
#include "graphics/stars.hpp"
|
2012-10-15 20:09:33 -04:00
|
|
|
#include "karts/explosion_animation.hpp"
|
2009-01-26 21:36:37 -05:00
|
|
|
#include "physics/btKart.hpp"
|
2009-01-22 07:02:40 -05:00
|
|
|
#include "physics/btUprightConstraint.hpp"
|
2010-02-22 07:02:13 -05:00
|
|
|
#include "physics/irr_debug_drawer.hpp"
|
2011-09-28 06:52:41 -04:00
|
|
|
#include "physics/physical_object.hpp"
|
2011-11-30 20:14:10 -05:00
|
|
|
#include "physics/stk_dynamics_world.hpp"
|
2011-04-17 18:06:45 -04:00
|
|
|
#include "physics/triangle_mesh.hpp"
|
2009-01-22 17:27:13 -05:00
|
|
|
#include "tracks/track.hpp"
|
2007-05-27 12:01:53 -04:00
|
|
|
|
2007-12-12 09:07:26 -05:00
|
|
|
// ----------------------------------------------------------------------------
|
2008-09-07 11:10:59 -04:00
|
|
|
/** Initialise physics.
|
|
|
|
* Create the bullet dynamics world.
|
|
|
|
*/
|
2008-09-19 02:07:29 -04:00
|
|
|
Physics::Physics() : btSequentialImpulseConstraintSolver()
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2012-05-07 17:59:21 -04:00
|
|
|
m_collision_conf = new btDefaultCollisionConfiguration();
|
|
|
|
m_dispatcher = new btCollisionDispatcher(m_collision_conf);
|
2008-09-19 02:07:29 -04:00
|
|
|
} // Physics
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** The actual initialisation of the physics, which is called after the track
|
|
|
|
* model is loaded. This allows the physics to use the actual track dimension
|
|
|
|
* for the axis sweep.
|
|
|
|
*/
|
|
|
|
void Physics::init(const Vec3 &world_min, const Vec3 &world_max)
|
|
|
|
{
|
2012-05-07 17:59:21 -04:00
|
|
|
m_physics_loop_active = false;
|
|
|
|
m_axis_sweep = new btAxisSweep3(world_min, world_max);
|
|
|
|
m_dynamics_world = new STKDynamicsWorld(m_dispatcher,
|
|
|
|
m_axis_sweep,
|
|
|
|
this,
|
|
|
|
m_collision_conf);
|
|
|
|
m_karts_to_delete.clear();
|
2011-12-19 17:02:48 -05:00
|
|
|
m_dynamics_world->setGravity(
|
|
|
|
btVector3(0.0f,
|
|
|
|
-World::getWorld()->getTrack()->getGravity(),
|
|
|
|
0.0f));
|
2010-02-22 07:02:13 -05:00
|
|
|
m_debug_drawer = new IrrDebugDrawer();
|
|
|
|
m_dynamics_world->setDebugDrawer(m_debug_drawer);
|
2008-09-19 02:07:29 -04:00
|
|
|
} // init
|
2009-01-14 23:08:41 -05:00
|
|
|
|
2007-05-27 12:01:53 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
Physics::~Physics()
|
|
|
|
{
|
2010-02-26 02:03:12 -05:00
|
|
|
delete m_debug_drawer;
|
2007-05-27 12:01:53 -04:00
|
|
|
delete m_dynamics_world;
|
2007-11-08 07:31:54 -05:00
|
|
|
delete m_axis_sweep;
|
2007-12-08 08:04:56 -05:00
|
|
|
delete m_dispatcher;
|
2007-11-08 07:31:54 -05:00
|
|
|
delete m_collision_conf;
|
2007-05-27 12:01:53 -04:00
|
|
|
} // ~Physics
|
|
|
|
|
2011-12-19 17:02:48 -05:00
|
|
|
// ----------------------------------------------------------------------------
|
2008-09-07 11:10:59 -04:00
|
|
|
/** Adds a kart to the physics engine.
|
2012-04-01 18:57:41 -04:00
|
|
|
* This adds the rigid body, the vehicle, and the upright constraint, but only
|
|
|
|
* if the kart is not already in the physics world.
|
2008-09-07 11:10:59 -04:00
|
|
|
* \param kart The kart to add.
|
|
|
|
* \param vehicle The raycast vehicle object.
|
|
|
|
*/
|
2012-03-19 16:21:11 -04:00
|
|
|
void Physics::addKart(const AbstractKart *kart)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2013-05-29 18:04:35 -04:00
|
|
|
const btCollisionObjectArray &all_objs =
|
2012-04-01 18:57:41 -04:00
|
|
|
m_dynamics_world->getCollisionObjectArray();
|
|
|
|
for(unsigned int i=0; i<(unsigned int)all_objs.size(); i++)
|
|
|
|
{
|
|
|
|
if(btRigidBody::upcast(all_objs[i])== kart->getBody())
|
|
|
|
return;
|
|
|
|
}
|
2007-09-30 10:17:28 -04:00
|
|
|
m_dynamics_world->addRigidBody(kart->getBody());
|
2009-01-22 07:02:40 -05:00
|
|
|
m_dynamics_world->addVehicle(kart->getVehicle());
|
2008-04-22 10:20:52 -04:00
|
|
|
m_dynamics_world->addConstraint(kart->getUprightConstraint());
|
2007-05-27 12:01:53 -04:00
|
|
|
} // addKart
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
2007-11-03 09:13:26 -04:00
|
|
|
/** Removes a kart from the physics engine. This is used when rescuing a kart
|
2008-09-07 11:10:59 -04:00
|
|
|
* (and during cleanup).
|
|
|
|
* \param kart The kart to remove.
|
2007-05-27 12:01:53 -04:00
|
|
|
*/
|
2012-03-19 16:21:11 -04:00
|
|
|
void Physics::removeKart(const AbstractKart *kart)
|
2007-05-27 12:01:53 -04:00
|
|
|
{
|
2012-05-07 17:59:21 -04:00
|
|
|
// We can't simply remove a kart from the physics world when currently
|
|
|
|
// loops over all kart objects are active. This can happen in collision
|
|
|
|
// handling, where a collision of a kart with a cake etc. removes
|
|
|
|
// a kart from the physics. In this case save pointers to the kart
|
|
|
|
// to be removed, and remove them once the physics processing is done.
|
|
|
|
if(m_physics_loop_active)
|
|
|
|
{
|
|
|
|
// Make sure to remove each kart only once.
|
|
|
|
if(std::find(m_karts_to_delete.begin(), m_karts_to_delete.end(), kart)
|
|
|
|
== m_karts_to_delete.end())
|
|
|
|
{
|
|
|
|
m_karts_to_delete.push_back(kart);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
m_dynamics_world->removeRigidBody(kart->getBody());
|
|
|
|
m_dynamics_world->removeVehicle(kart->getVehicle());
|
|
|
|
m_dynamics_world->removeConstraint(kart->getUprightConstraint());
|
|
|
|
}
|
2007-05-27 12:01:53 -04:00
|
|
|
} // removeKart
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
2008-09-07 11:10:59 -04:00
|
|
|
/** Updates the physics simulation and handles all collisions.
|
|
|
|
* \param dt Time step.
|
|
|
|
*/
|
2007-05-27 12:01:53 -04:00
|
|
|
void Physics::update(float dt)
|
|
|
|
{
|
2012-05-07 17:59:21 -04:00
|
|
|
m_physics_loop_active = true;
|
2009-01-22 07:02:40 -05:00
|
|
|
// Bullet can report the same collision more than once (up to 4
|
2011-09-28 06:52:41 -04:00
|
|
|
// contact points per collision). Additionally, more than one internal
|
2009-01-22 07:02:40 -05:00
|
|
|
// substep might be taken, resulting in potentially even more
|
2007-12-08 08:04:56 -05:00
|
|
|
// duplicates. To handle this, all collisions (i.e. pair of objects)
|
|
|
|
// are stored in a vector, but only one entry per collision pair
|
|
|
|
// of objects.
|
|
|
|
m_all_collisions.clear();
|
2008-02-17 21:03:06 -05:00
|
|
|
|
|
|
|
// Maximum of three substeps. This will work for framerate down to
|
|
|
|
// 20 FPS (bullet default frequency is 60 HZ).
|
2008-05-07 21:48:51 -04:00
|
|
|
m_dynamics_world->stepSimulation(dt, 3);
|
2007-12-08 08:04:56 -05:00
|
|
|
|
2011-09-28 06:52:41 -04:00
|
|
|
// Now handle the actual collision. Note: flyables can not be removed
|
|
|
|
// inside of this loop, since the same flyables might hit more than one
|
|
|
|
// other object. So only a flag is set in the flyables, the actual
|
2007-12-08 08:04:56 -05:00
|
|
|
// clean up is then done later in the projectile manager.
|
|
|
|
std::vector<CollisionPair>::iterator p;
|
|
|
|
for(p=m_all_collisions.begin(); p!=m_all_collisions.end(); ++p)
|
|
|
|
{
|
2011-09-28 06:52:41 -04:00
|
|
|
// Kart-kart collision
|
|
|
|
// --------------------
|
2011-12-14 16:02:15 -05:00
|
|
|
if(p->getUserPointer(0)->is(UserPointer::UP_KART))
|
2011-09-28 06:52:41 -04:00
|
|
|
{
|
2012-03-19 16:21:11 -04:00
|
|
|
AbstractKart *a=p->getUserPointer(0)->getPointerKart();
|
|
|
|
AbstractKart *b=p->getUserPointer(1)->getPointerKart();
|
2013-05-29 18:04:35 -04:00
|
|
|
KartKartCollision(p->getUserPointer(0)->getPointerKart(),
|
2011-12-14 16:02:15 -05:00
|
|
|
p->getContactPointCS(0),
|
2013-05-29 18:04:35 -04:00
|
|
|
p->getUserPointer(1)->getPointerKart(),
|
2011-12-14 16:02:15 -05:00
|
|
|
p->getContactPointCS(1) );
|
2011-09-28 06:52:41 -04:00
|
|
|
continue;
|
2007-12-08 08:04:56 -05:00
|
|
|
} // if kart-kart collision
|
2011-09-28 06:52:41 -04:00
|
|
|
|
2011-12-14 16:02:15 -05:00
|
|
|
if(p->getUserPointer(0)->is(UserPointer::UP_PHYSICAL_OBJECT))
|
2007-12-08 08:04:56 -05:00
|
|
|
{
|
2011-09-28 06:52:41 -04:00
|
|
|
// Kart hits physical object
|
|
|
|
// -------------------------
|
2011-12-19 17:02:48 -05:00
|
|
|
PhysicalObject *obj = p->getUserPointer(0)
|
|
|
|
->getPointerPhysicalObject();
|
2011-09-28 06:52:41 -04:00
|
|
|
if(obj->isCrashReset())
|
2007-12-08 08:04:56 -05:00
|
|
|
{
|
2012-03-19 16:21:11 -04:00
|
|
|
AbstractKart *kart = p->getUserPointer(1)->getPointerKart();
|
2012-04-01 18:57:41 -04:00
|
|
|
new RescueAnimation(kart);
|
2007-12-08 08:04:56 -05:00
|
|
|
}
|
2012-10-15 20:09:33 -04:00
|
|
|
else if (obj->isExplodeKartObject())
|
|
|
|
{
|
|
|
|
AbstractKart *kart = p->getUserPointer(1)->getPointerKart();
|
|
|
|
ExplosionAnimation::create(kart);
|
|
|
|
}
|
2013-08-12 20:36:37 -04:00
|
|
|
else if (obj->isFlattenKartObject())
|
|
|
|
{
|
|
|
|
AbstractKart *kart = p->getUserPointer(1)->getPointerKart();
|
|
|
|
const KartProperties* kp = kart->getKartProperties();
|
|
|
|
kart->setSquash(kp->getSquashDuration(), kp->getSquashSlowdown());
|
|
|
|
}
|
2011-09-28 06:52:41 -04:00
|
|
|
continue;
|
|
|
|
}
|
2008-02-05 06:56:21 -05:00
|
|
|
|
2011-12-14 16:02:15 -05:00
|
|
|
if(p->getUserPointer(0)->is(UserPointer::UP_ANIMATION))
|
2011-09-30 02:29:00 -04:00
|
|
|
{
|
|
|
|
// Kart hits animation
|
2011-12-14 16:02:15 -05:00
|
|
|
ThreeDAnimation *anim=p->getUserPointer(0)->getPointerAnimation();
|
2011-09-30 02:29:00 -04:00
|
|
|
if(anim->isCrashReset())
|
|
|
|
{
|
2012-03-19 16:21:11 -04:00
|
|
|
AbstractKart *kart = p->getUserPointer(1)->getPointerKart();
|
2012-04-01 18:57:41 -04:00
|
|
|
new RescueAnimation(kart);
|
2011-09-30 02:29:00 -04:00
|
|
|
}
|
2012-10-15 20:09:33 -04:00
|
|
|
else if (anim->isExplodeKartObject())
|
|
|
|
{
|
|
|
|
AbstractKart *kart = p->getUserPointer(1)->getPointerKart();
|
|
|
|
ExplosionAnimation::create(kart);
|
|
|
|
}
|
2013-08-12 20:36:37 -04:00
|
|
|
else if (anim->isFlattenKartObject())
|
|
|
|
{
|
|
|
|
AbstractKart *kart = p->getUserPointer(1)->getPointerKart();
|
|
|
|
const KartProperties* kp = kart->getKartProperties();
|
|
|
|
kart->setSquash(kp->getSquashDuration(), kp->getSquashSlowdown());
|
|
|
|
}
|
2011-09-30 02:29:00 -04:00
|
|
|
continue;
|
2011-09-28 06:52:41 -04:00
|
|
|
|
2011-09-30 02:29:00 -04:00
|
|
|
}
|
2011-09-28 06:52:41 -04:00
|
|
|
// now the first object must be a projectile
|
|
|
|
// =========================================
|
2011-12-14 16:02:15 -05:00
|
|
|
if(p->getUserPointer(1)->is(UserPointer::UP_TRACK))
|
2011-09-28 06:52:41 -04:00
|
|
|
{
|
|
|
|
// Projectile hits track
|
|
|
|
// ---------------------
|
2011-12-14 16:02:15 -05:00
|
|
|
p->getUserPointer(0)->getPointerFlyable()->hitTrack();
|
2011-09-28 06:52:41 -04:00
|
|
|
}
|
2011-12-14 16:02:15 -05:00
|
|
|
else if(p->getUserPointer(1)->is(UserPointer::UP_PHYSICAL_OBJECT))
|
2011-09-28 06:52:41 -04:00
|
|
|
{
|
|
|
|
// Projectile hits physical object
|
|
|
|
// -------------------------------
|
2011-12-14 16:02:15 -05:00
|
|
|
p->getUserPointer(0)->getPointerFlyable()
|
|
|
|
->hit(NULL, p->getUserPointer(1)->getPointerPhysicalObject());
|
2011-09-28 06:52:41 -04:00
|
|
|
|
|
|
|
}
|
2011-12-14 16:02:15 -05:00
|
|
|
else if(p->getUserPointer(1)->is(UserPointer::UP_KART))
|
2011-09-28 06:52:41 -04:00
|
|
|
{
|
|
|
|
// Projectile hits kart
|
|
|
|
// --------------------
|
|
|
|
// Only explode a bowling ball if the target is
|
|
|
|
// not invulnerable
|
2013-08-07 23:11:03 -04:00
|
|
|
AbstractKart* target_kart = p->getUserPointer(1)->getPointerKart();
|
2011-12-14 16:02:15 -05:00
|
|
|
if(p->getUserPointer(0)->getPointerFlyable()->getType()
|
2013-08-07 23:11:03 -04:00
|
|
|
!=PowerupManager::POWERUP_BOWLING ||
|
|
|
|
!target_kart->isInvulnerable() )
|
|
|
|
{
|
2011-12-14 16:02:15 -05:00
|
|
|
p->getUserPointer(0)->getPointerFlyable()
|
2013-08-07 23:11:03 -04:00
|
|
|
->hit(target_kart);
|
|
|
|
}
|
|
|
|
|
2011-09-28 06:52:41 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// Projectile hits projectile
|
|
|
|
// --------------------------
|
2011-12-14 16:02:15 -05:00
|
|
|
p->getUserPointer(0)->getPointerFlyable()->hit(NULL);
|
|
|
|
p->getUserPointer(1)->getPointerFlyable()->hit(NULL);
|
2007-12-08 08:04:56 -05:00
|
|
|
}
|
|
|
|
} // for all p in m_all_collisions
|
2012-05-07 17:59:21 -04:00
|
|
|
|
|
|
|
m_physics_loop_active = false;
|
|
|
|
// Now remove the karts that were removed while the above loop
|
|
|
|
// was active. Now we can safely call removeKart, since the loop
|
|
|
|
// is finished and m_physics_world_active is not set anymore.
|
|
|
|
for(unsigned int i=0; i<m_karts_to_delete.size(); i++)
|
|
|
|
removeKart(m_karts_to_delete[i]);
|
|
|
|
m_karts_to_delete.clear();
|
2007-05-27 12:01:53 -04:00
|
|
|
} // update
|
|
|
|
|
2007-11-03 09:13:26 -04:00
|
|
|
//-----------------------------------------------------------------------------
|
2013-05-29 18:04:35 -04:00
|
|
|
/** Handles the special case of two karts colliding with each other, which
|
|
|
|
* means that bombs must be passed on. If both karts have a bomb, they'll
|
|
|
|
* explode immediately. This function is called from physics::update() on the
|
|
|
|
* server and if no networking is used, and from race_state on the client to
|
2011-09-28 06:52:41 -04:00
|
|
|
* replay what happened on the server.
|
2011-12-12 00:42:40 -05:00
|
|
|
* \param kart_a First kart involved in the collision.
|
2013-05-29 18:04:35 -04:00
|
|
|
* \param contact_point_a Location of collision at first kart (in kart
|
2012-01-31 07:02:10 -05:00
|
|
|
* coordinates).
|
2011-12-12 00:42:40 -05:00
|
|
|
* \param kart_b Second kart involved in the collision.
|
2012-01-31 07:02:10 -05:00
|
|
|
* \param contact_point_b Location of collision at second kart (in kart
|
|
|
|
* coordinates).
|
2007-12-08 08:04:56 -05:00
|
|
|
*/
|
2013-05-29 18:04:35 -04:00
|
|
|
void Physics::KartKartCollision(AbstractKart *kart_a,
|
2012-03-19 16:21:11 -04:00
|
|
|
const Vec3 &contact_point_a,
|
2013-05-29 18:04:35 -04:00
|
|
|
AbstractKart *kart_b,
|
2012-03-19 16:21:11 -04:00
|
|
|
const Vec3 &contact_point_b)
|
2007-12-08 08:04:56 -05:00
|
|
|
{
|
2012-02-08 15:22:00 -05:00
|
|
|
// Only one kart needs to handle the attachments, it will
|
|
|
|
// fix the attachments for the other kart.
|
2012-01-31 07:02:10 -05:00
|
|
|
kart_a->crashed(kart_b, /*handle_attachments*/true);
|
|
|
|
kart_b->crashed(kart_a, /*handle_attachments*/false);
|
2010-01-10 07:16:32 -05:00
|
|
|
|
2012-03-19 16:21:11 -04:00
|
|
|
AbstractKart *left_kart, *right_kart;
|
2012-02-08 15:22:00 -05:00
|
|
|
|
|
|
|
// Determine which kart is pushed to the left, and which one to the
|
2013-05-29 18:04:35 -04:00
|
|
|
// right. Ideally the sign of the X coordinate of the local conact point
|
2012-02-08 15:22:00 -05:00
|
|
|
// could decide the direction (negative X --> was hit on left side, gets
|
|
|
|
// push to right), but that can lead to both karts being pushed in the
|
|
|
|
// same direction (front left of kart hits rear left).
|
|
|
|
// So we just use a simple test (which does the right thing in ideal
|
2013-05-29 18:04:35 -04:00
|
|
|
// crashes, but avoids pushing both karts in corner cases
|
2012-02-08 15:22:00 -05:00
|
|
|
// - pun intended ;) ).
|
|
|
|
if(contact_point_a.getX() < contact_point_b.getX())
|
2010-01-10 07:16:32 -05:00
|
|
|
{
|
2012-02-19 16:41:54 -05:00
|
|
|
left_kart = kart_b;
|
|
|
|
right_kart = kart_a;
|
2010-01-10 07:16:32 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-02-19 16:41:54 -05:00
|
|
|
left_kart = kart_a;
|
|
|
|
right_kart = kart_b;
|
2010-01-10 07:16:32 -05:00
|
|
|
}
|
2011-12-12 00:42:40 -05:00
|
|
|
|
2012-02-19 16:41:54 -05:00
|
|
|
// Add a scaling factor depending on the mass (avoid div by zero).
|
|
|
|
// The value of f_right is applied to the right kart, and f_left
|
|
|
|
// to the left kart. f_left = 1 / f_right
|
2013-05-29 18:04:35 -04:00
|
|
|
float f_right = right_kart->getKartProperties()->getMass() > 0
|
|
|
|
? left_kart->getKartProperties()->getMass()
|
2012-02-09 00:41:13 -05:00
|
|
|
/ right_kart->getKartProperties()->getMass()
|
|
|
|
: 1.5f;
|
2012-02-08 15:22:00 -05:00
|
|
|
// Add a scaling factor depending on speed (avoid div by 0)
|
2013-05-29 18:04:35 -04:00
|
|
|
f_right *= right_kart->getSpeed() > 0
|
2012-02-09 00:41:13 -05:00
|
|
|
? left_kart->getSpeed()
|
|
|
|
/ right_kart->getSpeed()
|
2012-02-08 15:22:00 -05:00
|
|
|
: 1.5f;
|
2012-02-09 00:41:13 -05:00
|
|
|
// Cap f_right to [0.8,1.25], which results in f_left being
|
2012-02-08 15:22:00 -05:00
|
|
|
// capped in the same interval
|
2012-02-09 00:41:13 -05:00
|
|
|
if(f_right > 1.25f)
|
|
|
|
f_right = 1.25f;
|
|
|
|
else if(f_right< 0.8f)
|
|
|
|
f_right = 0.8f;
|
2012-10-25 16:54:19 -04:00
|
|
|
float f_left = 1/f_right;
|
2011-12-12 00:42:40 -05:00
|
|
|
|
2012-02-09 00:41:13 -05:00
|
|
|
// Check if a kart is more 'actively' trying to push another kart
|
|
|
|
// by checking its local sidewards velocity
|
|
|
|
float vel_left = left_kart->getVelocityLC().getX();
|
|
|
|
float vel_right = right_kart->getVelocityLC().getX();
|
|
|
|
|
2013-05-29 18:04:35 -04:00
|
|
|
// Use the difference in speed to determine which kart gets a
|
2012-02-09 00:41:13 -05:00
|
|
|
// ramming bonus. Normally vel_right and vel_left will have
|
|
|
|
// a different sign: right kart will be driving to the left,
|
|
|
|
// and left kart to the right (both pushing at each other).
|
|
|
|
// By using the sum we get the intended effect: if both karts
|
|
|
|
// are pushing with the same speed, vel_diff is 0, if the right
|
|
|
|
// kart is driving faster vel_diff will be < 0. If both velocities
|
|
|
|
// have the same sign, one kart is trying to steer away from the
|
|
|
|
// other, in which case it gets an even bigger push.
|
|
|
|
float vel_diff = vel_right + vel_left;
|
|
|
|
|
|
|
|
// More driving towards left --> left kart gets bigger impulse
|
|
|
|
if(vel_diff<0)
|
|
|
|
{
|
2012-11-14 19:10:36 -05:00
|
|
|
// Avoid too large impulse for karts that are driving
|
|
|
|
// slow (and division by zero)
|
|
|
|
if(fabsf(vel_left)>2.0f)
|
|
|
|
f_left *= 1.0f - vel_diff/fabsf(vel_left);
|
2012-02-19 16:41:54 -05:00
|
|
|
if(f_left > 2.0f)
|
|
|
|
f_left = 2.0f;
|
2012-02-09 00:41:13 -05:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2012-11-14 19:10:36 -05:00
|
|
|
// Avoid too large impulse for karts that are driving
|
|
|
|
// slow (and division by zero)
|
|
|
|
if(fabsf(vel_right)>2.0f)
|
|
|
|
f_right *= 1.0f + vel_diff/fabsf(vel_right);
|
2012-02-19 16:41:54 -05:00
|
|
|
if(f_right > 2.0f)
|
|
|
|
f_right = 2.0f;
|
2012-02-09 00:41:13 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Increase the effect somewhat by squaring the factors
|
|
|
|
f_left = f_left * f_left;
|
|
|
|
f_right = f_right * f_right;
|
2011-12-14 16:02:15 -05:00
|
|
|
|
2012-02-08 15:22:00 -05:00
|
|
|
// First push one kart to the left (if there is not already
|
|
|
|
// an impulse happening - one collision might cause more
|
|
|
|
// than one impulse otherwise)
|
2012-02-09 00:41:13 -05:00
|
|
|
if(right_kart->getVehicle()->getCentralImpulseTime()<=0)
|
2010-01-10 07:16:32 -05:00
|
|
|
{
|
2012-02-09 00:41:13 -05:00
|
|
|
const KartProperties *kp = left_kart->getKartProperties();
|
2012-02-19 16:41:54 -05:00
|
|
|
Vec3 impulse(kp->getCollisionImpulse()*f_right, 0, 0);
|
2012-02-09 00:41:13 -05:00
|
|
|
impulse = right_kart->getTrans().getBasis() * impulse;
|
|
|
|
right_kart->getVehicle()
|
2012-02-08 15:22:00 -05:00
|
|
|
->setTimedCentralImpulse(kp->getCollisionImpulseTime(),
|
|
|
|
impulse);
|
2012-02-09 00:41:13 -05:00
|
|
|
right_kart ->getBody()->setAngularVelocity(btVector3(0,0,0));
|
2012-02-08 15:22:00 -05:00
|
|
|
}
|
2011-12-14 16:02:15 -05:00
|
|
|
|
2012-02-08 15:22:00 -05:00
|
|
|
// Then push the other kart to the right (if there is no
|
|
|
|
// impulse happening atm).
|
2012-02-09 00:41:13 -05:00
|
|
|
if(left_kart->getVehicle()->getCentralImpulseTime()<=0)
|
2012-02-08 15:22:00 -05:00
|
|
|
{
|
2012-02-09 00:41:13 -05:00
|
|
|
const KartProperties *kp = right_kart->getKartProperties();
|
2012-02-19 16:41:54 -05:00
|
|
|
Vec3 impulse = Vec3(-kp->getCollisionImpulse()*f_left, 0, 0);
|
2012-02-09 00:41:13 -05:00
|
|
|
impulse = left_kart->getTrans().getBasis() * impulse;
|
|
|
|
left_kart->getVehicle()
|
2012-02-08 15:22:00 -05:00
|
|
|
->setTimedCentralImpulse(kp->getCollisionImpulseTime(),
|
|
|
|
impulse);
|
2012-02-09 00:41:13 -05:00
|
|
|
left_kart->getBody()->setAngularVelocity(btVector3(0,0,0));
|
2010-01-10 07:16:32 -05:00
|
|
|
}
|
2011-12-14 16:02:15 -05:00
|
|
|
|
2007-12-08 08:04:56 -05:00
|
|
|
} // KartKartCollision
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
|
|
/** This function is called at each internal bullet timestep. It is used
|
2008-09-07 11:10:59 -04:00
|
|
|
* here to do the collision handling: using the contact manifolds after a
|
|
|
|
* physics time step might miss some collisions (when more than one internal
|
|
|
|
* time step was done, and the collision is added and removed). So this
|
|
|
|
* function stores all collisions in a list, which is then handled after the
|
2011-09-28 06:52:41 -04:00
|
|
|
* actual physics timestep. This list only stores a collision if it's not
|
2008-09-07 11:10:59 -04:00
|
|
|
* already in the list, so a collisions which is reported more than once is
|
|
|
|
* nevertheless only handled once.
|
2013-05-29 18:04:35 -04:00
|
|
|
* The list of collision
|
2008-09-07 11:10:59 -04:00
|
|
|
* Parameters: see bullet documentation for details.
|
|
|
|
*/
|
2007-12-08 08:04:56 -05:00
|
|
|
btScalar Physics::solveGroup(btCollisionObject** bodies, int numBodies,
|
|
|
|
btPersistentManifold** manifold,int numManifolds,
|
2011-12-19 17:02:48 -05:00
|
|
|
btTypedConstraint** constraints,
|
|
|
|
int numConstraints,
|
2009-08-18 06:59:21 -04:00
|
|
|
const btContactSolverInfo& info,
|
2013-05-29 18:04:35 -04:00
|
|
|
btIDebugDraw* debugDrawer,
|
|
|
|
btStackAlloc* stackAlloc,
|
2011-12-19 17:02:48 -05:00
|
|
|
btDispatcher* dispatcher)
|
|
|
|
{
|
2007-12-08 08:04:56 -05:00
|
|
|
btScalar returnValue=
|
2013-05-29 18:04:35 -04:00
|
|
|
btSequentialImpulseConstraintSolver::solveGroup(bodies, numBodies,
|
2011-12-19 17:02:48 -05:00
|
|
|
manifold, numManifolds,
|
|
|
|
constraints,
|
2009-08-18 06:59:21 -04:00
|
|
|
numConstraints, info,
|
2013-05-29 18:04:35 -04:00
|
|
|
debugDrawer,
|
2011-12-19 17:02:48 -05:00
|
|
|
stackAlloc,
|
2007-12-08 08:04:56 -05:00
|
|
|
dispatcher);
|
|
|
|
int currentNumManifolds = m_dispatcher->getNumManifolds();
|
2009-08-18 06:59:21 -04:00
|
|
|
// We can't explode a rocket in a loop, since a rocket might collide with
|
|
|
|
// more than one object, and/or more than once with each object (if there
|
2007-12-08 08:04:56 -05:00
|
|
|
// is more than one collision point). So keep a list of rockets that will
|
|
|
|
// be exploded after the collisions
|
|
|
|
for(int i=0; i<currentNumManifolds; i++)
|
2009-08-18 06:59:21 -04:00
|
|
|
{
|
2013-05-29 18:04:35 -04:00
|
|
|
btPersistentManifold* contact_manifold =
|
2011-12-19 17:02:48 -05:00
|
|
|
m_dynamics_world->getDispatcher()->getManifoldByIndexInternal(i);
|
2007-11-03 09:13:26 -04:00
|
|
|
|
2013-05-29 18:04:35 -04:00
|
|
|
btCollisionObject* objA =
|
2011-12-19 17:02:48 -05:00
|
|
|
static_cast<btCollisionObject*>(contact_manifold->getBody0());
|
2013-05-29 18:04:35 -04:00
|
|
|
btCollisionObject* objB =
|
2011-12-19 17:02:48 -05:00
|
|
|
static_cast<btCollisionObject*>(contact_manifold->getBody1());
|
2009-08-18 06:59:21 -04:00
|
|
|
|
2011-12-19 17:02:48 -05:00
|
|
|
unsigned int num_contacts = contact_manifold->getNumContacts();
|
|
|
|
if(!num_contacts) continue; // no real collision
|
2007-11-03 09:13:26 -04:00
|
|
|
|
2008-01-30 00:30:10 -05:00
|
|
|
UserPointer *upA = (UserPointer*)(objA->getUserPointer());
|
|
|
|
UserPointer *upB = (UserPointer*)(objB->getUserPointer());
|
2007-11-03 09:13:26 -04:00
|
|
|
|
2008-04-28 23:29:28 -04:00
|
|
|
if(!upA || !upB) continue;
|
2011-09-28 06:52:41 -04:00
|
|
|
|
2007-11-03 09:13:26 -04:00
|
|
|
// 1) object A is a track
|
|
|
|
// =======================
|
2009-08-18 06:59:21 -04:00
|
|
|
if(upA->is(UserPointer::UP_TRACK))
|
|
|
|
{
|
2008-02-05 06:56:21 -05:00
|
|
|
if(upB->is(UserPointer::UP_FLYABLE)) // 1.1 projectile hits track
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB,
|
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA);
|
2008-01-30 00:30:10 -05:00
|
|
|
else if(upB->is(UserPointer::UP_KART))
|
2008-09-07 11:10:59 -04:00
|
|
|
{
|
2012-03-19 16:21:11 -04:00
|
|
|
AbstractKart *kart=upB->getPointerKart();
|
2011-12-19 17:02:48 -05:00
|
|
|
int n = contact_manifold->getContactPoint(0).m_index0;
|
2013-05-29 18:04:35 -04:00
|
|
|
const Material *m
|
2011-04-17 18:06:45 -04:00
|
|
|
= n>=0 ? upA->getPointerTriangleMesh()->getMaterial(n)
|
|
|
|
: NULL;
|
2012-11-11 16:56:54 -05:00
|
|
|
// I assume that the normal needs to be flipped in this case,
|
|
|
|
// but I can't verify this since it appears that bullet
|
|
|
|
// always has the kart as object A, not B.
|
|
|
|
const btVector3 &normal = -contact_manifold->getContactPoint(0)
|
|
|
|
.m_normalWorldOnB;
|
|
|
|
kart->crashed(m, normal);
|
2008-09-07 11:10:59 -04:00
|
|
|
}
|
2008-01-30 00:30:10 -05:00
|
|
|
}
|
2007-11-03 09:13:26 -04:00
|
|
|
// 2) object a is a kart
|
|
|
|
// =====================
|
2008-01-30 00:30:10 -05:00
|
|
|
else if(upA->is(UserPointer::UP_KART))
|
2007-11-03 09:13:26 -04:00
|
|
|
{
|
2008-01-30 00:30:10 -05:00
|
|
|
if(upB->is(UserPointer::UP_TRACK))
|
2008-09-07 11:10:59 -04:00
|
|
|
{
|
2012-03-19 16:21:11 -04:00
|
|
|
AbstractKart *kart = upA->getPointerKart();
|
2011-12-19 17:02:48 -05:00
|
|
|
int n = contact_manifold->getContactPoint(0).m_index1;
|
2013-05-29 18:04:35 -04:00
|
|
|
const Material *m
|
2011-04-17 18:06:45 -04:00
|
|
|
= n>=0 ? upB->getPointerTriangleMesh()->getMaterial(n)
|
|
|
|
: NULL;
|
2012-11-11 16:56:54 -05:00
|
|
|
const btVector3 &normal = contact_manifold->getContactPoint(0)
|
|
|
|
.m_normalWorldOnB;
|
|
|
|
kart->crashed(m, normal); // Kart hit track
|
2008-09-07 11:10:59 -04:00
|
|
|
}
|
2008-02-05 06:56:21 -05:00
|
|
|
else if(upB->is(UserPointer::UP_FLYABLE))
|
2011-09-28 06:52:41 -04:00
|
|
|
// 2.1 projectile hits kart
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB,
|
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA);
|
2008-01-30 00:30:10 -05:00
|
|
|
else if(upB->is(UserPointer::UP_KART))
|
2011-09-28 06:52:41 -04:00
|
|
|
// 2.2 kart hits kart
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA,
|
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB);
|
2011-09-28 06:52:41 -04:00
|
|
|
else if(upB->is(UserPointer::UP_PHYSICAL_OBJECT))
|
|
|
|
// 2.3 kart hits physical object
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB,
|
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA);
|
2011-09-30 02:29:00 -04:00
|
|
|
else if(upB->is(UserPointer::UP_ANIMATION))
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB,
|
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA);
|
2007-11-03 09:13:26 -04:00
|
|
|
}
|
|
|
|
// 3) object is a projectile
|
2008-09-07 11:10:59 -04:00
|
|
|
// =========================
|
2008-02-05 06:56:21 -05:00
|
|
|
else if(upA->is(UserPointer::UP_FLYABLE))
|
2007-11-03 09:13:26 -04:00
|
|
|
{
|
2011-09-28 06:52:41 -04:00
|
|
|
// 3.1) projectile hits track
|
|
|
|
// 3.2) projectile hits projectile
|
|
|
|
// 3.3) projectile hits physical object
|
|
|
|
// 3.4) projectile hits kart
|
|
|
|
if(upB->is(UserPointer::UP_TRACK ) ||
|
|
|
|
upB->is(UserPointer::UP_FLYABLE ) ||
|
|
|
|
upB->is(UserPointer::UP_PHYSICAL_OBJECT) ||
|
|
|
|
upB->is(UserPointer::UP_KART ) )
|
2008-01-30 00:30:10 -05:00
|
|
|
{
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA,
|
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB);
|
2007-11-03 09:13:26 -04:00
|
|
|
}
|
2009-08-18 06:59:21 -04:00
|
|
|
}
|
2011-09-28 06:52:41 -04:00
|
|
|
// Object is a physical object
|
|
|
|
// ===========================
|
2009-03-31 21:30:10 -04:00
|
|
|
else if(upA->is(UserPointer::UP_PHYSICAL_OBJECT))
|
2008-02-05 06:56:21 -05:00
|
|
|
{
|
2009-08-18 06:59:21 -04:00
|
|
|
if(upB->is(UserPointer::UP_FLYABLE))
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB,
|
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA);
|
2011-09-28 06:52:41 -04:00
|
|
|
else if(upB->is(UserPointer::UP_KART))
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA,
|
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB);
|
2007-11-03 09:13:26 -04:00
|
|
|
}
|
2011-09-30 02:29:00 -04:00
|
|
|
else if (upA->is(UserPointer::UP_ANIMATION))
|
|
|
|
{
|
|
|
|
if(upB->is(UserPointer::UP_KART))
|
2011-12-14 16:02:15 -05:00
|
|
|
m_all_collisions.push_back(
|
2011-12-19 17:02:48 -05:00
|
|
|
upA, contact_manifold->getContactPoint(0).m_localPointA,
|
|
|
|
upB, contact_manifold->getContactPoint(0).m_localPointB);
|
2011-09-30 02:29:00 -04:00
|
|
|
}
|
2013-05-29 18:04:35 -04:00
|
|
|
else
|
2011-12-19 17:02:48 -05:00
|
|
|
assert("Unknown user pointer"); // 4) Should never happen
|
2007-11-03 09:13:26 -04:00
|
|
|
} // for i<numManifolds
|
2007-12-08 08:04:56 -05:00
|
|
|
|
|
|
|
return returnValue;
|
|
|
|
} // solveGroup
|
2007-11-03 09:13:26 -04:00
|
|
|
|
2011-12-19 17:02:48 -05:00
|
|
|
// ----------------------------------------------------------------------------
|
|
|
|
/** A debug draw function to show the track and all karts.
|
|
|
|
*/
|
2007-05-27 12:01:53 -04:00
|
|
|
void Physics::draw()
|
|
|
|
{
|
2010-03-02 00:15:33 -05:00
|
|
|
if(!m_debug_drawer->debugEnabled() ||
|
|
|
|
!World::getWorld()->isRacePhase()) return;
|
2010-02-26 02:03:12 -05:00
|
|
|
|
2010-02-22 07:02:13 -05:00
|
|
|
video::SColor color(77,179,0,0);
|
|
|
|
video::SMaterial material;
|
|
|
|
material.Thickness = 2;
|
|
|
|
material.AmbientColor = color;
|
|
|
|
material.DiffuseColor = color;
|
|
|
|
material.EmissiveColor= color;
|
|
|
|
material.BackfaceCulling = false;
|
|
|
|
material.setFlag(video::EMF_LIGHTING, false);
|
|
|
|
irr_driver->getVideoDriver()->setMaterial(material);
|
2013-05-29 18:04:35 -04:00
|
|
|
irr_driver->getVideoDriver()->setTransform(video::ETS_WORLD,
|
2010-02-22 07:02:13 -05:00
|
|
|
core::IdentityMatrix);
|
|
|
|
m_dynamics_world->debugDrawWorld();
|
|
|
|
return;
|
2007-05-27 12:01:53 -04:00
|
|
|
} // draw
|
|
|
|
|
2011-12-19 17:02:48 -05:00
|
|
|
// ----------------------------------------------------------------------------
|
2007-05-27 12:01:53 -04:00
|
|
|
|
|
|
|
/* EOF */
|
|
|
|
|