Files
bullet3/Demos/HeightFieldFluidDemo/BulletHfFluid/btHfFluidRigidDynamicsWorld.cpp
2010-03-06 15:23:36 +00:00

274 lines
8.7 KiB
C++

/*
Bullet Continuous Collision Detection and Physics Library
Copyright (c) 2003-2009 Erwin Coumans http://bulletphysics.com
This software is provided 'as-is', without any express or implied warranty.
In no event will the authors be held liable for any damages arising from the use of this software.
Permission is granted to anyone to use this software for any purpose,
including commercial applications, and to alter it and redistribute it freely,
subject to the following restrictions:
1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required.
2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software.
3. This notice may not be removed or altered from any source distribution.
Experimental Buoyancy fluid demo written by John McCutchan
*/
#include <stdio.h>
#include "LinearMath/btQuickprof.h"
#include "LinearMath/btIDebugDraw.h"
#include "BulletCollision/CollisionShapes/btCollisionShape.h"
// height field fluid
#include "btHfFluid.h"
#include "btHfFluidBuoyantConvexShape.h"
#include "btHfFluidRigidDynamicsWorld.h"
btHfFluidRigidDynamicsWorld::btHfFluidRigidDynamicsWorld(btDispatcher* dispatcher,btBroadphaseInterface* pairCache,btConstraintSolver* constraintSolver,btCollisionConfiguration* collisionConfiguration)
:btDiscreteDynamicsWorld(dispatcher,pairCache,constraintSolver,collisionConfiguration)
{
m_drawMode = DRAWMODE_NORMAL;
m_bodyDrawMode = BODY_DRAWMODE_NORMAL;
}
btHfFluidRigidDynamicsWorld::~btHfFluidRigidDynamicsWorld()
{
}
void btHfFluidRigidDynamicsWorld::predictUnconstraintMotion(btScalar timeStep)
{
btDiscreteDynamicsWorld::predictUnconstraintMotion( timeStep);
for ( int i=0;i<m_hfFluids.size();++i)
{
btHfFluid* phff = m_hfFluids[i];
// XXX: phff->predictMotion(timeStep);
}
}
void btHfFluidRigidDynamicsWorld::internalSingleStepSimulation( btScalar timeStep)
{
btDiscreteDynamicsWorld::internalSingleStepSimulation( timeStep );
updateFluids (timeStep);
solveFluidConstraints (timeStep);
}
void btHfFluidRigidDynamicsWorld::updateFluids(btScalar timeStep)
{
BT_PROFILE("updateFluids");
for ( int i=0;i<m_hfFluids.size();i++)
{
btHfFluid* phff=(btHfFluid*)m_hfFluids[i];
phff->predictMotion (timeStep);
}
}
void btHfFluidRigidDynamicsWorld::solveFluidConstraints(btScalar timeStep)
{
BT_PROFILE("solve Fluid Contacts");
#if 0
if(m_hfFluids.size())
{
btHfFluid::solveClusters(m_hfFluids);
}
for(int i=0;i<m_hfFluids.size();++i)
{
btHfFluid* psb=(btHfFluid*)m_hfFluids[i];
psb->solveConstraints();
}
#endif
}
void btHfFluidRigidDynamicsWorld::addHfFluid(btHfFluid* body)
{
m_hfFluids.push_back(body);
btCollisionWorld::addCollisionObject(body,
btBroadphaseProxy::DefaultFilter,
btBroadphaseProxy::AllFilter);
}
void btHfFluidRigidDynamicsWorld::removeHfFluid(btHfFluid* body)
{
m_hfFluids.remove(body);
btCollisionWorld::removeCollisionObject(body);
}
void btHfFluidRigidDynamicsWorld::drawHfFluidGround (btIDebugDraw* debugDraw, btHfFluid* fluid)
{
const btScalar* ground = fluid->getGroundArray ();
btVector3 com = fluid->getWorldTransform().getOrigin();
btVector3 color = btVector3(btScalar(0.13f), btScalar(0.13f), btScalar(0.0));
for (int i = 1; i < fluid->getNumNodesWidth()-1; i++)
{
for (int j = 1; j < fluid->getNumNodesLength()-1; j++)
{
int sw = fluid->arrayIndex (i, j);
int se = fluid->arrayIndex (i+1, j);
int nw = fluid->arrayIndex (i, j+1);
int ne = fluid->arrayIndex (i+1, j+1);
btVector3 swV = btVector3 (fluid->widthPos (i), ground[sw], fluid->lengthPos (j));
btVector3 seV = btVector3 (fluid->widthPos (i+1), ground[se], fluid->lengthPos (j));
btVector3 nwV = btVector3 (fluid->widthPos (i), ground[nw], fluid->lengthPos (j+1));
btVector3 neV = btVector3 (fluid->widthPos (i+1), ground[ne], fluid->lengthPos (j+1));
debugDraw->drawTriangle (swV+com, seV+com, nwV+com, color, btScalar(1.0f));
debugDraw->drawTriangle (seV+com, neV+com, nwV+com, color, btScalar(1.0f));
}
}
}
void btHfFluidRigidDynamicsWorld::drawHfFluidVelocity (btIDebugDraw* debugDraw, btHfFluid* fluid)
{
btScalar alpha(0.7f);
const btScalar* height = fluid->getHeightArray ();
btVector3 com = fluid->getWorldTransform().getOrigin();
btVector3 red = btVector3(btScalar(1.0f), btScalar(0.0f), btScalar(0.0));
btVector3 green = btVector3(btScalar(0.0f), btScalar(1.0f), btScalar(0.0));
const bool* flags = fluid->getFlagsArray ();
for (int i = 1; i < fluid->getNumNodesWidth()-1; i++)
{
for (int j = 1; j < fluid->getNumNodesLength()-1; j++)
{
int index = fluid->arrayIndex (i, j);
if (!flags[index])
continue;
btVector3 from = btVector3 (fluid->widthPos (i), height[index]+btScalar(0.1f), fluid->lengthPos (j));
btVector3 velocity;
velocity.setX (fluid->getVelocityUArray()[index]);
velocity.setY (btScalar(0.0f));
velocity.setZ (fluid->getVelocityVArray()[index]);
velocity.normalize();
btVector3 to = from + velocity;
debugDraw->drawLine (from+com, to+com, red, green);
}
}
}
void btHfFluidRigidDynamicsWorld::drawHfFluidBuoyantConvexShape (btIDebugDraw* debugDrawer, btCollisionObject* object, btHfFluidBuoyantConvexShape* buoyantShape, int voxelDraw)
{
if (voxelDraw)
{
btTransform xform = object->getWorldTransform();
for (int i = 0; i < buoyantShape->getNumVoxels(); i++)
{
btVector3 p = buoyantShape->getVoxelPositionsArray()[i];
p = xform.getBasis() * p;
p += xform.getOrigin();
debugDrawer->drawSphere (p, buoyantShape->getVoxelRadius(), btVector3(1.0, 0.0, 0.0));
}
} else {
btVector3 color(btScalar(255.),btScalar(255.),btScalar(255.));
switch(object->getActivationState())
{
case ACTIVE_TAG:
color = btVector3(btScalar(255.),btScalar(255.),btScalar(255.)); break;
case ISLAND_SLEEPING:
color = btVector3(btScalar(0.),btScalar(255.),btScalar(0.));break;
case WANTS_DEACTIVATION:
color = btVector3(btScalar(0.),btScalar(255.),btScalar(255.));break;
case DISABLE_DEACTIVATION:
color = btVector3(btScalar(255.),btScalar(0.),btScalar(0.));break;
case DISABLE_SIMULATION:
color = btVector3(btScalar(255.),btScalar(255.),btScalar(0.));break;
default:
{
color = btVector3(btScalar(255.),btScalar(0.),btScalar(0.));
}
};
btConvexShape* convexShape = ((btHfFluidBuoyantConvexShape*)object->getCollisionShape())->getConvexShape();
debugDrawObject(object->getWorldTransform(),(btCollisionShape*)convexShape,color);
}
}
void btHfFluidRigidDynamicsWorld::drawHfFluidNormal (btIDebugDraw* debugDraw, btHfFluid* fluid)
{
int colIndex = 0;
btVector3 col[2];
col[0] = btVector3(btScalar(0.0f), btScalar(0.0f), btScalar(1.0));
col[1] = btVector3(btScalar(0.0f), btScalar(0.5f), btScalar(0.5));
btScalar alpha(0.7f);
const btScalar* height = fluid->getHeightArray ();
const btScalar* eta = fluid->getEtaArray ();
const btScalar* ground = fluid->getGroundArray ();
btVector3 com = fluid->getWorldTransform().getOrigin();
const bool* flags = fluid->getFlagsArray ();
for (int i = 0; i < fluid->getNumNodesWidth()-1; i++)
{
for (int j = 0; j < fluid->getNumNodesLength()-1; j++)
{
int sw = fluid->arrayIndex (i, j);
int se = fluid->arrayIndex (i+1, j);
int nw = fluid->arrayIndex (i, j+1);
int ne = fluid->arrayIndex (i+1, j+1);
btScalar h = eta[sw];
btScalar g = ground[sw];
if (h < btScalar(0.05f))
continue;
if (h <= btScalar(0.01f))
continue;
btVector3 boxMin = btVector3(fluid->widthPos (i), g, fluid->lengthPos(j));
btVector3 boxMax = btVector3(fluid->widthPos(i+1), g+h, fluid->lengthPos(j+1));
boxMin += com;
boxMax += com;
debugDraw->drawBox (boxMin, boxMax, btVector3(btScalar(0.0f), btScalar(0.0f), btScalar(1.0f)));
}
}
}
void btHfFluidRigidDynamicsWorld::debugDrawWorld()
{
if (getDebugDrawer())
{
int i;
for ( i=0;i<this->m_hfFluids.size();i++)
{
btHfFluid* phh=(btHfFluid*)this->m_hfFluids[i];
switch (m_drawMode)
{
case DRAWMODE_NORMAL:
drawHfFluidGround (m_debugDrawer, phh);
//drawHfFluidNormal (m_debugDrawer, phh);
break;
case DRAWMODE_VELOCITY:
drawHfFluidGround (m_debugDrawer, phh);
//drawHfFluidNormal (m_debugDrawer, phh);
drawHfFluidVelocity (m_debugDrawer, phh);
break;
default:
btAssert (0);
break;
}
}
for (i = 0; i < this->m_collisionObjects.size(); i++)
{
btCollisionShape* shape = m_collisionObjects[i]->getCollisionShape();
if (shape->getShapeType() == HFFLUID_BUOYANT_CONVEX_SHAPE_PROXYTYPE)
{
btHfFluidBuoyantConvexShape* buoyantShape = (btHfFluidBuoyantConvexShape*)shape;
drawHfFluidBuoyantConvexShape (m_debugDrawer, m_collisionObjects[i], buoyantShape, m_bodyDrawMode);
}
}
}
btDiscreteDynamicsWorld::debugDrawWorld();
}