Я пытался разработать работающую FPS камеру в openGL с C ++.
Используя углы Эйлера и взглянув на пример кода из openBL superBible v.5, я смог получить даже хорошо работающие переводы, и если я ограничу повороты одним измерением (например, просто шагом или рысканием) , это тоже хорошо работает. Но когда я комбинирую тангаж и рыскание, я получаю странные результирующие повороты, а также изменения крена (то есть горизонт не остается горизонтальным). Это лучшая система, которая у меня есть, и она была бы идеальной, если бы не прокатка. Однако я не очень хорошо понимаю, что происходит, потому что все, что я делаю, - это вызов методов для перевода и поворота GLFrame.
Я читал об углах Эйлера и обнаружил, что они страдают от проблемы, называемой блокировкой карданного подвеса, поэтому я попытался вместо этого использовать кватернионы. Я сохраняю переменные высоты тона и рыскания и получаю кватернион из них, а затем строю матрицу из кватерниона и помещаю ее в стек MODELVIEW. Что происходит, так это то, что он вращает весь мир, а не POV камеры, и очень странно растягивает формы.
Вот исходный код:
#include <GLTools.h>
#include <GLShaderManager.h>
#include <GL/glut.h>
#include <GLBatch.h>
#include <GLFrustum.h>
#include <GLFrame.h>
#include <GLMatrixStack.h>
#include <GLGeometryTransform.h>
#include <math.h>
#include <stdio.h>
#include <iostream>
using namespace std;
#define CUBE_COUNT 400
#define WINDOW_WIDTH 800
#define WINDOW_HEIGHT 600
#define ESCAPE_KEY 27
//CLASSES
GLShaderManager shaderManager;
GLMatrixStack modelViewMatrix;
GLMatrixStack projectionMatrix;
GLFrame cameraFrame;
GLFrame objectFrame;
GLFrame cubes[CUBE_COUNT];
GLFrustum viewFrustum;
GLBatch triangleBatch;
GLBatch cubeBatch;
GLGeometryTransform transformPipeline;
M3DMatrix44f shadowMatrix;
M3DMatrix44f camMatrix;
GLfloat vGreen[] = {0.0f, 1.0f, 0.0f, 1.0f};
GLfloat vBlack[] = {0.0f, 0.0f, 0.0f, 1.0f};
float pitch;
float yaw;
float roll;
float mouse_x;
float mouse_y;
bool w_pressed = false;
bool a_pressed = false;
bool s_pressed = false;
bool d_pressed = false;
float cam_x = 0.0f;
float cam_y = 0.0f;
float cam_z = 0.0f;
///////////////////////////////////////////////////////////////////////////////
// This function does any needed initialization on the rendering context.
// This is the first opportunity to do any OpenGL related tasks.
typedef struct {
GLfloat x;
GLfloat y;
GLfloat z;
GLfloat w;
} Quaternion3D;
static inline void Quaternion3DNormalise(Quaternion3D *quaternion) {
GLfloat magnitude;
magnitude = sqrtf((quaternion->x * quaternion->x) +
(quaternion->y * quaternion->y) +
(quaternion->z * quaternion->z) +
(quaternion->w * quaternion->w));
quaternion->x /= magnitude;
quaternion->y /= magnitude;
quaternion->z /= magnitude;
quaternion->w /= magnitude;
}
static inline void Quaternion3DMultiplication(Quaternion3D *newQ, Quaternion3D *q1, Quaternion3D *q2) {
newQ->w = q1->w * q2->w - q1->x * q2->x - q1->y * q2->y - q1->z * q2->z;
newQ->x = q1->w * q2->x + q1->x * q2->w + q1->y * q2->z - q1->z * q2->y;
newQ->y = q1->w * q2->y - q1->x * q2->z + q1->y * q2->w + q1->z * q2->x;
newQ->z = q1->w * q2->z + q1->x * q2->y - q1->y * q2->x + q1->z * q2->w;
}
static inline void Quaternion3DMultIdentity(Quaternion3D *q) {
q->x = 0;
q->y = 0;
q->z = 0;
q->w = 1;
}
static inline void Quaternion3DInverse(Quaternion3D *q) {
q->x *= -1;
q->y *= -1;
q->z *= -1;
}
static inline void EulerToQuaternion3D(Quaternion3D *q, float pitch, float yaw, float roll) {
float radiansPitch = m3dDegToRad(pitch);
float radiansYaw = m3dDegToRad(yaw);
float radiansRoll = m3dDegToRad(roll);
float sinPitch = sin(radiansPitch * 0.5);
float cosPitch = cos(radiansPitch * 0.5);
float sinYaw = sin(radiansYaw * 0.5);
float cosYaw = cos(radiansYaw * 0.5);
float sinRoll = sin(radiansRoll * 0.5);
float cosRoll = cos(radiansRoll * 0.5);
q->w = cosYaw * cosRoll * cosPitch - sinYaw * sinRoll * sinPitch;
q->x = sinYaw * sinRoll * cosPitch + cosYaw * cosRoll * sinPitch;
q->y = sinYaw * cosRoll * cosPitch + cosYaw * sinRoll * sinPitch;
q->z = cosYaw * sinRoll * cosPitch - sinYaw * cosRoll * sinPitch;
Quaternion3DNormalise(q);
}
static inline void QuatToMatrix(M3DMatrix44f matrix, Quaternion3D *q) {
//first column
matrix[0] = 1 - 2 * (q->y * q->y + q->z + q->z);
matrix[1] = 2 * (q->x * q->y * q->z * q->w);
matrix[2] = 2 * (q->x * q->z * q->y * q->w);
matrix[3] = 0;
//second column
matrix[4] = 2 * (q->x * q->y - q->z * q->w);
matrix[5] = 1 - 2 * (q->x * q->x + q->z * q->z);
matrix[6] = 2 * (q->y * q->z + q->x * q->w);
matrix[7] = 0;
//third column
matrix[8] = 2 * (q->x * q->z + q->y * q->w);
matrix[9] = 2 * (q->y * q->z - q->x * q->w);
matrix[10] = 1 - 2 * (q->x * q->x - q->y * q->y);
matrix[11] = 0;
//fourth column
matrix[12] = 0;
matrix[13] = 0;
matrix[14] = 0;
matrix[15] = 1;
}
void setupRC()
{
// Black background
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
shaderManager.InitializeStockShaders();
glEnable(GL_DEPTH_TEST);
transformPipeline.SetMatrixStacks(modelViewMatrix, projectionMatrix);
cameraFrame.MoveForward(-15.0f);
cam_x -= 15.0f;
cameraFrame.TranslateWorld(0.0f, 5.0f, 0.0f);
cam_y += 5.0f;
GLfloat cVerts[] = {
//FRONT FACE
0.0f, 0.0f, 0.0f,
1.0f, 0.0f, 0.0f,
1.0f, 1.0f, 0.0f,
0.0f, 1.0f, 0.0f,
//BACK FACE
0.0f, 0.0f, 1.0f,
0.0f, 1.0f, 1.0f,
1.0f, 1.0f, 1.0f,
1.0f, 0.0f, 1.0f,
//LEFT FACE
0.0f, 0.0f, 1.0f,
0.0f, 0.0f, 0.0f,
0.0f, 1.0f, 0.0f,
0.0f, 1.0f, 1.0f,
//RIGHT FACE
1.0f, 0.0f, 0.0f,
1.0f, 0.0f, 1.0f,
1.0f, 1.0f, 1.0f,
1.0f, 1.0f, 0.0f,
//TOP FACE
0.0f, 1.0f, 0.0f,
1.0f, 1.0f, 0.0f,
1.0f, 1.0f, 1.0f,
0.0f, 1.0f, 1.0f,
//BOTTOM FACE
0.0f, 0.0f, 0.0f,
0.0f, 0.0f, 1.0f,
1.0f, 0.0f, 1.0f,
1.0f, 0.0f, 0.0f
};
cubeBatch.Begin(GL_QUADS, 24);
cubeBatch.CopyVertexData3f(cVerts);
cubeBatch.End();
for(int i=0; i < CUBE_COUNT; i++) {
GLfloat x = (GLfloat)(i / int(sqrt(double(CUBE_COUNT))));
GLfloat z = (GLfloat)(i % int(sqrt(double(CUBE_COUNT))));
cubes[i].SetOrigin(x, 0.0f, z);
}
}
void drawWireFramedBatch(GLBatch* pBatch) {
//Draws the batch solid green
shaderManager.UseStockShader(GLT_SHADER_FLAT, transformPipeline.GetModelViewProjectionMatrix(), vGreen);
pBatch->Draw();
//Draws the outline black
glPolygonOffset(-1.0f, -1.0f);
glEnable(GL_POLYGON_OFFSET_LINE);
//Draws the lines antialiased
glEnable(GL_LINE_SMOOTH);
glEnable(GL_BLEND);
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
//Draws black wireframe version of geometry
glPolygonMode(GL_FRONT_AND_BACK, GL_LINE);
glLineWidth(1.0f);
shaderManager.UseStockShader(GLT_SHADER_FLAT, transformPipeline.GetModelViewProjectionMatrix(), vBlack);
pBatch->Draw();
//Reset everything
glPolygonMode(GL_FRONT_AND_BACK, GL_FILL);
glDisable(GL_POLYGON_OFFSET_LINE);
glLineWidth(1.0f);
glDisable(GL_BLEND);
glDisable(GL_LINE_SMOOTH);
}
///////////////////////////////////////////////////////////////////////////////
// Called to draw scene
void renderScene(void) {
//Cubes are green
static GLfloat vCubeColour[] = {0.0f, 1.0f, 0.0f, 1.0f};
// Clear the window with current clearing color
glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT | GL_STENCIL_BUFFER_BIT);
//This pushes the identity matrix onto the matrix stack
modelViewMatrix.PushMatrix();
//printf("%i, %i, %i\n", cam_x, cam_y, cam_z);
//cout << "PITCH: " << pitch << " YAW: " << yaw << " ROLL: " << roll << endl;
float forwards = cam_x * cos(m3dDegToRad(yaw)) + cam_z * -sin(m3dDegToRad(yaw));
float sideways = cam_x * sin(m3dDegToRad(yaw)) + cam_z * cos(m3dDegToRad(yaw));
float upwards = cam_x * sin(m3dDegToRad(pitch));
//cameraFrame.MoveForward(forwards);
//cameraFrame.MoveUp(upwards);
//cameraFrame.MoveRight(sideways);
//cameraFrame.RotateLocal(m3dDegToRad(yaw), 0.0f, 1.0f, 0.0f);
//cameraFrame.RotateLocal(m3dDegToRad(-pitch), 1.0f, 0.0f, 0.0f);
//cameraFrame.RotateLocal(m3dDegToRad(roll), 0.0f, 0.0f, 1.0f);
//create matrix for camera
M3DMatrix44f mCamera;
cameraFrame.GetCameraMatrix(mCamera);
modelViewMatrix.MultMatrix(mCamera);
modelViewMatrix.MultMatrix(camMatrix);
M3DVector4f vLightPos = {0.0f, 10.0f, 5.0f, 1.0f };
M3DVector4f vLightEyePos;
m3dTransformVector4(vLightEyePos, vLightPos, mCamera);
//create matrix for object handling
M3DMatrix44f mObjectFrame;
objectFrame.GetMatrix(mObjectFrame);
modelViewMatrix.MultMatrix(mObjectFrame);
//use a basic stock shader - pass in modelview projection matrix
//shaderManager.UseStockShader(GLT_SHADER_FLAT, transformPipeline.GetModelViewProjectionMatrix(), vBlack);
for(int i=0; i < CUBE_COUNT; i++) {
modelViewMatrix.PushMatrix();
modelViewMatrix.MultMatrix(cubes[i]);
drawWireFramedBatch(&cubeBatch);
modelViewMatrix.PopMatrix();
}
modelViewMatrix.PopMatrix();
// Perform the buffer swap to display back buffer
glutSwapBuffers();
}
void resetCameraFrame() {
cameraFrame = GLFrame();
cameraFrame.MoveForward(-15.0f);
cameraFrame.TranslateWorld(0.0f, 5.0f, 0.0f);
}
void getMouseCoords(int x, int y) {
mouse_x = x;
mouse_y = y;
}
void update(void) {
float turn_angle = 0.005f;
float move_speed = 0.5f;
bool rotate = false;
float rotation;
//HANDLE CAMERA ROTATION
yaw += turn_angle * (WINDOW_WIDTH / 2 - mouse_x);
pitch += turn_angle * (WINDOW_HEIGHT / 2 - mouse_y);
pitch = pitch > 360 ? pitch - 360.0f : pitch;
pitch = pitch < -360 ? pitch + 360.0f : pitch;
yaw = yaw > 360 ? yaw - 360.0f : yaw;
yaw = yaw < -360 ? yaw + 360.0f : yaw;
roll = roll > 360 ? roll - 360.0f : roll;
roll = roll < -360 ? roll + 360.0f : roll;
Quaternion3D q = Quaternion3D();
EulerToQuaternion3D(&q, pitch, yaw, roll);
QuatToMatrix(camMatrix, &q);
//cameraFrame.RotateWorld(m3dDegToRad(yaw), 0.0f, 1.0f, 0.0f);
//cameraFrame.RotateWorld(m3dDegToRad(pitch), 1.0f, 0.0f, 0.0f);
//cameraFrame.RotateWorld(m3dDegToRad(roll), 0.0f, 0.0f, 1.0f);
//cout << "PITCH: " << pitch << " YAW: " << yaw << " ROLL: " << roll << endl;
//resetCameraFrame();
//HANDLE CAMERA POSITION
if (w_pressed) { cameraFrame.MoveForward(move_speed); cout << "FORWARD" << endl; }
if (a_pressed) { cameraFrame.MoveRight(move_speed); cout << "LEFT" << endl; }
if (s_pressed) { cameraFrame.MoveForward(-move_speed); cout << "RIGHT" << endl; }
if (d_pressed) { cameraFrame.MoveRight(-move_speed); cout << "BACKWARD" << endl; }
/*
if (w_pressed) { cam_x += move_speed; cout << "FORWARD" << endl; }
if (a_pressed) { cam_z += move_speed; cout << "LEFT" << endl; }
if (s_pressed) { cam_x -= move_speed; cout << "BACKWARD" << endl; }
if (d_pressed) { cam_z -= move_speed; cout << "RIGHT" << endl; }
*/
glutPostRedisplay();
}
void specialKeyPress(int key, int x, int y) {
float linear = 0.1f;
float angular = float(m3dDegToRad(5.0f));
if(key == GLUT_KEY_UP)
cameraFrame.MoveForward(linear);
if(key == GLUT_KEY_DOWN)
cameraFrame.MoveForward(-linear);
if(key == GLUT_KEY_LEFT)
cameraFrame.RotateWorld(angular, 0.0f, 1.0f, 0.0f);
if(key == GLUT_KEY_RIGHT)
cameraFrame.RotateWorld(-angular, 0.0f, 1.0f, 0.0f);
}
void specialKeyRelease(int key, int x, int y) {
}
void normalKeyPress(unsigned char key, int x, int y) {
printf ("The '%c' key was pressed\n", key);
switch(key) {
case 'w':
w_pressed = true;
break;
case 'a':
a_pressed = true;
break;
case 's':
s_pressed = true;
break;
case 'd':
d_pressed = true;
break;
case ESCAPE_KEY:
exit(0);
break;
}
}
void normalKeyRelease(unsigned char key, int x, int y) {
printf ("The '%c' key was released\n", key);
switch(key) {
case 'w':
w_pressed = false;
break;
case 'a':
a_pressed = false;
break;
case 's':
s_pressed = false;
break;
case 'd':
d_pressed = false;
break;
}
}
///////////////////////////////////////////////////////////////////////////////
// Window has changed size, or has just been created. In either case, we need
// to use the window dimensions to set the viewport and the projection matrix.
void changeSize(int w, int h) {
glViewport(0, 0, w, h);
viewFrustum.SetPerspective(35.0f, float(w) / float(h), 1.0f, 500.0f);
projectionMatrix.LoadMatrix(viewFrustum.GetProjectionMatrix());
transformPipeline.SetMatrixStacks(modelViewMatrix, projectionMatrix);
}
///////////////////////////////////////////////////////////////////////////////
// Main entry point for GLUT based programs
int main(int argc, char* argv[])
{
glutInit(&argc,argv);
glutInitDisplayMode(GLUT_DOUBLE|GLUT_RGBA|GLUT_DEPTH|GLUT_STENCIL);
glutInitWindowSize(WINDOW_WIDTH, WINDOW_HEIGHT);
glutCreateWindow("Cube");
glutReshapeFunc(changeSize);
glutKeyboardFunc(normalKeyPress);
glutKeyboardUpFunc(normalKeyRelease);
glutSpecialFunc(specialKeyPress);
glutSpecialUpFunc(specialKeyRelease);
glutPassiveMotionFunc(getMouseCoords);
glutDisplayFunc(renderScene);
glutIdleFunc(update);
GLenum err = glewInit();
if(GLEW_OK != err){
fprintf(stderr, "GLEW Error: %s\n", glewGetErrorString(err));
return 1;
}
setupRC();
glutMainLoop();
return 0;
}