mirror of
https://github.com/bulletphysics/bullet3
synced 2025-01-10 17:30:12 +00:00
5dd43ab3a2
TODO: errors while scene loading / saving
169 lines
6.5 KiB
C++
169 lines
6.5 KiB
C++
/*
|
|
Bullet Continuous Collision Detection and Physics Library Maya Plugin
|
|
Copyright (c) 2008 Walt Disney Studios
|
|
|
|
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.
|
|
|
|
Written by: Nicola Candussi <nicola@fluidinteractive.com>
|
|
*/
|
|
|
|
//bt_mesh_shape.h
|
|
|
|
#ifndef DYN_BT_MESH_SHAPE_H
|
|
#define DYN_BT_MESH_SHAPE_H
|
|
|
|
#include <vector>
|
|
#include <iterator>
|
|
|
|
#include "mesh_shape_impl.h"
|
|
#include "bt_collision_shape.h"
|
|
|
|
class bt_mesh_shape_t: public bt_collision_shape_t, public mesh_shape_impl_t
|
|
{
|
|
public:
|
|
virtual void gl_draw(size_t draw_style) {
|
|
|
|
if(m_vertices.empty() || m_indices.empty()) return;
|
|
|
|
btVector3 const& scale = shape()->getLocalScaling();
|
|
glPushMatrix();
|
|
glTranslatef(m_center[0], m_center[1], m_center[2]);
|
|
float angle;
|
|
vec3f axis;
|
|
q_to_axis_angle(m_rotation, axis, angle);
|
|
glRotatef(rad2deg(angle), axis[0], axis[1], axis[2]);
|
|
glScalef(scale.x(), scale.y(), scale.z());
|
|
glEnableClientState(GL_VERTEX_ARRAY);
|
|
glEnableClientState(GL_NORMAL_ARRAY);
|
|
if(draw_style & collision_shape_t::kDSSolid) {
|
|
glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
|
|
} else {
|
|
glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
|
|
}
|
|
glVertexPointer(3, GL_FLOAT, 0, &(m_vertices[0]));
|
|
glNormalPointer(GL_FLOAT, 0, &(m_normals[0]));
|
|
glDrawElements(GL_TRIANGLES, m_indices.size(), GL_UNSIGNED_INT, &(m_indices[0]));
|
|
glDisableClientState(GL_VERTEX_ARRAY);
|
|
glDisableClientState(GL_NORMAL_ARRAY);
|
|
glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
|
|
glPopMatrix();
|
|
}
|
|
|
|
virtual void set_scale(vec3f const& s) {
|
|
btGImpactMeshShape *gi_shape = static_cast<btGImpactMeshShape*>(shape());
|
|
gi_shape->setLocalScaling(btVector3(s[0], s[1], s[2]));
|
|
gi_shape->updateBound();
|
|
update();
|
|
}
|
|
|
|
virtual void get_scale(vec3f& s) {
|
|
const btVector3& scale = shape()->getLocalScaling();
|
|
s = vec3f(scale.x(), scale.y(), scale.z());
|
|
}
|
|
|
|
virtual float volume() { return m_volume; }
|
|
virtual vec3f const& local_inertia() { return m_local_inertia; }
|
|
virtual vec3f const& center() { return m_center; }
|
|
virtual quatf const& rotation() { return m_rotation; }
|
|
|
|
protected:
|
|
friend class bt_solver_t;
|
|
|
|
bt_mesh_shape_t(vec3f const* vertices, size_t num_vertices,
|
|
vec3f const* normals,
|
|
unsigned int const *indices, size_t num_indices):
|
|
bt_collision_shape_t(),
|
|
m_normals(normals, normals + num_vertices),
|
|
m_indices(indices, indices + num_indices)
|
|
{
|
|
m_volume = ::volume(vertices, (int*)indices, num_indices);
|
|
m_center = center_of_mass(vertices, (int*)indices, num_indices);
|
|
mat3x3f I = inertia(vertices, (int*)indices, num_indices, m_center);
|
|
m_rotation = diagonalizer(I);
|
|
|
|
mat3x3f Q, Qinv;
|
|
q_to_mat(m_rotation, Q);
|
|
q_to_mat(qconj(m_rotation), Qinv);
|
|
|
|
//D = trans(Q) * I * Q;
|
|
m_local_inertia = diag(prod(trans(Q), mat3x3f(prod(I, Q))));
|
|
|
|
m_vertices.resize(num_vertices);
|
|
for(size_t i = 0; i < m_vertices.size(); ++i) {
|
|
m_vertices[i] = prod(Qinv, vertices[i] - m_center);
|
|
}
|
|
|
|
m_tiva.reset(new btTriangleIndexVertexArray(num_indices / 3, (int*)&(m_indices[0]), 3 * sizeof(unsigned int),
|
|
num_vertices, (float*)&(m_vertices[0]), sizeof(vec3f)));
|
|
|
|
m_gi_shape.reset(new btGImpactMeshShape(m_tiva.get()));
|
|
m_gi_shape->setLocalScaling(btVector3(1.0f,1.0f,1.0f));
|
|
m_gi_shape->updateBound();
|
|
btCompoundShape *compound_shape = new btCompoundShape;
|
|
compound_shape->addChildShape(btTransform(btQuaternion(m_rotation[1],
|
|
m_rotation[2],
|
|
m_rotation[3],
|
|
m_rotation[0]),
|
|
btVector3(m_center[0],
|
|
m_center[1],
|
|
m_center[2])),
|
|
m_gi_shape.get());
|
|
set_shape(compound_shape);
|
|
|
|
//std::cout << "construtor: " << m_center << std::endl;
|
|
|
|
// gimpactShape->setMargin(0.05);
|
|
}
|
|
|
|
void update()
|
|
{
|
|
//apply the scaling
|
|
btVector3 const& scale = m_gi_shape->getLocalScaling();
|
|
|
|
std::vector<vec3f> vertices(m_vertices.size());
|
|
for(unsigned int i = 0; i < vertices.size(); ++i) {
|
|
vertices[i] = vec3f(scale.x() * m_vertices[i][0], scale.y() * m_vertices[i][1], scale.z() * m_vertices[i][2]);
|
|
}
|
|
m_volume = ::volume(&(vertices[0]), (int*)&(m_indices[0]), (int)m_indices.size());
|
|
mat3x3f I = inertia(&(vertices[0]), (int*)&(m_indices[0]), (int)m_indices.size(), vec3f(0, 0, 0));
|
|
// std::cout << I << std::endl;
|
|
//m_rotation = diagonalizer(I);
|
|
//std::cout << rotation << std::endl;
|
|
//the rotation shouldn't change from scaling
|
|
|
|
mat3x3f Q, Qinv;
|
|
q_to_mat(m_rotation, Q);
|
|
q_to_mat(qconj(m_rotation), Qinv);
|
|
|
|
//D = Q * I * trans(Q);
|
|
m_local_inertia = diag(prod(trans(Q), mat3x3f(prod(I, Q))));
|
|
}
|
|
|
|
private:
|
|
shared_ptr<btGImpactMeshShape> m_gi_shape;
|
|
std::vector<vec3f> m_vertices;
|
|
std::vector<vec3f> m_normals;
|
|
std::vector<unsigned int> m_indices;
|
|
shared_ptr<btTriangleIndexVertexArray> m_tiva;
|
|
|
|
float m_volume;
|
|
vec3f m_center;
|
|
quatf m_rotation;
|
|
vec3f m_local_inertia;
|
|
};
|
|
|
|
#endif
|