SWBF2-Classic-Msh-Viewer/QtMeshViewer/Source/GeometryEngine.cpp

164 lines
5.0 KiB
C++
Raw Normal View History

#include "..\Header\GeometryEngine.h"
#include <QVector2D>
#include <QVector3D>
2016-12-29 12:37:15 +00:00
struct VertexData
{
2016-12-29 12:37:15 +00:00
QVector3D position;
QVector2D texCoord;
};
2016-12-29 13:06:25 +00:00
/////////////////////////////////////////////////////////////////////////
// public constructor/destructor
GeometryEngine::GeometryEngine()
2016-12-29 13:06:25 +00:00
: m_indexBuf(QOpenGLBuffer::IndexBuffer)
{
2016-12-29 12:37:15 +00:00
initializeOpenGLFunctions();
2016-12-29 12:37:15 +00:00
// Generate 2 VBOs
2016-12-29 13:06:25 +00:00
m_arrayBuf.create();
m_indexBuf.create();
2016-12-29 12:37:15 +00:00
// Initializes cube geometry and transfers it to VBOs
initCubeGeometry();
}
GeometryEngine::~GeometryEngine()
{
2016-12-29 13:06:25 +00:00
m_arrayBuf.destroy();
m_indexBuf.destroy();
for (auto it : m_textures)
delete it;
m_textures.clear();
m_textures.squeeze();
}
2016-12-29 13:06:25 +00:00
/////////////////////////////////////////////////////////////////////////
// private functions
void GeometryEngine::initCubeGeometry()
{
2016-12-29 13:06:25 +00:00
QVector<VertexData> vertices = {
// Vertex data for face 0
{QVector3D(-1.0f, -1.0f, 1.0f), QVector2D(0.0f, 0.0f)}, // v0
{QVector3D( 1.0f, -1.0f, 1.0f), QVector2D(0.33f, 0.0f)}, // v1
{QVector3D(-1.0f, 1.0f, 1.0f), QVector2D(0.0f, 0.5f)}, // v2
{QVector3D( 1.0f, 1.0f, 1.0f), QVector2D(0.33f, 0.5f)}, // v3
// Vertex data for face 1
{QVector3D( 1.0f, -1.0f, 1.0f), QVector2D( 0.0f, 0.5f)}, // v4
{QVector3D( 1.0f, -1.0f, -1.0f), QVector2D(0.33f, 0.5f)}, // v5
{QVector3D( 1.0f, 1.0f, 1.0f), QVector2D(0.0f, 1.0f)}, // v6
{QVector3D( 1.0f, 1.0f, -1.0f), QVector2D(0.33f, 1.0f)}, // v7
// Vertex data for face 2
{QVector3D( 1.0f, -1.0f, -1.0f), QVector2D(0.66f, 0.5f)}, // v8
{QVector3D(-1.0f, -1.0f, -1.0f), QVector2D(1.0f, 0.5f)}, // v9
{QVector3D( 1.0f, 1.0f, -1.0f), QVector2D(0.66f, 1.0f)}, // v10
{QVector3D(-1.0f, 1.0f, -1.0f), QVector2D(1.0f, 1.0f)}, // v11
// Vertex data for face 3
{QVector3D(-1.0f, -1.0f, -1.0f), QVector2D(0.66f, 0.0f)}, // v12
{QVector3D(-1.0f, -1.0f, 1.0f), QVector2D(1.0f, 0.0f)}, // v13
{QVector3D(-1.0f, 1.0f, -1.0f), QVector2D(0.66f, 0.5f)}, // v14
{QVector3D(-1.0f, 1.0f, 1.0f), QVector2D(1.0f, 0.5f)}, // v15
// Vertex data for face 4
{QVector3D(-1.0f, -1.0f, -1.0f), QVector2D(0.33f, 0.0f)}, // v16
{QVector3D( 1.0f, -1.0f, -1.0f), QVector2D(0.66f, 0.0f)}, // v17
{QVector3D(-1.0f, -1.0f, 1.0f), QVector2D(0.33f, 0.5f)}, // v18
{QVector3D( 1.0f, -1.0f, 1.0f), QVector2D(0.66f, 0.5f)}, // v19
// Vertex data for face 5
{QVector3D(-1.0f, 1.0f, 1.0f), QVector2D(0.33f, 0.5f)}, // v20
{QVector3D( 1.0f, 1.0f, 1.0f), QVector2D(0.66f, 0.5f)}, // v21
{QVector3D(-1.0f, 1.0f, -1.0f), QVector2D(0.33f, 1.0f)}, // v22
{QVector3D( 1.0f, 1.0f, -1.0f), QVector2D(0.66f, 1.0f)} // v23
};
2016-12-29 13:06:25 +00:00
QVector<GLushort> indices = {
2016-12-29 12:37:15 +00:00
0,1,2, //vorne (4)
3,2,1,
4,5,7, //rechts (1)
6,4,7,
8,9,11, //hinten (3)*
8,11,10,
14,12,13, //links (6)*
14,13,15,
18,16,17, //unten (5)
19,18,17,
23,22,20, //oben (2)*
23,20,21
2016-12-29 13:06:25 +00:00
};
2016-12-29 13:06:25 +00:00
// Transfer vertex data to VBO 0
m_arrayBuf.bind();
m_arrayBuf.allocate(vertices.data(), vertices.size() * sizeof(VertexData));
2016-12-29 13:06:25 +00:00
// Transfer index data to VBO 1
m_indexBuf.bind();
m_indexBuf.allocate(indices.data(), indices.size() * sizeof(GLushort));
2016-12-29 12:37:15 +00:00
2016-12-29 13:06:25 +00:00
// load the texture
2016-12-29 12:37:15 +00:00
initTexture();
}
2016-12-29 12:37:15 +00:00
void GeometryEngine::initTexture()
{
2016-12-29 12:37:15 +00:00
// Load cube.png image
2016-12-29 13:06:25 +00:00
QOpenGLTexture* new_texture = new QOpenGLTexture(QImage(":images/cube.png").mirrored());
2016-12-29 12:37:15 +00:00
// Set nearest filtering mode for texture minification
2016-12-29 13:06:25 +00:00
new_texture->setMinificationFilter(QOpenGLTexture::Nearest);
2016-12-29 12:37:15 +00:00
// Set bilinear filtering mode for texture magnification
2016-12-29 13:06:25 +00:00
new_texture->setMagnificationFilter(QOpenGLTexture::Linear);
2016-12-29 12:37:15 +00:00
// Wrap texture coordinates by repeating
// f.ex. texture coordinate (1.1, 1.2) is same as (0.1, 0.2)
2016-12-29 13:06:25 +00:00
new_texture->setWrapMode(QOpenGLTexture::Repeat);
m_textures.push_back(new_texture);
2016-12-29 12:37:15 +00:00
}
2016-12-29 13:06:25 +00:00
/////////////////////////////////////////////////////////////////////////
// public functions
2016-12-29 12:37:15 +00:00
void GeometryEngine::drawGeometry(QOpenGLShaderProgram *program)
{
// Tell OpenGL which VBOs to use
2016-12-29 13:06:25 +00:00
m_arrayBuf.bind();
m_indexBuf.bind();
m_textures.first()->bind();
2016-12-29 12:37:15 +00:00
// Use texture unit 0 which contains cube.png
program->setUniformValue("texture", 0);
2016-12-29 12:37:15 +00:00
// Offset for position
quintptr offset = 0;
2016-12-29 12:37:15 +00:00
// Tell OpenGL programmable pipeline how to locate vertex position data
int vertexLocation = program->attributeLocation("a_position");
program->enableAttributeArray(vertexLocation);
program->setAttributeBuffer(vertexLocation, GL_FLOAT, offset, 3, sizeof(VertexData));
2016-12-29 12:37:15 +00:00
// Offset for texture coordinate
offset += sizeof(QVector3D);
2016-12-29 12:37:15 +00:00
// Tell OpenGL programmable pipeline how to locate vertex texture coordinate data
int texcoordLocation = program->attributeLocation("a_texcoord");
program->enableAttributeArray(texcoordLocation);
program->setAttributeBuffer(texcoordLocation, GL_FLOAT, offset, 2, sizeof(VertexData));
2016-12-29 12:37:15 +00:00
// Draw cube geometry using indices from VBO 1
glDrawElements(GL_TRIANGLES, 36, GL_UNSIGNED_SHORT, 0);
}