2

我正在尝试使用带有 Qt 的 OpenGL 中的顶点缓冲区对象绘制一个样方。这是我的几何:

numVertices = 4;
vertices = new float[3*numVertices];
int i = 0;
vertices[i++] = 0.0f; vertices[i++] = 0.0f; vertices[i++] = 0.0f; // (0,0,0)
vertices[i++] = 1.0f; vertices[i++] = 0.0f; vertices[i++] = 0.0f; // (1,0,0)
vertices[i++] = 1.0f; vertices[i++] = 1.0f; vertices[i++] = 0.0f; // (1,1,0)
vertices[i++] = 0.0f; vertices[i++] = 1.0f; vertices[i++] = 0.0f; // (0,1,0)
i = 0;
// spilt quad into two triangles:
numTriangles = 2;
indices = new unsigned int[numTriangles*3];
indices[i++] = 0; indices[i++] = 1; indices[i++] = 2; 
indices[i++] = 0; indices[i++] = 2; indices[i++] = 3;

接下来在 initializeGL 方法中:

QGLBuffer vertexBuffer;
vertexBuffer.create();
vertexBuffer.bind();
vertexBuffer.allocate(vertices, numVertices*sizeof(float));

QGLShaderProgram* shaderProgram_ = new QGLShaderProgram;
shaderProgram_->addShaderFromSourceFile(QGLShader::Vertex,"C:/src/light.vert.glsl") ) {
shaderProgram_->addShaderFromSourceFile(QGLShader::Fragment, "C:/src/light.frag.glsl");
bool ok = shaderProgram_->link();
ok = shaderProgram_->bind();

我认为 VBO 部分所做的就是将顶点复制到 GPU?(为什么这么多行?)

着色器部分适用于旧式 glBegin(GL_QUADS);

接下来在我的paintGL方法中:

shaderProgram_->setAttributeBuffer("vertex", GL_FLOAT, 0, 3, 0);
shaderProgram_->enableAttributeArray("vertex");
glDrawElements(GL_TRIANGLES, numTriangles, GL_UNSIGNED_INT, indices);

前两条线在做什么?也许告诉着色器有一个名为“顶点”的顶点缓冲区,类型为 GL_FLOAT?但是我在创建 VBO 时没有指定任何名称!?OpenGL如何知道这是“顶点”?

反正我什么都没看到!?有没有我遗漏的步骤?我的着色器很简单:

# version 120  
varying vec4 color;
void main() {
    vec4 vertex = gl_Vertex;
    // pass trough:
    gl_Position = gl_ProjectionMatrix * gl_ModelViewMatrix * vertex; 
    color = gl_Color; 
}

# version 120 
varying vec4 color;
void main (void) 
{   
    // pass-trough:    
 gl_FragColor = color;
 }
4

1 回答 1

2

我猜你在这里输了 3 个:

vertexBuffer.allocate(vertices, numVertices*sizeof(float));

->

vertexBuffer.allocate(vertices, numVertices*sizeof(float)*3);

我认为 VBO 部分所做的就是将顶点复制到 GPU?(为什么这么多行?)

是的。这就是 OpenGL 的工作原理。要在 VBO 中存储某些内容,您必须创建它、绑定它并复制数据。

前两条线在做什么?也许告诉着色器有一个名为“顶点”的顶点缓冲区,类型为 GL_FLOAT?但是我在创建 VBO 时没有指定任何名称!?OpenGL如何知道这是“顶点”?

第一行:

QGLShaderProgram::setAttributeBuffer()“在此着色器程序中设置名为 name 的属性的顶点值数组,从当前绑定的顶点缓冲区中的特定偏移量开始。” - 从手册。同样,这就是 OpenGL 的工作原理。这是一个状态机。您将特定缓冲区绑定到 GL_ARRAY_BUFFER 绑定点,然后告诉 OpenGL 这是存储“顶点”属性数据的缓冲区。“顶点”是您在着色器程序中的属性名称。(也许你应该把它改成“gl_Vertex”)猜猜,Qt 调用glVertexAttribLocation()来查找你的属性的位置和下一个调用glVertexAttribPointer()

第二行:

OpenGL 必须知道它应该将数据从某个缓冲区复制到着色器程序可以找到它的某个特殊位置。这是通过启用特定位置的属性数组来完成的。猜猜,Qt 调用glVertexAttribLocation()glEnableAttribArray()这里。

顺便说一句:您是否在代码中指定了任何模型视图和投影矩阵?我不确定它们是否是默认设置的。尝试从着色器中删除这些值以进行测试。

添加:

glDrawElements(GL_TRIANGLES, numTriangles, GL_UNSIGNED_INT, indices);

第二个参数不是三角形的数量,而是索引的数量,将从索引数组中读取。这里应该是numTriangles*3

于 2012-12-15T16:49:48.680 回答