创建原始OpenGL纹理数据(C ++)

时间:2017-10-19 13:13:31

标签: c++ opengl glsl glew

所以基本上我正在尝试在代码中创建自己的纹理。具体来说,我正在编写一个光线跟踪器,并希望将结果显示为在屏幕上呈现为四边形的纹理。

但是,如何创建自己的纹理然后将其应用于屏幕四边形呢?我已经在这几天没有任何特别的结果了。我现在得到的是屏幕上的四边形,但纹理是整个四边形的一种颜色。但在我的例子中,我希望它是1/3白色和2/3黑色,这就是我想象它要完成的。

我希望尽可能保持OpenGL的准确性。我不打算扩展它,所以任何包含“丑陋黑客”的答案只要它有效就完全没问题。我正在使用Glew,GLFW和GLM。

这就是我目前的做法:

GLFWwindow* window;

//Create a texture that should be 1 third black and 2 thirds white?
unsigned char texdata[SCREEN_HEIGHT * SCREEN_WIDTH * 3];
for(int i=0; i<SCREEN_HEIGHT * SCREEN_WIDTH * 3; i++)
{
    if(i < SCREEN_HEIGHT * SCREEN_WIDTH)
        texdata[i] = 255;
    else
        texdata[i] = 0;
}
//cam.createTexture(texdata);

initOpenGL(window);

GLuint t = 0;

glEnable(GL_TEXTURE_2D); // Required for glBuildMipmap() to work (!)
glGenTextures( 1, &t );
glBindTexture(GL_TEXTURE_2D, t);
// Set parameters to determine how the texture is resized
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MIN_FILTER , GL_LINEAR_MIPMAP_LINEAR );
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_MAG_FILTER , GL_LINEAR );
// Set parameters to determine how the texture wraps at edges
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_WRAP_S , GL_REPEAT );
glTexParameteri ( GL_TEXTURE_2D , GL_TEXTURE_WRAP_T , GL_REPEAT );
// Read the texture data from file and upload it to the GPU
glTexImage2D(GL_TEXTURE_2D, 0, GL_RGB, SCREEN_WIDTH, SCREEN_HEIGHT, 0,
             GL_RGB, GL_UNSIGNED_BYTE, texdata);
glGenerateMipmap(GL_TEXTURE_2D);

GLuint programID = LoadShaders("vertex.glsl", "fragment.glsl");
glUseProgram(programID);
// Create and compile our GLSL program from the shaders

Objects screenquad;
screenquad.createBox(1.0, 1.0, 0.0);

do{

    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

    screenquad.render();
    // Swap buffers
    glfwSwapBuffers(window);
    glfwPollEvents();
} // Check if the ESC key was pressed or the window was closed
while( glfwGetKey(window, GLFW_KEY_ESCAPE ) != GLFW_PRESS &&
       glfwWindowShouldClose(window) == 0 );

GLSL着色器代码:

#version 450

in vec2 Texcoord;

out vec4 outColor;

uniform sampler2D tex;

void main(){
    outColor = texture(tex, Texcoord);
}

顶点着色器:

#version 450
layout(location = 0) in vec3 vertexPosition_modelspace;
layout(location = 1) in vec3 Normal;
layout(location = 2) in vec2 texcoord;

out vec2 Texcoord;

void main(){
    Texcoord = texcoord;
    gl_Position.xyz = vertexPosition_modelspace;
    gl_Position.w = 1.0;
}

编辑: 创建screenquad的内部: https://raw.githubusercontent.com/ddavidtran/Fishualization/master/OpenGL/Objects.cpp

1 个答案:

答案 0 :(得分:2)

代码中未设置tex制服。它可以大致如下:

GLint texUniformLocation = glGetUniformLocation(programID, "tex");
glUniform1i(texUniformLocation, 0);

此处glUniform1i中的0是当前有效texture image unit,默认为GL_TEXTURE0。您可以通过glActiveTexture更改它。

但是,正如BDL注意到的那样,0实际上是默认值,因此您可以跳过此部分。

另一个问题是你的纹理坐标是错误的。如file you linked中所示,在函数Objects::createBox中,顶点数据为

const GLfloat vertex_array_data[] = {
    -xsize, -ysize, -zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // Vertex 0
    xsize, -ysize, -zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // Vertex 1
    -xsize, ysize, -zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,  // Vertex 2
    xsize, ysize, -zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,  // Vertex 3
    -xsize, -ysize, zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // Vertex 0
    xsize, -ysize, zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // Vertex 1
    -xsize, ysize, zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f,  // Vertex 2
    xsize, ysize, zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f  // Vertex 3
};

每行中的最后两个浮点数是纹理坐标。正如你所看到的,它们都是0. 法线也很奇怪,但你还是不能使用它们。

我不知道你想要的盒子的纹理坐标是什么,但是对于四边形我会做这样的事情:

const GLfloat vertex_array_data[] = {
    -xsize, -ysize, -zsize, 0.0f, 0.0f, 1.0f, 0.0f, 0.0f, // Vertex 0
    xsize, -ysize, -zsize, 0.0f, 0.0f, 1.0f, 1.0f, 0.0f, // Vertex 1
    -xsize, ysize, -zsize, 0.0f, 0.0f, 1.0f, 0.0f, 1.0f,  // Vertex 2
    xsize, ysize, -zsize, 0.0f, 0.0f, 1.0f, 1.0f, 1.0f,  // Vertex 3
};