Saxum/extern/bullet/Extras/CDTestFramework/RenderingHelpers.cpp
Fabian Klemp aeb6218d2d Renaming.
2014-10-24 11:49:46 +02:00

1165 lines
31 KiB
C++

/*
CDTestFramework http://codercorner.com
Copyright (c) 2007-2008 Pierre Terdiman, pierre@codercorner.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.
*/
#include "stdafx.h"
#include <stdio.h>
#include "GL/glut.h"
#include "RenderingHelpers.h"
////////////////////////
void DrawLine(const Point& p0, const Point& p1, const Point& color, float line_width)
{
glDisable(GL_LIGHTING);
glLineWidth(line_width);
glColor4f(color.x, color.y, color.z, 1.0f);
Point tmp[] = {p0, p1};
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, sizeof(Point), &tmp[0].x);
glDrawArrays(GL_LINES, 0, 2);
glDisableClientState(GL_VERTEX_ARRAY);
glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
glEnable(GL_LIGHTING);
}
void DrawTriangle(const Point& p0, const Point& p1, const Point& p2, const Point& color)
{
// glDisable(GL_LIGHTING);
glColor4f(color.x, color.y, color.z, 1.0f);
Point tmp[] = {p0, p1, p2};
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, sizeof(Point), &tmp[0].x);
glDrawArrays(GL_TRIANGLES, 0, 3);
glDisableClientState(GL_VERTEX_ARRAY);
// glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
// glEnable(GL_LIGHTING);
}
////////////////////////
static void SetupGLMatrix(const Point& pos)
{
float glmat[16]; //4x4 column major matrix for OpenGL.
glmat[0] = 1.0f;
glmat[1] = 0.0f;
glmat[2] = 0.0f;
glmat[3] = 0.0f;
glmat[4] = 0.0f;
glmat[5] = 1.0f;
glmat[6] = 0.0f;
glmat[7] = 0.0f;
glmat[8] = 0.0f;
glmat[9] = 0.0f;
glmat[10] = 1.0f;
glmat[11] = 0.0f;
glmat[12] = pos.x;
glmat[13] = pos.y;
glmat[14] = pos.z;
glmat[15] = 1.0f;
glMultMatrixf(glmat);
}
void SetupGLMatrix(const Point& pos, const Matrix3x3& rot)
{
float glmat[16]; //4x4 column major matrix for OpenGL.
glmat[0] = rot.m[0][0];
glmat[1] = rot.m[0][1];
glmat[2] = rot.m[0][2];
glmat[3] = 0.0f;
glmat[4] = rot.m[1][0];
glmat[5] = rot.m[1][1];
glmat[6] = rot.m[1][2];
glmat[7] = 0.0f;
glmat[8] = rot.m[2][0];
glmat[9] = rot.m[2][1];
glmat[10] = rot.m[2][2];
glmat[11] = 0.0f;
glmat[12] = pos.x;
glmat[13] = pos.y;
glmat[14] = pos.z;
glmat[15] = 1.0f;
glMultMatrixf(glmat);
}
void SetupGLMatrix(const Matrix4x4& world)
{
SetupGLMatrix(world.GetTrans(), world);
}
////////////////////////
static void RenderSphere(float radius)
{
glutSolidSphere(radius, 12, 12); // Radius / slices / stacks
}
void DrawSphere(const Sphere& sphere)
{
glPushMatrix();
SetupGLMatrix(sphere.mCenter);
RenderSphere(sphere.mRadius);
glPopMatrix();
}
////////////////////////
static void RenderBox(int size)
{
glutSolidCube(size);
}
void DrawOBB(const OBB& obb)
{
glPushMatrix();
SetupGLMatrix(obb.mCenter, obb.mRot);
glScalef(obb.mExtents.x, obb.mExtents.y, obb.mExtents.z);
RenderBox(2);
glPopMatrix();
}
////////////////////////
static float gCylinderData[]={
1.0f,0.0f,1.0f,1.0f,0.0f,1.0f,1.0f,0.0f,0.0f,1.0f,0.0f,0.0f,
0.866025f,0.500000f,1.0f,0.866025f,0.500000f,1.0f,0.866025f,0.500000f,0.0f,0.866025f,0.500000f,0.0f,
0.500000f,0.866025f,1.0f,0.500000f,0.866025f,1.0f,0.500000f,0.866025f,0.0f,0.500000f,0.866025f,0.0f,
-0.0f,1.0f,1.0f,-0.0f,1.0f,1.0f,-0.0f,1.0f,0.0f,-0.0f,1.0f,0.0f,
-0.500000f,0.866025f,1.0f,-0.500000f,0.866025f,1.0f,-0.500000f,0.866025f,0.0f,-0.500000f,0.866025f,0.0f,
-0.866025f,0.500000f,1.0f,-0.866025f,0.500000f,1.0f,-0.866025f,0.500000f,0.0f,-0.866025f,0.500000f,0.0f,
-1.0f,-0.0f,1.0f,-1.0f,-0.0f,1.0f,-1.0f,-0.0f,0.0f,-1.0f,-0.0f,0.0f,
-0.866025f,-0.500000f,1.0f,-0.866025f,-0.500000f,1.0f,-0.866025f,-0.500000f,0.0f,-0.866025f,-0.500000f,0.0f,
-0.500000f,-0.866025f,1.0f,-0.500000f,-0.866025f,1.0f,-0.500000f,-0.866025f,0.0f,-0.500000f,-0.866025f,0.0f,
0.0f,-1.0f,1.0f,0.0f,-1.0f,1.0f,0.0f,-1.0f,0.0f,0.0f,-1.0f,0.0f,
0.500000f,-0.866025f,1.0f,0.500000f,-0.866025f,1.0f,0.500000f,-0.866025f,0.0f,0.500000f,-0.866025f,0.0f,
0.866026f,-0.500000f,1.0f,0.866026f,-0.500000f,1.0f,0.866026f,-0.500000f,0.0f,0.866026f,-0.500000f,0.0f,
1.0f,0.0f,1.0f,1.0f,0.0f,1.0f,1.0f,0.0f,0.0f,1.0f,0.0f,0.0f
};
static float gCylinderDataCapsTop[]={
0.866026f,-0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.500000f,-0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
0.500000f,-0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,-1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,-1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.500000f,-0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.500000f,-0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.866025f,-0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.866025f,-0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-1.000000f,-0.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-1.000000f,-0.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.866025f,0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.866025f,0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.500000f,0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.500000f,0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
-0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.500000f,0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
0.500000f,0.866025f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.866025f,0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.866025f,0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
1.000000f,0.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
1.000000f,0.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.000000f,1.000000f,1.000000f,0.000000f,1.000000f,1.000000f,
0.866026f,-0.500000f,1.000000f,0.000000f,1.000000f,1.000000f,
};
static float gCylinderDataCapsBottom[]={
1.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.866025f,0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.866025f,0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.500000f,0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.500000f,0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.000000f,1.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.000000f,1.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.500000f,0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.500000f,0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.866025f,0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.866025f,0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-1.000000f,-0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-1.000000f,-0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.866025f,-0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.866025f,-0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.500000f,-0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
-0.500000f,-0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,-1.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,-1.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.500000f,-0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.500000f,-0.866025f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.866026f,-0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.866026f,-0.500000f,0.000000f,0.000000f,-1.000000f,0.000000f,
0.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
1.000000f,0.000000f,0.000000f,0.000000f,-1.000000f,0.000000f,
};
static void RenderCylinder()
{
glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_NORMAL_ARRAY);
glVertexPointer(3, GL_FLOAT, 2*3*sizeof(float), gCylinderData);
glNormalPointer(GL_FLOAT, 2*3*sizeof(float), gCylinderData+3);
glDrawArrays(GL_TRIANGLE_STRIP, 0, 13*2);
glVertexPointer(3, GL_FLOAT, 2*3*sizeof(float), gCylinderDataCapsTop);
glNormalPointer(GL_FLOAT, 2*3*sizeof(float), gCylinderDataCapsTop+3);
glDrawArrays(GL_TRIANGLES, 0, 36);
glVertexPointer(3, GL_FLOAT, 2*3*sizeof(float), gCylinderDataCapsBottom);
glNormalPointer(GL_FLOAT, 2*3*sizeof(float), gCylinderDataCapsBottom+3);
glDrawArrays(GL_TRIANGLES, 0, 36);
glDisableClientState(GL_VERTEX_ARRAY);
glDisableClientState(GL_NORMAL_ARRAY);
}
void DrawCapsule(const Matrix4x4& world, const Point& p0, const Point& p1, float r)
{
const float h = p0.Distance(p1);
glPushMatrix();
SetupGLMatrix(world);
glPushMatrix();
glTranslatef(0.0f, h*0.5f, 0.0f);
RenderSphere(r);
glPopMatrix();
glPushMatrix();
glTranslatef(0.0f,-h*0.5f, 0.0f);
RenderSphere(r);
glPopMatrix();
glPushMatrix();
glTranslatef(0.0f,h*0.5f, 0.0f);
glScalef(r,h,r);
glRotatef(90.0f,1.0f,0.0f,0.0f);
RenderCylinder();
glPopMatrix();
glPopMatrix();
}
#ifdef TOSEE
#include "NxPhysics.h"
#include "DrawObjects.h"
#include <GL/glut.h>
static float gPlaneData[]={
-1.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f,
1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, -1.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f,
1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, 1.0f, 0.0f, -1.0f, 0.0f, 1.0f, 0.0f
};
static void RenderPlane()
{
glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_NORMAL_ARRAY);
glVertexPointer(3, GL_FLOAT, 2*3*sizeof(float), gPlaneData);
glNormalPointer(GL_FLOAT, 2*3*sizeof(float), gPlaneData+3);
glDrawArrays(GL_TRIANGLES, 0, 6);
glDisableClientState(GL_VERTEX_ARRAY);
glDisableClientState(GL_NORMAL_ARRAY);
}
void SetupGLMatrix(const NxVec3& pos, const NxMat33& orient)
{
float glmat[16]; //4x4 column major matrix for OpenGL.
orient.getColumnMajorStride4(&(glmat[0]));
pos.get(&(glmat[12]));
//clear the elements we don't need:
glmat[3] = glmat[7] = glmat[11] = 0.0f;
glmat[15] = 1.0f;
glMultMatrixf(&(glmat[0]));
}
void DrawLine(const NxVec3& p0, const NxVec3& p1, const NxVec3& color, float lineWidth)
{
glDisable(GL_LIGHTING);
glLineWidth(lineWidth);
glColor4f(color.x, color.y, color.z, 1.0f);
NxVec3 av3LineEndpoints[] = {p0, p1};
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, sizeof(NxVec3), &av3LineEndpoints[0].x);
glDrawArrays(GL_LINES, 0, 2);
glDisableClientState(GL_VERTEX_ARRAY);
glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
glEnable(GL_LIGHTING);
}
void DrawTriangle(const NxVec3& p0, const NxVec3& p1, const NxVec3& p2, const NxVec3& color)
{
glDisable(GL_LIGHTING);
glColor4f(color.x, color.y, color.z, 1.0f);
NxVec3 av3LineEndpoints[] = {p0, p1, p2};
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, sizeof(NxVec3), &av3LineEndpoints[0].x);
glDrawArrays(GL_TRIANGLES, 0, 3);
glDisableClientState(GL_VERTEX_ARRAY);
glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
glEnable(GL_LIGHTING);
}
void DrawCircle(NxU32 nbSegments, const NxMat34& matrix, const NxVec3& color, const NxF32 radius, const bool semicircle)
{
NxF32 step = NxTwoPiF32/NxF32(nbSegments);
NxU32 segs = nbSegments;
if(semicircle)
{
segs /= 2;
}
for(NxU32 i=0;i<segs;i++)
{
NxU32 j=i+1;
if(j==nbSegments) j=0;
NxF32 angle0 = NxF32(i)*step;
NxF32 angle1 = NxF32(j)*step;
NxVec3 p0,p1;
matrix.multiply(NxVec3(radius * sinf(angle0), radius * cosf(angle0), 0.0f), p0);
matrix.multiply(NxVec3(radius * sinf(angle1), radius * cosf(angle1), 0.0f), p1);
DrawLine(p0, p1, color);
}
}
void DrawEllipse(NxU32 nbSegments, const NxMat34& matrix, const NxVec3& color, const NxF32 radius1, const NxF32 radius2, const bool semicircle)
{
NxF32 step = NxTwoPiF32/NxF32(nbSegments);
NxU32 segs = nbSegments;
if(semicircle)
{
segs /= 2;
}
for(NxU32 i=0;i<segs;i++)
{
NxU32 j=i+1;
if(j==nbSegments) j=0;
NxF32 angle0 = NxF32(i)*step;
NxF32 angle1 = NxF32(j)*step;
NxVec3 p0,p1;
matrix.multiply(NxVec3(radius1 * sinf(angle0), radius2 * cosf(angle0), 0.0f), p0);
matrix.multiply(NxVec3(radius1 * sinf(angle1), radius2 * cosf(angle1), 0.0f), p1);
DrawLine(p0, p1, color);
}
}
void DrawWirePlane(NxShape* plane, const NxVec3& color)
{
}
void DrawPlane(NxShape* plane)
{
NxMat34 pose = plane->getGlobalPose();
NxPlaneShape* planeShape = plane->isPlane();
NxPlane p = planeShape->getPlane();
pose.t.x += p.d;
pose.t.y += p.d;
pose.t.z += p.d;
glPushMatrix();
glDisable(GL_LIGHTING);
glColor4f(0.1f, 0.2f, 0.3f, 1.0f);
pose.t.y -= 0.1f;
SetupGLMatrix(pose.t, pose.M);
glScalef(1024,0,1024);
RenderPlane();
glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
glEnable(GL_LIGHTING);
glPopMatrix();
}
void DrawWireBox(const NxBox& obb, const NxVec3& color, float lineWidth)
{
// Compute obb vertices
NxVec3 pp[8];
NxComputeBoxPoints(obb, pp);
// Draw all lines
const NxU32* Indices = NxGetBoxEdges();
for(NxU32 i=0;i<12;i++)
{
NxU32 VRef0 = *Indices++;
NxU32 VRef1 = *Indices++;
DrawLine(pp[VRef0], pp[VRef1], color, lineWidth);
}
}
void DrawWireBox(NxShape* box, const NxVec3& color, float lineWidth)
{
NxBox obb;
box->isBox()->getWorldOBB(obb);
DrawWireBox(obb, color, lineWidth);
}
void DrawWireSphere(NxShape* sphere, const NxVec3& color)
{
NxMat34 pose = sphere->getGlobalPose();
glPushMatrix();
NxReal r = sphere->isSphere()->getRadius();
NxVec3 c0; pose.M.getColumn(0, c0);
NxVec3 c1; pose.M.getColumn(1, c1);
NxVec3 c2; pose.M.getColumn(2, c2);
DrawCircle(20, pose, color, r);
pose.M.setColumn(0, c1);
pose.M.setColumn(1, c2);
pose.M.setColumn(2, c0);
DrawCircle(20, pose, color, r);
pose.M.setColumn(0, c2);
pose.M.setColumn(1, c0);
pose.M.setColumn(2, c1);
DrawCircle(20, pose, color, r);
glPopMatrix();
}
void DrawSphere(NxShape* sphere)
{
NxMat34 pose = sphere->getGlobalPose();
glPushMatrix();
SetupGLMatrix(pose.t, pose.M);
NxReal r = sphere->isSphere()->getRadius();
glScalef(r,r,r);
RenderSphere();
glPopMatrix();
}
void DrawWireCapsule(NxShape* capsule, const NxVec3& color)
{
NxMat34 pose = capsule->getGlobalPose();
NxReal r = capsule->isCapsule()->getRadius();
NxReal h = capsule->isCapsule()->getHeight();
NxSegment SG;
pose.M.getColumn(1, SG.p1);
SG.p1 *= 0.5f*h;
SG.p0 = -SG.p1;
SG.p0 += pose.t;
SG.p1 += pose.t;
NxVec3 c0; pose.M.getColumn(0, c0);
NxVec3 c1; pose.M.getColumn(1, c1);
NxVec3 c2; pose.M.getColumn(2, c2);
DrawLine(SG.p0 + c0*r, SG.p1 + c0*r, color);
DrawLine(SG.p0 - c0*r, SG.p1 - c0*r, color);
DrawLine(SG.p0 + c2*r, SG.p1 + c2*r, color);
DrawLine(SG.p0 - c2*r, SG.p1 - c2*r, color);
pose.M.setColumn(0, -c1);
pose.M.setColumn(1, -c0);
pose.M.setColumn(2, c2);
pose.t = SG.p0;
DrawCircle(20, pose, color, r, true); //halfcircle -- flipped
pose.M.setColumn(0, c1);
pose.M.setColumn(1, -c0);
pose.M.setColumn(2, c2);
pose.t = SG.p1;
DrawCircle(20, pose, color, r, true);
pose.M.setColumn(0, -c1);
pose.M.setColumn(1, c2);
pose.M.setColumn(2, c0);
pose.t = SG.p0;
DrawCircle(20, pose, color, r, true);//halfcircle -- good
pose.M.setColumn(0, c1);
pose.M.setColumn(1, c2);
pose.M.setColumn(2, c0);
pose.t = SG.p1;
DrawCircle(20, pose, color, r, true);
pose.M.setColumn(0, c2);
pose.M.setColumn(1, c0);
pose.M.setColumn(2, c1);
pose.t = SG.p0;
DrawCircle(20, pose, color, r); //full circle
pose.t = SG.p1;
DrawCircle(20, pose, color, r);
}
static void computeBasis(const NxVec3& dir, NxVec3& right, NxVec3& up)
{
// Derive two remaining vectors
if(fabsf(dir.y)>0.9999f) right = NxVec3(1.0f, 0.0f, 0.0f);
else right = (NxVec3(0.0f, 1.0f, 0.0f) ^ dir);
right.normalize();
up = dir ^ right;
}
void DrawWireCapsule(const NxCapsule& capsule, const NxVec3& color)
{
NxReal r = capsule.radius;
NxVec3 dir = capsule.p0 - capsule.p1;
NxReal h = dir.magnitude();
NxMat34 pose;
pose.t = (capsule.p0 + capsule.p1)*0.5f;
if(h!=0.0f)
{
dir/=h;
NxVec3 right, up;
computeBasis(dir, right, up);
pose.M.setColumn(0, right);
pose.M.setColumn(1, dir);
pose.M.setColumn(2, up);
}
else
{
pose.M.id();
}
// NxMat34 pose = capsule->getGlobalPose();
// const NxReal & r = capsule->isCapsule()->getRadius();
// const NxReal & h = capsule->isCapsule()->getHeight();
NxSegment SG;
pose.M.getColumn(1, SG.p1);
SG.p1 *= 0.5f*h;
SG.p0 = -SG.p1;
SG.p0 += pose.t;
SG.p1 += pose.t;
NxVec3 c0; pose.M.getColumn(0, c0);
NxVec3 c1; pose.M.getColumn(1, c1);
NxVec3 c2; pose.M.getColumn(2, c2);
DrawLine(SG.p0 + c0*r, SG.p1 + c0*r, color);
DrawLine(SG.p0 - c0*r, SG.p1 - c0*r, color);
DrawLine(SG.p0 + c2*r, SG.p1 + c2*r, color);
DrawLine(SG.p0 - c2*r, SG.p1 - c2*r, color);
pose.M.setColumn(0, -c1);
pose.M.setColumn(1, -c0);
pose.M.setColumn(2, c2);
pose.t = SG.p0;
DrawCircle(20, pose, color, r, true); //halfcircle -- flipped
pose.M.setColumn(0, c1);
pose.M.setColumn(1, -c0);
pose.M.setColumn(2, c2);
pose.t = SG.p1;
DrawCircle(20, pose, color, r, true);
pose.M.setColumn(0, -c1);
pose.M.setColumn(1, c2);
pose.M.setColumn(2, c0);
pose.t = SG.p0;
DrawCircle(20, pose, color, r, true);//halfcircle -- good
pose.M.setColumn(0, c1);
pose.M.setColumn(1, c2);
pose.M.setColumn(2, c0);
pose.t = SG.p1;
DrawCircle(20, pose, color, r, true);
pose.M.setColumn(0, c2);
pose.M.setColumn(1, c0);
pose.M.setColumn(2, c1);
pose.t = SG.p0;
DrawCircle(20, pose, color, r); //full circle
pose.t = SG.p1;
DrawCircle(20, pose, color, r);
}
void DrawCapsule(const NxVec3& color, NxF32 r, NxF32 h)
{
glColor4f(color.x, color.y, color.z, 1.0f);
glPushMatrix();
glTranslatef(0.0f, h*0.5f, 0.0f);
glScalef(r,r,r);
RenderSphere();
glPopMatrix();
glPushMatrix();
glTranslatef(0.0f,-h*0.5f, 0.0f);
glScalef(r,r,r);
RenderSphere();
glPopMatrix();
glPushMatrix();
glTranslatef(0.0f,h*0.5f, 0.0f);
glScalef(r,h,r);
glRotatef(90.0f,1.0f,0.0f,0.0f);
RenderCylinder();
glPopMatrix();
}
typedef NxVec3 Point;
typedef struct _Triangle { NxU32 p0; NxU32 p1; NxU32 p2; } Triangle;
void DrawWireConvex(NxShape* mesh, const NxVec3& color)
{
if(mesh->userData == NULL) return;
NxMat34 pose = mesh->getGlobalPose();
NxConvexMeshDesc meshDesc = *((NxConvexMeshDesc*)(mesh->userData));
// mesh->isConvexMesh()->getConvexMesh().saveToDesc(meshDesc);
NxU32 nbVerts = meshDesc.numVertices;
NxU32 nbTriangles = meshDesc.numTriangles;
Point* points = (Point *)meshDesc.points;
Triangle* triangles = (Triangle *)meshDesc.triangles;
glPushMatrix();
float glmat[16]; //4x4 column major matrix for OpenGL.
pose.M.getColumnMajorStride4(&(glmat[0]));
pose.t.get(&(glmat[12]));
//clear the elements we don't need:
glmat[3] = glmat[7] = glmat[11] = 0.0f;
glmat[15] = 1.0f;
glMultMatrixf(&(glmat[0]));
while(nbTriangles--)
{
DrawLine(points[triangles->p0], points[triangles->p1], color);
DrawLine(points[triangles->p1], points[triangles->p2], color);
DrawLine(points[triangles->p2], points[triangles->p0], color);
triangles++;
}
glPopMatrix();
}
void DrawTriangleList(int iTriangleCount, Triangle *pTriangles, Point *pPoints)
{
static int iBufferSize=0;
static float *pfVertexBuffer=NULL;
static float *pfNormalBuffer=NULL;
if(iBufferSize < iTriangleCount*3)
{
iBufferSize=3*iTriangleCount;
delete[] pfVertexBuffer;
pfVertexBuffer = new float[iBufferSize*3];
delete[] pfNormalBuffer;
pfNormalBuffer = new float[iBufferSize*3];
}
float *pfDestinationVertex=pfVertexBuffer;
float *pfDestinationNormal=pfNormalBuffer;
for(int iTriangle=0; iTriangle<iTriangleCount; iTriangle++)
{
*pfDestinationVertex++=pPoints[pTriangles->p0].x;
*pfDestinationVertex++=pPoints[pTriangles->p0].y;
*pfDestinationVertex++=pPoints[pTriangles->p0].z;
*pfDestinationVertex++=pPoints[pTriangles->p1].x;
*pfDestinationVertex++=pPoints[pTriangles->p1].y;
*pfDestinationVertex++=pPoints[pTriangles->p1].z;
*pfDestinationVertex++=pPoints[pTriangles->p2].x;
*pfDestinationVertex++=pPoints[pTriangles->p2].y;
*pfDestinationVertex++=pPoints[pTriangles->p2].z;
NxVec3 edge1 = pPoints[pTriangles->p1] - pPoints[pTriangles->p0];
NxVec3 edge2 = pPoints[pTriangles->p2] - pPoints[pTriangles->p0];
NxVec3 normal = edge1.cross(edge2);
normal.normalize();
*pfDestinationNormal++=normal.x;
*pfDestinationNormal++=normal.y;
*pfDestinationNormal++=normal.z;
*pfDestinationNormal++=normal.x;
*pfDestinationNormal++=normal.y;
*pfDestinationNormal++=normal.z;
*pfDestinationNormal++=normal.x;
*pfDestinationNormal++=normal.y;
*pfDestinationNormal++=normal.z;
pTriangles++;
}
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, 0, pfVertexBuffer);
glEnableClientState(GL_NORMAL_ARRAY);
glNormalPointer(GL_FLOAT, 0, pfNormalBuffer);
glDrawArrays(GL_TRIANGLES, 0, 3*iTriangleCount);
glDisableClientState(GL_VERTEX_ARRAY);
glDisableClientState(GL_NORMAL_ARRAY);
}
void DrawConvex(NxShape* mesh)
{
NxConvexMeshDesc meshDesc;
// SRM : Please note that I looked into a crash issue posed by
// one of our customers, and this code (i.e. reinterpreting the
// NxShape's userData as an NxConvexMeshDesc * was crashing because
// in the SampleRaycastCar demo, it sets that pointer equal to the
// NxWheel because that is used in NxVehicle::handleContactPair(). Thus
// in order to allow this code not to crash on the PS3, we should
// simply force the shape to save its description here.
mesh->isConvexMesh()->getConvexMesh().saveToDesc(meshDesc);
/**
if(mesh->userData != NULL)
{
meshDesc = *((NxConvexMeshDesc*)(mesh->userData));
} else {
mesh->isConvexMesh()->getConvexMesh().saveToDesc(meshDesc);
}
**/
NxMat34 pose = mesh->getGlobalPose();
NxU32 nbVerts = meshDesc.numVertices;
NxU32 nbTriangles = meshDesc.numTriangles;
Point* points = (Point *)meshDesc.points;
Triangle* triangles = (Triangle *)meshDesc.triangles;
glPushMatrix();
float glmat[16]; //4x4 column major matrix for OpenGL.
pose.M.getColumnMajorStride4(&(glmat[0]));
pose.t.get(&(glmat[12]));
//clear the elements we don't need:
glmat[3] = glmat[7] = glmat[11] = 0.0f;
glmat[15] = 1.0f;
glMultMatrixf(&(glmat[0]));
DrawTriangleList(nbTriangles, triangles, points);
glPopMatrix();
}
void DrawWireMesh(NxShape* mesh, const NxVec3& color)
{
if(mesh->userData == NULL) return;
NxMat34 pose = mesh->getGlobalPose();
NxTriangleMeshDesc meshDesc = *((NxTriangleMeshDesc*)(mesh->userData));
// mesh->isTriangleMesh()->getTriangleMesh().saveToDesc(meshDesc);
NxU32 nbVerts = meshDesc.numVertices;
NxU32 nbTriangles = meshDesc.numTriangles;
Point* points = (Point *)meshDesc.points;
Triangle* triangles = (Triangle *)meshDesc.triangles;
glPushMatrix();
float glmat[16]; //4x4 column major matrix for OpenGL.
pose.M.getColumnMajorStride4(&(glmat[0]));
pose.t.get(&(glmat[12]));
//clear the elements we don't need:
glmat[3] = glmat[7] = glmat[11] = 0.0f;
glmat[15] = 1.0f;
glMultMatrixf(&(glmat[0]));
while(nbTriangles--)
{
DrawLine(points[triangles->p0], points[triangles->p1], color);
DrawLine(points[triangles->p1], points[triangles->p2], color);
DrawLine(points[triangles->p2], points[triangles->p0], color);
triangles++;
}
glPopMatrix();
}
void DrawMesh(NxShape* mesh)
{
if(mesh->userData == NULL) return;
NxMat34 pose = mesh->getGlobalPose();
NxTriangleMeshDesc meshDesc = *((NxTriangleMeshDesc*)(mesh->userData));
// mesh->isTriangleMesh()->getTriangleMesh().saveToDesc(meshDesc);
NxU32 nbVerts = meshDesc.numVertices;
NxU32 nbTriangles = meshDesc.numTriangles;
Point* points = (Point *)meshDesc.points;
Triangle* triangles = (Triangle *)meshDesc.triangles;
glPushMatrix();
float glmat[16]; //4x4 column major matrix for OpenGL.
pose.M.getColumnMajorStride4(&(glmat[0]));
pose.t.get(&(glmat[12]));
//clear the elements we don't need:
glmat[3] = glmat[7] = glmat[11] = 0.0f;
glmat[15] = 1.0f;
glMultMatrixf(&(glmat[0]));
if(meshDesc.heightFieldVerticalAxis != NX_NOT_HEIGHTFIELD)
{
glDisable(GL_LIGHT0);
glEnable(GL_LIGHT1);
}
DrawTriangleList(nbTriangles, triangles, points);
if(meshDesc.heightFieldVerticalAxis != NX_NOT_HEIGHTFIELD)
{
glDisable(GL_LIGHT1);
glEnable(GL_LIGHT0);
}
glPopMatrix();
}
void DrawWheelShape(NxShape* wheel)
{
if(wheel->is(NX_SHAPE_WHEEL) != NULL)
{
NxWheelShape* wheelShape = (NxWheelShape*)wheel;
glPushMatrix();
float glmat[16];
wheel->getGlobalPose().getColumnMajor44(glmat);
NxWheelShapeDesc wheelShapeDesc;
float r = wheelShape->getRadius();
float a = wheelShape->getSteerAngle();
glMultMatrixf(&(glmat[0]));
glTranslatef(-r/2,0,0);
glRotatef(90,0,1,0);
glRotatef(NxMath::radToDeg(a),0,1,0);
glScalef(r, r, r);
RenderCylinder();
glPopMatrix();
}
}
void DrawArrow(const NxVec3& posA, const NxVec3& posB, const NxVec3& color)
{
NxVec3 vec = posB - posA;
NxVec3 t0, t1, t2;
NxNormalToTangents(vec, t1, t2);
t0 = posB - posA;
t0.normalize();
NxVec3 lobe1 = posB - t0*0.15 + t1 * 0.15;
NxVec3 lobe2 = posB - t0*0.15 - t1 * 0.15;
NxVec3 lobe3 = posB - t0*0.15 + t2 * 0.15;
NxVec3 lobe4 = posB - t0*0.15 - t2 * 0.15;
NxVec3 v3ArrowShape[] = {
NxVec3(posA), NxVec3(posB),
NxVec3(posB), NxVec3(lobe1),
NxVec3(posB), NxVec3(lobe2),
NxVec3(posB), NxVec3(lobe3),
NxVec3(posB), NxVec3(lobe4),
};
glDisable(GL_LIGHTING);
glLineWidth(3.0f);
glColor4f(color.x,color.y,color.z,1.0f);
glEnableClientState(GL_VERTEX_ARRAY);
glVertexPointer(3, GL_FLOAT, sizeof(NxVec3), &v3ArrowShape[0].x);
glDrawArrays(GL_LINES, 0, sizeof(v3ArrowShape)/sizeof(NxVec3));
glDisableClientState(GL_VERTEX_ARRAY);
glColor4f(1.0f,1.0f,1.0f,1.0f);
}
void DrawContactPoint(const NxVec3& pos, const NxReal radius, const NxVec3& color)
{
NxMat34 pose;
pose.t = pos;
NxVec3 c0; pose.M.getColumn(0, c0);
NxVec3 c1; pose.M.getColumn(1, c1);
NxVec3 c2; pose.M.getColumn(2, c2);
DrawCircle(20, pose, color, radius);
pose.M.setColumn(0, c1);
pose.M.setColumn(1, c2);
pose.M.setColumn(2, c0);
DrawCircle(20, pose, color, radius);
pose.M.setColumn(0, c2);
pose.M.setColumn(1, c0);
pose.M.setColumn(2, c1);
DrawCircle(20, pose, color, radius);
}
void DrawWireShape(NxShape *shape, const NxVec3& color)
{
switch(shape->getType())
{
case NX_SHAPE_PLANE:
DrawWirePlane(shape, color);
break;
case NX_SHAPE_BOX:
DrawWireBox(shape, color);
break;
case NX_SHAPE_SPHERE:
DrawWireSphere(shape, color);
break;
case NX_SHAPE_CAPSULE:
DrawWireCapsule(shape, color);
break;
case NX_SHAPE_CONVEX:
DrawWireConvex(shape, color);
break;
case NX_SHAPE_MESH:
DrawWireMesh(shape, color);
break;
}
}
void DrawShape(NxShape* shape)
{
switch(shape->getType())
{
case NX_SHAPE_PLANE:
DrawPlane(shape);
break;
case NX_SHAPE_BOX:
DrawBox(shape);
break;
case NX_SHAPE_SPHERE:
DrawSphere(shape);
break;
case NX_SHAPE_CAPSULE:
DrawCapsule(shape);
break;
case NX_SHAPE_CONVEX:
DrawConvex(shape);
break;
case NX_SHAPE_MESH:
DrawMesh(shape);
break;
case NX_SHAPE_WHEEL:
DrawWheelShape(shape);
break;
}
}
void DrawActor(NxActor* actor, NxActor* gSelectedActor)
{
NxShape*const* shapes = actor->getShapes();
NxU32 nShapes = actor->getNbShapes();
if (actor == gSelectedActor)
{
while (nShapes--)
{
DrawWireShape(shapes[nShapes], NxVec3(1,1,1));
}
}
nShapes = actor->getNbShapes();
while (nShapes--)
{
DrawShape(shapes[nShapes]);
}
}
void DrawWireActor(NxActor* actor)
{
NxShape*const* shapes = actor->getShapes();
NxU32 nShapes = actor->getNbShapes();
nShapes = actor->getNbShapes();
while (nShapes--)
{
DrawWireShape(shapes[nShapes], NxVec3(1,1,1));
}
}
static void DrawActorShadow(NxActor* actor, const float* ShadowMat)
{
glPushMatrix();
glMultMatrixf(ShadowMat);
glDisable(GL_LIGHTING);
glColor4f(0.05f, 0.1f, 0.15f, 1.0f);
NxShape*const* shapes = actor->getShapes();
NxU32 nShapes = actor->getNbShapes();
while (nShapes--)
{
switch(shapes[nShapes]->getType())
{
case NX_SHAPE_BOX:
DrawBox(shapes[nShapes]);
break;
case NX_SHAPE_SPHERE:
DrawSphere(shapes[nShapes]);
break;
case NX_SHAPE_CAPSULE:
DrawCapsule(shapes[nShapes]);
break;
case NX_SHAPE_CONVEX:
DrawConvex(shapes[nShapes]);
break;
}
}
glColor4f(1.0f, 1.0f, 1.0f, 1.0f);
glEnable(GL_LIGHTING);
glPopMatrix();
}
void DrawActorShadow(NxActor* actor)
{
const static float ShadowMat[]={ 1,0,0,0, 0,0,0,0, 0,0,1,0, 0,0,0,1 };
DrawActorShadow(actor, ShadowMat);
}
void DrawActorShadow2(NxActor* actor)
{
const static float ShadowMat[]={ 1,0,0,0, 1,0,-0.2,0, 0,0,1,0, 0,0,0,1 };
DrawActorShadow(actor, ShadowMat);
}
void DrawActorShadowZUp(NxActor* actor)
{
const static float ShadowMat[]={ 1,0,0,0, 0,1,0,0, 0,0,0,0, 0,0,0,1 };
DrawActorShadow(actor, ShadowMat);
}
void DrawCloth(NxCloth *cloth, bool shadows)
{
NxMeshData meshData = cloth->getMeshData();
NxU32 numVertices = *meshData.numVerticesPtr;
NxU32 numTriangles = *meshData.numIndicesPtr / 3;
glEnableClientState(GL_VERTEX_ARRAY);
glEnableClientState(GL_NORMAL_ARRAY);
glVertexPointer(3, GL_FLOAT, 0, meshData.verticesPosBegin);
glNormalPointer(GL_FLOAT, 0, meshData.verticesNormalBegin);
#ifdef __CELLOS_LV2__
glDrawRangeElements(GL_TRIANGLES, 0, numVertices-1, 3*numTriangles, GL_UNSIGNED_INT, meshData.indicesBegin);
#else
glDrawElements(GL_TRIANGLES, 3*numTriangles, GL_UNSIGNED_INT, meshData.indicesBegin);
#endif
if (shadows) {
const static float ShadowMat[]={ 1,0,0,0, 0,0,0,0, 0,0,1,0, 0,0,0,1 };
glPushMatrix();
glMultMatrixf(ShadowMat);
glDisable(GL_LIGHTING);
glColor4f(0.05f, 0.1f, 0.15f,1.0f);
#ifdef __CELLOS_LV2__
glDrawRangeElements(GL_TRIANGLES, 0, numVertices-1, 3*numTriangles, GL_UNSIGNED_INT, meshData.indicesBegin);
#else
glDrawElements(GL_TRIANGLES, 3*numTriangles, GL_UNSIGNED_INT, meshData.indicesBegin);
#endif
glColor4f(1.0f, 1.0f, 1.0f,1.0f);
glEnable(GL_LIGHTING);
glPopMatrix();
}
glDisableClientState(GL_VERTEX_ARRAY);
glDisableClientState(GL_NORMAL_ARRAY);
}
#ifdef WIN32
#include <windows.h>
typedef BOOL (APIENTRY *PFNWGLSWAPINTERVALFARPROC)( int );
PFNWGLSWAPINTERVALFARPROC wglSwapIntervalEXT = 0;
int vsyncState = 1;
#endif
void toggleVSync()
{
#ifdef WIN32
if (wglSwapIntervalEXT == NULL)
{
const char* extensions = (const char*)glGetString(GL_EXTENSIONS);
if (strstr(extensions, "WGL_EXT_swap_control") != 0)
{
wglSwapIntervalEXT = (PFNWGLSWAPINTERVALFARPROC)wglGetProcAddress( "wglSwapIntervalEXT" );
}
}
if (wglSwapIntervalEXT != NULL)
{
vsyncState = 1 - vsyncState;
wglSwapIntervalEXT(vsyncState);
}
#endif
}
#endif