1
0
Fork 0
mirror of https://github.com/OpenMW/openmw.git synced 2025-01-22 17:53:53 +00:00
openmw/libs/openengine/bullet/BtOgre.cpp
2012-07-03 15:32:38 +02:00

1094 lines
38 KiB
C++

/*
* =============================================================================================
*
* Filename: BtOgre.cpp
*
* Description: BtOgre implementation.
*
* Version: 1.0
* Created: 27/12/2008 01:47:56 PM
*
* Author: Nikhilesh (nikki)
*
* =============================================================================================
*/
#include "BtOgrePG.h"
#include "BtOgreGP.h"
#include "BtOgreExtras.h"
#include <OgreEntity.h>
#include <OgreSubMesh.h>
#include <OgreSubEntity.h>
using namespace Ogre;
namespace BtOgre {
/*
* =============================================================================================
* BtOgre::VertexIndexToShape
* =============================================================================================
*/
void VertexIndexToShape::addStaticVertexData(const VertexData *vertex_data)
{
if (!vertex_data)
return;
const VertexData *data = vertex_data;
const unsigned int prev_size = mVertexCount;
mVertexCount += (unsigned int)data->vertexCount;
Ogre::Vector3* tmp_vert = new Ogre::Vector3[mVertexCount];
if (mVertexBuffer)
{
memcpy(tmp_vert, mVertexBuffer, sizeof(Vector3) * prev_size);
delete[] mVertexBuffer;
}
mVertexBuffer = tmp_vert;
// Get the positional buffer element
{
const Ogre::VertexElement* posElem = data->vertexDeclaration->findElementBySemantic(Ogre::VES_POSITION);
Ogre::HardwareVertexBufferSharedPtr vbuf = data->vertexBufferBinding->getBuffer(posElem->getSource());
const unsigned int vSize = (unsigned int)vbuf->getVertexSize();
unsigned char* vertex = static_cast<unsigned char*>(vbuf->lock(Ogre::HardwareBuffer::HBL_READ_ONLY));
float* pReal;
Ogre::Vector3 * curVertices = &mVertexBuffer[prev_size];
const unsigned int vertexCount = (unsigned int)data->vertexCount;
for(unsigned int j = 0; j < vertexCount; ++j)
{
posElem->baseVertexPointerToElement(vertex, &pReal);
vertex += vSize;
curVertices->x = (*pReal++);
curVertices->y = (*pReal++);
curVertices->z = (*pReal++);
*curVertices = mTransform * (*curVertices);
curVertices++;
}
vbuf->unlock();
}
}
//------------------------------------------------------------------------------------------------
void VertexIndexToShape::addAnimatedVertexData(const Ogre::VertexData *vertex_data,
const Ogre::VertexData *blend_data,
const Ogre::Mesh::IndexMap *indexMap)
{
// Get the bone index element
assert(vertex_data);
const VertexData *data = blend_data;
const unsigned int prev_size = mVertexCount;
mVertexCount += (unsigned int)data->vertexCount;
Ogre::Vector3* tmp_vert = new Ogre::Vector3[mVertexCount];
if (mVertexBuffer)
{
memcpy(tmp_vert, mVertexBuffer, sizeof(Vector3) * prev_size);
delete[] mVertexBuffer;
}
mVertexBuffer = tmp_vert;
// Get the positional buffer element
{
const Ogre::VertexElement* posElem = data->vertexDeclaration->findElementBySemantic(Ogre::VES_POSITION);
assert (posElem);
Ogre::HardwareVertexBufferSharedPtr vbuf = data->vertexBufferBinding->getBuffer(posElem->getSource());
const unsigned int vSize = (unsigned int)vbuf->getVertexSize();
unsigned char* vertex = static_cast<unsigned char*>(vbuf->lock(Ogre::HardwareBuffer::HBL_READ_ONLY));
float* pReal;
Ogre::Vector3 * curVertices = &mVertexBuffer[prev_size];
const unsigned int vertexCount = (unsigned int)data->vertexCount;
for(unsigned int j = 0; j < vertexCount; ++j)
{
posElem->baseVertexPointerToElement(vertex, &pReal);
vertex += vSize;
curVertices->x = (*pReal++);
curVertices->y = (*pReal++);
curVertices->z = (*pReal++);
*curVertices = mTransform * (*curVertices);
curVertices++;
}
vbuf->unlock();
}
{
const Ogre::VertexElement* bneElem = vertex_data->vertexDeclaration->findElementBySemantic(Ogre::VES_BLEND_INDICES);
assert (bneElem);
Ogre::HardwareVertexBufferSharedPtr vbuf = vertex_data->vertexBufferBinding->getBuffer(bneElem->getSource());
const unsigned int vSize = (unsigned int)vbuf->getVertexSize();
unsigned char* vertex = static_cast<unsigned char*>(vbuf->lock(Ogre::HardwareBuffer::HBL_READ_ONLY));
unsigned char* pBone;
if (!mBoneIndex)
mBoneIndex = new BoneIndex();
BoneIndex::iterator i;
Ogre::Vector3 * curVertices = &mVertexBuffer[prev_size];
const unsigned int vertexCount = (unsigned int)vertex_data->vertexCount;
for(unsigned int j = 0; j < vertexCount; ++j)
{
bneElem->baseVertexPointerToElement(vertex, &pBone);
vertex += vSize;
const unsigned char currBone = (indexMap) ? (*indexMap)[*pBone] : *pBone;
i = mBoneIndex->find (currBone);
Vector3Array* l = 0;
if (i == mBoneIndex->end())
{
l = new Vector3Array;
mBoneIndex->insert(BoneKeyIndex(currBone, l));
}
else
{
l = i->second;
}
l->push_back(*curVertices);
curVertices++;
}
vbuf->unlock();
}
}
//------------------------------------------------------------------------------------------------
void VertexIndexToShape::addIndexData(IndexData *data, const unsigned int offset)
{
const unsigned int prev_size = mIndexCount;
mIndexCount += (unsigned int)data->indexCount;
unsigned int* tmp_ind = new unsigned int[mIndexCount];
if (mIndexBuffer)
{
memcpy (tmp_ind, mIndexBuffer, sizeof(unsigned int) * prev_size);
delete[] mIndexBuffer;
}
mIndexBuffer = tmp_ind;
const unsigned int numTris = (unsigned int) data->indexCount / 3;
HardwareIndexBufferSharedPtr ibuf = data->indexBuffer;
const bool use32bitindexes = (ibuf->getType() == HardwareIndexBuffer::IT_32BIT);
unsigned int index_offset = prev_size;
if (use32bitindexes)
{
const unsigned int* pInt = static_cast<unsigned int*>(ibuf->lock(HardwareBuffer::HBL_READ_ONLY));
for(unsigned int k = 0; k < numTris; ++k)
{
mIndexBuffer[index_offset ++] = offset + *pInt++;
mIndexBuffer[index_offset ++] = offset + *pInt++;
mIndexBuffer[index_offset ++] = offset + *pInt++;
}
ibuf->unlock();
}
else
{
const unsigned short* pShort = static_cast<unsigned short*>(ibuf->lock(HardwareBuffer::HBL_READ_ONLY));
for(unsigned int k = 0; k < numTris; ++k)
{
mIndexBuffer[index_offset ++] = offset + static_cast<unsigned int> (*pShort++);
mIndexBuffer[index_offset ++] = offset + static_cast<unsigned int> (*pShort++);
mIndexBuffer[index_offset ++] = offset + static_cast<unsigned int> (*pShort++);
}
ibuf->unlock();
}
}
//------------------------------------------------------------------------------------------------
Real VertexIndexToShape::getRadius()
{
if (mBoundRadius == (-1))
{
getSize();
mBoundRadius = (std::max(mBounds.x,std::max(mBounds.y,mBounds.z)) * 0.5);
}
return mBoundRadius;
}
//------------------------------------------------------------------------------------------------
Vector3 VertexIndexToShape::getSize()
{
const unsigned int vCount = getVertexCount();
if (mBounds == Ogre::Vector3(-1,-1,-1) && vCount > 0)
{
const Ogre::Vector3 * const v = getVertices();
Ogre::Vector3 vmin(v[0]);
Ogre::Vector3 vmax(v[0]);
for(unsigned int j = 1; j < vCount; j++)
{
vmin.x = std::min(vmin.x, v[j].x);
vmin.y = std::min(vmin.y, v[j].y);
vmin.z = std::min(vmin.z, v[j].z);
vmax.x = std::max(vmax.x, v[j].x);
vmax.y = std::max(vmax.y, v[j].y);
vmax.z = std::max(vmax.z, v[j].z);
}
mBounds.x = vmax.x - vmin.x;
mBounds.y = vmax.y - vmin.y;
mBounds.z = vmax.z - vmin.z;
}
return mBounds;
}
//------------------------------------------------------------------------------------------------
const Ogre::Vector3* VertexIndexToShape::getVertices()
{
return mVertexBuffer;
}
//------------------------------------------------------------------------------------------------
unsigned int VertexIndexToShape::getVertexCount()
{
return mVertexCount;
}
//------------------------------------------------------------------------------------------------
const unsigned int* VertexIndexToShape::getIndices()
{
return mIndexBuffer;
}
//------------------------------------------------------------------------------------------------
unsigned int VertexIndexToShape::getIndexCount()
{
return mIndexCount;
}
//------------------------------------------------------------------------------------------------
btSphereShape* VertexIndexToShape::createSphere()
{
const Ogre::Real rad = getRadius();
assert((rad > 0.0) &&
("Sphere radius must be greater than zero"));
btSphereShape* shape = new btSphereShape(rad);
shape->setLocalScaling(Convert::toBullet(mScale));
return shape;
}
//------------------------------------------------------------------------------------------------
btBoxShape* VertexIndexToShape::createBox()
{
const Ogre::Vector3 sz = getSize();
assert((sz.x > 0.0) && (sz.y > 0.0) && (sz.y > 0.0) &&
("Size of box must be greater than zero on all axes"));
btBoxShape* shape = new btBoxShape(Convert::toBullet(sz * 0.5));
shape->setLocalScaling(Convert::toBullet(mScale));
return shape;
}
//------------------------------------------------------------------------------------------------
btCylinderShape* VertexIndexToShape::createCylinder()
{
const Ogre::Vector3 sz = getSize();
assert((sz.x > 0.0) && (sz.y > 0.0) && (sz.y > 0.0) &&
("Size of Cylinder must be greater than zero on all axes"));
btCylinderShape* shape = new btCylinderShapeX(Convert::toBullet(sz * 0.5));
shape->setLocalScaling(Convert::toBullet(mScale));
return shape;
}
//------------------------------------------------------------------------------------------------
btConvexHullShape* VertexIndexToShape::createConvex()
{
assert(mVertexCount && (mIndexCount >= 6) &&
("Mesh must have some vertices and at least 6 indices (2 triangles)"));
return new btConvexHullShape((btScalar*) &mVertexBuffer[0].x, mVertexCount, sizeof(Vector3));
}
//------------------------------------------------------------------------------------------------
btBvhTriangleMeshShape* VertexIndexToShape::createTrimesh()
{
assert(mVertexCount && (mIndexCount >= 6) &&
("Mesh must have some vertices and at least 6 indices (2 triangles)"));
unsigned int numFaces = mIndexCount / 3;
btTriangleMesh *trimesh = new btTriangleMesh();
unsigned int *indices = mIndexBuffer;
Vector3 *vertices = mVertexBuffer;
btVector3 vertexPos[3];
for (unsigned int n = 0; n < numFaces; ++n)
{
{
const Vector3 &vec = vertices[*indices];
vertexPos[0][0] = vec.x;
vertexPos[0][1] = vec.y;
vertexPos[0][2] = vec.z;
}
{
const Vector3 &vec = vertices[*(indices + 1)];
vertexPos[1][0] = vec.x;
vertexPos[1][1] = vec.y;
vertexPos[1][2] = vec.z;
}
{
const Vector3 &vec = vertices[*(indices + 2)];
vertexPos[2][0] = vec.x;
vertexPos[2][1] = vec.y;
vertexPos[2][2] = vec.z;
}
indices += 3;
trimesh->addTriangle(vertexPos[0], vertexPos[1], vertexPos[2]);
}
const bool useQuantizedAABB = true;
btBvhTriangleMeshShape *shape = new btBvhTriangleMeshShape(trimesh, useQuantizedAABB);
shape->setLocalScaling(Convert::toBullet(mScale));
return shape;
}
//------------------------------------------------------------------------------------------------
VertexIndexToShape::~VertexIndexToShape()
{
delete[] mVertexBuffer;
delete[] mIndexBuffer;
if (mBoneIndex)
{
for(BoneIndex::iterator i = mBoneIndex->begin();
i != mBoneIndex->end();
++i)
{
delete i->second;
}
delete mBoneIndex;
}
}
//------------------------------------------------------------------------------------------------
VertexIndexToShape::VertexIndexToShape(const Matrix4 &transform) :
mVertexBuffer (0),
mIndexBuffer (0),
mVertexCount (0),
mIndexCount (0),
mTransform (transform),
mBoundRadius (-1),
mBounds (Vector3(-1,-1,-1)),
mBoneIndex (0),
mScale(1)
{
}
/*
* =============================================================================================
* BtOgre::StaticMeshToShapeConverter
* =============================================================================================
*/
StaticMeshToShapeConverter::StaticMeshToShapeConverter() :
VertexIndexToShape(),
mEntity (0),
mNode (0)
{
}
//------------------------------------------------------------------------------------------------
StaticMeshToShapeConverter::~StaticMeshToShapeConverter()
{
}
//------------------------------------------------------------------------------------------------
StaticMeshToShapeConverter::StaticMeshToShapeConverter(Entity *entity, const Matrix4 &transform) :
VertexIndexToShape(transform),
mEntity (0),
mNode (0)
{
addEntity(entity, transform);
}
//------------------------------------------------------------------------------------------------
StaticMeshToShapeConverter::StaticMeshToShapeConverter(Renderable *rend, const Matrix4 &transform) :
VertexIndexToShape(transform),
mEntity (0),
mNode (0)
{
RenderOperation op;
rend->getRenderOperation(op);
VertexIndexToShape::addStaticVertexData(op.vertexData);
if(op.useIndexes)
VertexIndexToShape::addIndexData(op.indexData);
}
//------------------------------------------------------------------------------------------------
void StaticMeshToShapeConverter::addEntity(Entity *entity,const Matrix4 &transform)
{
// Each entity added need to reset size and radius
// next time getRadius and getSize are asked, they're computed.
mBounds = Ogre::Vector3(-1,-1,-1);
mBoundRadius = -1;
mEntity = entity;
mNode = (SceneNode*)(mEntity->getParentNode());
mTransform = transform;
mScale = mNode->getScale();
if (mEntity->getMesh()->sharedVertexData)
{
VertexIndexToShape::addStaticVertexData (mEntity->getMesh()->sharedVertexData);
}
for (unsigned int i = 0;i < mEntity->getNumSubEntities();++i)
{
SubMesh *sub_mesh = mEntity->getSubEntity(i)->getSubMesh();
if (!sub_mesh->useSharedVertices)
{
VertexIndexToShape::addIndexData(sub_mesh->indexData, mVertexCount);
VertexIndexToShape::addStaticVertexData (sub_mesh->vertexData);
}
else
{
VertexIndexToShape::addIndexData (sub_mesh->indexData);
}
}
}
//------------------------------------------------------------------------------------------------
void StaticMeshToShapeConverter::addMesh(const MeshPtr &mesh, const Matrix4 &transform)
{
// Each entity added need to reset size and radius
// next time getRadius and getSize are asked, they're computed.
mBounds = Ogre::Vector3(-1,-1,-1);
mBoundRadius = -1;
//_entity = entity;
//_node = (SceneNode*)(_entity->getParentNode());
mTransform = transform;
if (mesh->hasSkeleton ())
Ogre::LogManager::getSingleton().logMessage("MeshToShapeConverter::addMesh : Mesh " + mesh->getName () + " as skeleton but added to trimesh non animated");
if (mesh->sharedVertexData)
{
VertexIndexToShape::addStaticVertexData (mesh->sharedVertexData);
}
for(unsigned int i = 0;i < mesh->getNumSubMeshes();++i)
{
SubMesh *sub_mesh = mesh->getSubMesh(i);
if (!sub_mesh->useSharedVertices)
{
VertexIndexToShape::addIndexData(sub_mesh->indexData, mVertexCount);
VertexIndexToShape::addStaticVertexData (sub_mesh->vertexData);
}
else
{
VertexIndexToShape::addIndexData (sub_mesh->indexData);
}
}
}
/*
* =============================================================================================
* BtOgre::AnimatedMeshToShapeConverter
* =============================================================================================
*/
AnimatedMeshToShapeConverter::AnimatedMeshToShapeConverter(Entity *entity,const Matrix4 &transform) :
VertexIndexToShape(transform),
mEntity (0),
mNode (0),
mTransformedVerticesTemp(0),
mTransformedVerticesTempSize(0)
{
addEntity(entity, transform);
}
//------------------------------------------------------------------------------------------------
AnimatedMeshToShapeConverter::AnimatedMeshToShapeConverter() :
VertexIndexToShape(),
mEntity (0),
mNode (0),
mTransformedVerticesTemp(0),
mTransformedVerticesTempSize(0)
{
}
//------------------------------------------------------------------------------------------------
AnimatedMeshToShapeConverter::~AnimatedMeshToShapeConverter()
{
delete[] mTransformedVerticesTemp;
}
//------------------------------------------------------------------------------------------------
void AnimatedMeshToShapeConverter::addEntity(Entity *entity,const Matrix4 &transform)
{
// Each entity added need to reset size and radius
// next time getRadius and getSize are asked, they're computed.
mBounds = Ogre::Vector3(-1,-1,-1);
mBoundRadius = -1;
mEntity = entity;
mNode = (SceneNode*)(mEntity->getParentNode());
mTransform = transform;
assert (entity->getMesh()->hasSkeleton ());
mEntity->addSoftwareAnimationRequest(false);
mEntity->_updateAnimation();
if (mEntity->getMesh()->sharedVertexData)
{
VertexIndexToShape::addAnimatedVertexData (mEntity->getMesh()->sharedVertexData,
mEntity->_getSkelAnimVertexData(),
&mEntity->getMesh()->sharedBlendIndexToBoneIndexMap);
}
for (unsigned int i = 0;i < mEntity->getNumSubEntities();++i)
{
SubMesh *sub_mesh = mEntity->getSubEntity(i)->getSubMesh();
if (!sub_mesh->useSharedVertices)
{
VertexIndexToShape::addIndexData(sub_mesh->indexData, mVertexCount);
VertexIndexToShape::addAnimatedVertexData (sub_mesh->vertexData,
mEntity->getSubEntity(i)->_getSkelAnimVertexData(),
&sub_mesh->blendIndexToBoneIndexMap);
}
else
{
VertexIndexToShape::addIndexData (sub_mesh->indexData);
}
}
mEntity->removeSoftwareAnimationRequest(false);
}
//------------------------------------------------------------------------------------------------
void AnimatedMeshToShapeConverter::addMesh(const MeshPtr &mesh, const Matrix4 &transform)
{
// Each entity added need to reset size and radius
// next time getRadius and getSize are asked, they're computed.
mBounds = Ogre::Vector3(-1,-1,-1);
mBoundRadius = -1;
//_entity = entity;
//_node = (SceneNode*)(_entity->getParentNode());
mTransform = transform;
assert (mesh->hasSkeleton ());
if (mesh->sharedVertexData)
{
VertexIndexToShape::addAnimatedVertexData (mesh->sharedVertexData,
0,
&mesh->sharedBlendIndexToBoneIndexMap);
}
for(unsigned int i = 0;i < mesh->getNumSubMeshes();++i)
{
SubMesh *sub_mesh = mesh->getSubMesh(i);
if (!sub_mesh->useSharedVertices)
{
VertexIndexToShape::addIndexData(sub_mesh->indexData, mVertexCount);
VertexIndexToShape::addAnimatedVertexData (sub_mesh->vertexData,
0,
&sub_mesh->blendIndexToBoneIndexMap);
}
else
{
VertexIndexToShape::addIndexData (sub_mesh->indexData);
}
}
}
//------------------------------------------------------------------------------------------------
bool AnimatedMeshToShapeConverter::getBoneVertices(unsigned char bone,
unsigned int &vertex_count,
Ogre::Vector3* &vertices,
const Vector3 &bonePosition)
{
BoneIndex::iterator i = mBoneIndex->find(bone);
if (i == mBoneIndex->end())
return false;
if (i->second->empty())
return false;
vertex_count = (unsigned int) i->second->size() + 1;
if (vertex_count > mTransformedVerticesTempSize)
{
if (mTransformedVerticesTemp)
delete[] mTransformedVerticesTemp;
mTransformedVerticesTemp = new Ogre::Vector3[vertex_count];
}
vertices = mTransformedVerticesTemp;
vertices[0] = bonePosition;
//mEntity->_getParentNodeFullTransform() *
//mEntity->getSkeleton()->getBone(bone)->_getDerivedPosition();
//mEntity->getSkeleton()->getBone(bone)->_getDerivedOrientation()
unsigned int currBoneVertex = 1;
Vector3Array::iterator j = i->second->begin();
while(j != i->second->end())
{
vertices[currBoneVertex] = (*j);
++j;
++currBoneVertex;
}
return true;
}
//------------------------------------------------------------------------------------------------
btBoxShape* AnimatedMeshToShapeConverter::createAlignedBox(unsigned char bone,
const Vector3 &bonePosition,
const Quaternion &boneOrientation)
{
unsigned int vertex_count;
Vector3* vertices;
if (!getBoneVertices(bone, vertex_count, vertices, bonePosition))
return 0;
Vector3 min_vec(vertices[0]);
Vector3 max_vec(vertices[0]);
for(unsigned int j = 1; j < vertex_count ;j++)
{
min_vec.x = std::min(min_vec.x,vertices[j].x);
min_vec.y = std::min(min_vec.y,vertices[j].y);
min_vec.z = std::min(min_vec.z,vertices[j].z);
max_vec.x = std::max(max_vec.x,vertices[j].x);
max_vec.y = std::max(max_vec.y,vertices[j].y);
max_vec.z = std::max(max_vec.z,vertices[j].z);
}
const Ogre::Vector3 maxMinusMin(max_vec - min_vec);
btBoxShape* box = new btBoxShape(Convert::toBullet(maxMinusMin));
/*const Ogre::Vector3 pos
(min_vec.x + (maxMinusMin.x * 0.5),
min_vec.y + (maxMinusMin.y * 0.5),
min_vec.z + (maxMinusMin.z * 0.5));*/
//box->setPosition(pos);
return box;
}
//------------------------------------------------------------------------------------------------
bool AnimatedMeshToShapeConverter::getOrientedBox(unsigned char bone,
const Vector3 &bonePosition,
const Quaternion &boneOrientation,
Vector3 &box_afExtent,
Vector3 *box_akAxis,
Vector3 &box_kCenter)
{
unsigned int vertex_count;
Vector3* vertices;
if (!getBoneVertices(bone, vertex_count, vertices, bonePosition))
return false;
box_kCenter = Vector3::ZERO;
{
for(unsigned int c = 0 ;c < vertex_count;c++)
{
box_kCenter += vertices[c];
}
const Ogre::Real invVertexCount = 1.0 / vertex_count;
box_kCenter *= invVertexCount;
}
Quaternion orient = boneOrientation;
orient.ToAxes(box_akAxis);
// Let C be the box center and let U0, U1, and U2 be the box axes. Each
// input point is of the form X = C + y0*U0 + y1*U1 + y2*U2. The
// following code computes min(y0), max(y0), min(y1), max(y1), min(y2),
// and max(y2). The box center is then adjusted to be
// C' = C + 0.5*(min(y0)+max(y0))*U0 + 0.5*(min(y1)+max(y1))*U1 +
// 0.5*(min(y2)+max(y2))*U2
Ogre::Vector3 kDiff (vertices[1] - box_kCenter);
Ogre::Real fY0Min = kDiff.dotProduct(box_akAxis[0]), fY0Max = fY0Min;
Ogre::Real fY1Min = kDiff.dotProduct(box_akAxis[1]), fY1Max = fY1Min;
Ogre::Real fY2Min = kDiff.dotProduct(box_akAxis[2]), fY2Max = fY2Min;
for (unsigned int i = 2; i < vertex_count; i++)
{
kDiff = vertices[i] - box_kCenter;
const Ogre::Real fY0 = kDiff.dotProduct(box_akAxis[0]);
if ( fY0 < fY0Min )
fY0Min = fY0;
else if ( fY0 > fY0Max )
fY0Max = fY0;
const Ogre::Real fY1 = kDiff.dotProduct(box_akAxis[1]);
if ( fY1 < fY1Min )
fY1Min = fY1;
else if ( fY1 > fY1Max )
fY1Max = fY1;
const Ogre::Real fY2 = kDiff.dotProduct(box_akAxis[2]);
if ( fY2 < fY2Min )
fY2Min = fY2;
else if ( fY2 > fY2Max )
fY2Max = fY2;
}
box_afExtent.x = ((Real)0.5)*(fY0Max - fY0Min);
box_afExtent.y = ((Real)0.5)*(fY1Max - fY1Min);
box_afExtent.z = ((Real)0.5)*(fY2Max - fY2Min);
box_kCenter += (0.5*(fY0Max+fY0Min))*box_akAxis[0] +
(0.5*(fY1Max+fY1Min))*box_akAxis[1] +
(0.5*(fY2Max+fY2Min))*box_akAxis[2];
box_afExtent *= 2.0;
return true;
}
//------------------------------------------------------------------------------------------------
btBoxShape *AnimatedMeshToShapeConverter::createOrientedBox(unsigned char bone,
const Vector3 &bonePosition,
const Quaternion &boneOrientation)
{
Ogre::Vector3 box_akAxis[3];
Ogre::Vector3 box_afExtent;
Ogre::Vector3 box_afCenter;
if (!getOrientedBox(bone, bonePosition, boneOrientation,
box_afExtent,
box_akAxis,
box_afCenter))
return 0;
btBoxShape *geom = new btBoxShape(Convert::toBullet(box_afExtent));
//geom->setOrientation(Quaternion(box_akAxis[0],box_akAxis[1],box_akAxis[2]));
//geom->setPosition(box_afCenter);
return geom;
}
/*
* =============================================================================================
* BtOgre::DynamicRenderable
* =============================================================================================
*/
DynamicRenderable::DynamicRenderable()
{
}
//------------------------------------------------------------------------------------------------
DynamicRenderable::~DynamicRenderable()
{
delete mRenderOp.vertexData;
delete mRenderOp.indexData;
}
//------------------------------------------------------------------------------------------------
void DynamicRenderable::initialize(RenderOperation::OperationType operationType,
bool useIndices)
{
// Initialize render operation
mRenderOp.operationType = operationType;
mRenderOp.useIndexes = useIndices;
mRenderOp.vertexData = new VertexData;
if (mRenderOp.useIndexes)
mRenderOp.indexData = new IndexData;
// Reset buffer capacities
mVertexBufferCapacity = 0;
mIndexBufferCapacity = 0;
// Create vertex declaration
createVertexDeclaration();
}
//------------------------------------------------------------------------------------------------
void DynamicRenderable::prepareHardwareBuffers(size_t vertexCount,
size_t indexCount)
{
// Prepare vertex buffer
size_t newVertCapacity = mVertexBufferCapacity;
if ((vertexCount > mVertexBufferCapacity) ||
(!mVertexBufferCapacity))
{
// vertexCount exceeds current capacity!
// It is necessary to reallocate the buffer.
// Check if this is the first call
if (!newVertCapacity)
newVertCapacity = 1;
// Make capacity the next power of two
while (newVertCapacity < vertexCount)
newVertCapacity <<= 1;
}
else if (vertexCount < mVertexBufferCapacity>>1) {
// Make capacity the previous power of two
while (vertexCount < newVertCapacity>>1)
newVertCapacity >>= 1;
}
if (newVertCapacity != mVertexBufferCapacity)
{
mVertexBufferCapacity = newVertCapacity;
// Create new vertex buffer
HardwareVertexBufferSharedPtr vbuf =
HardwareBufferManager::getSingleton().createVertexBuffer(
mRenderOp.vertexData->vertexDeclaration->getVertexSize(0),
mVertexBufferCapacity,
HardwareBuffer::HBU_DYNAMIC_WRITE_ONLY); // TODO: Custom HBU_?
// Bind buffer
mRenderOp.vertexData->vertexBufferBinding->setBinding(0, vbuf);
}
// Update vertex count in the render operation
mRenderOp.vertexData->vertexCount = vertexCount;
if (mRenderOp.useIndexes)
{
OgreAssert(indexCount <= std::numeric_limits<unsigned short>::max(), "indexCount exceeds 16 bit");
size_t newIndexCapacity = mIndexBufferCapacity;
// Prepare index buffer
if ((indexCount > newIndexCapacity) ||
(!newIndexCapacity))
{
// indexCount exceeds current capacity!
// It is necessary to reallocate the buffer.
// Check if this is the first call
if (!newIndexCapacity)
newIndexCapacity = 1;
// Make capacity the next power of two
while (newIndexCapacity < indexCount)
newIndexCapacity <<= 1;
}
else if (indexCount < newIndexCapacity>>1)
{
// Make capacity the previous power of two
while (indexCount < newIndexCapacity>>1)
newIndexCapacity >>= 1;
}
if (newIndexCapacity != mIndexBufferCapacity)
{
mIndexBufferCapacity = newIndexCapacity;
// Create new index buffer
mRenderOp.indexData->indexBuffer =
HardwareBufferManager::getSingleton().createIndexBuffer(
HardwareIndexBuffer::IT_16BIT,
mIndexBufferCapacity,
HardwareBuffer::HBU_DYNAMIC_WRITE_ONLY); // TODO: Custom HBU_?
}
// Update index count in the render operation
mRenderOp.indexData->indexCount = indexCount;
}
}
//------------------------------------------------------------------------------------------------
Real DynamicRenderable::getBoundingRadius(void) const
{
return Math::Sqrt(std::max(mBox.getMaximum().squaredLength(), mBox.getMinimum().squaredLength()));
}
//------------------------------------------------------------------------------------------------
Real DynamicRenderable::getSquaredViewDepth(const Camera* cam) const
{
Vector3 vMin, vMax, vMid, vDist;
vMin = mBox.getMinimum();
vMax = mBox.getMaximum();
vMid = ((vMax - vMin) * 0.5) + vMin;
vDist = cam->getDerivedPosition() - vMid;
return vDist.squaredLength();
}
/*
* =============================================================================================
* BtOgre::DynamicLines
* =============================================================================================
*/
enum {
POSITION_BINDING,
TEXCOORD_BINDING
};
//------------------------------------------------------------------------------------------------
DynamicLines::DynamicLines(OperationType opType)
{
initialize(opType,false);
setMaterial("BaseWhiteNoLighting");
mDirty = true;
}
//------------------------------------------------------------------------------------------------
DynamicLines::~DynamicLines()
{
}
//------------------------------------------------------------------------------------------------
void DynamicLines::setOperationType(OperationType opType)
{
mRenderOp.operationType = opType;
}
//------------------------------------------------------------------------------------------------
RenderOperation::OperationType DynamicLines::getOperationType() const
{
return mRenderOp.operationType;
}
//------------------------------------------------------------------------------------------------
void DynamicLines::addPoint(const Vector3 &p)
{
mPoints.push_back(p);
mDirty = true;
}
//------------------------------------------------------------------------------------------------
void DynamicLines::addPoint(Real x, Real y, Real z)
{
mPoints.push_back(Vector3(x,y,z));
mDirty = true;
}
//------------------------------------------------------------------------------------------------
const Vector3& DynamicLines::getPoint(unsigned short index) const
{
assert(index < mPoints.size() && "Point index is out of bounds!!");
return mPoints[index];
}
//------------------------------------------------------------------------------------------------
unsigned short DynamicLines::getNumPoints(void) const
{
return (unsigned short)mPoints.size();
}
//------------------------------------------------------------------------------------------------
void DynamicLines::setPoint(unsigned short index, const Vector3 &value)
{
assert(index < mPoints.size() && "Point index is out of bounds!!");
mPoints[index] = value;
mDirty = true;
}
//------------------------------------------------------------------------------------------------
void DynamicLines::clear()
{
mPoints.clear();
mDirty = true;
}
//------------------------------------------------------------------------------------------------
void DynamicLines::update()
{
if (mDirty) fillHardwareBuffers();
}
//------------------------------------------------------------------------------------------------
void DynamicLines::createVertexDeclaration()
{
VertexDeclaration *decl = mRenderOp.vertexData->vertexDeclaration;
decl->addElement(POSITION_BINDING, 0, VET_FLOAT3, VES_POSITION);
}
//------------------------------------------------------------------------------------------------
void DynamicLines::fillHardwareBuffers()
{
int size = mPoints.size();
prepareHardwareBuffers(size,0);
if (!size) {
mBox.setExtents(Vector3::ZERO,Vector3::ZERO);
mDirty=false;
return;
}
Vector3 vaabMin = mPoints[0];
Vector3 vaabMax = mPoints[0];
HardwareVertexBufferSharedPtr vbuf =
mRenderOp.vertexData->vertexBufferBinding->getBuffer(0);
Real *prPos = static_cast<Real*>(vbuf->lock(HardwareBuffer::HBL_DISCARD));
{
for(int i = 0; i < size; i++)
{
*prPos++ = mPoints[i].x;
*prPos++ = mPoints[i].y;
*prPos++ = mPoints[i].z;
if(mPoints[i].x < vaabMin.x)
vaabMin.x = mPoints[i].x;
if(mPoints[i].y < vaabMin.y)
vaabMin.y = mPoints[i].y;
if(mPoints[i].z < vaabMin.z)
vaabMin.z = mPoints[i].z;
if(mPoints[i].x > vaabMax.x)
vaabMax.x = mPoints[i].x;
if(mPoints[i].y > vaabMax.y)
vaabMax.y = mPoints[i].y;
if(mPoints[i].z > vaabMax.z)
vaabMax.z = mPoints[i].z;
}
}
vbuf->unlock();
mBox.setExtents(vaabMin, vaabMax);
mDirty = false;
}
}