所以基本上我正在尝试在代码中创建自己的纹理。具体来说,我正在编写一个光线跟踪器,并希望将结果显示为在屏幕上呈现为四边形的纹理。
但是,如何创建自己的纹理然后将其应用于屏幕四边形呢?我已经在这几天没有任何特别的结果了。我现在得到的是屏幕上的四边形,但纹理是整个四边形的一种颜色。但在我的例子中,我希望它是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
答案 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
};