帧缓冲区(framebuffer object)简称 FBO,用于写入颜色值,写入深度信息和深度缓冲和允许我们根据一些条件丢弃特定片段的模板缓冲。
图像在绘制时最终都是绘制到帧缓冲区中的,一般情况下我们都是使用的是默认FBO,也就是我们的屏幕,也就是着色器各方面绘制结果存储的逻辑对象。
使用glGenFramebuffers 来创建一个帧缓冲对象
unsigned int framebuffer;
glGenFramebuffers(1,&framebuffer);
然后使用glBindFramebuffer 来绑定帧缓冲对象,这个意思表示为将这个帧缓冲对象是激活的,之后的读写帧缓冲的操作都会使用我们绑定的这个FBO
glBindFramebuffer(GL_FRAMEBUFFER,framebuffer);
如果要解绑定的话,也就是恢复到默认缓冲区
glBindFramebuffer(GL_FRAMEBUFFER,0);
一个完整的帧缓冲区需要满足如下的条件:
把一个纹理附件添加到缓冲区后,之后所有的渲染指令都被写入这个纹理之中,可以把它当作一个普通的颜色/深度或模板缓冲一样
glGenTextures(1,&texColorBuffer); //创建纹理
glBindTexture(GL_TEXTURE_2D,texColorBuffer);
glTexImage2D(GL_TEXTURE_2D,0,GL_RGB,800,600,0,GL_RGB,GL_UNSIGNED_BYTE,0);//最后一个参数为0表示仅仅分配内存但是没有填充
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
glFramebufferTexture2D(GL_FRAMEBUFFER,GL_COLOR_ATTACHMENT0,GL_TEXTURE_2D,texColorBuffer,0);//附加到帧缓冲区上
在渲染前绑定你的FBO,调用渲染指令将效果渲染到纹理
glViewport(0,0,800,600);
glBindFramebuffer(GL_FRAMEBUFFER,framebuffer);
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D,texture1);
glActiveTexture(GL_TEXTURE1);
glBindTexture(GL_TEXTURE_2D,texture2);
glUseProgram(program);
glClearColor(1.0,0.0,0.0,1.0);
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glBindVertexArray(VAO);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER,EBO);
glDrawElements(GL_TRIANGLES,6,GL_UNSIGNED_INT,0);
//glDrawArrays(GL_TRIANGLES,0,6);
glBindFramebuffer(GL_FRAMEBUFFER,0);
首先在准备好两张图片
在进行相关的opengl shader和program初始化后,加载图片数据到两张纹理中:
ShaderManager *shaderManager = new ShaderManager();
shaderManager->initVertexData(nullptr);
shaderManager->loadShader();
shaderManager->createProgram();
shaderManager->loadImage();
void ShaderManager::loadImage()
{
int img_width;
int img_height;
int img_nrChannels;
unsigned char *img_data;
stbi_set_flip_vertically_on_load(true);
img_data = stbi_load("/Users/bytedance/Desktop/container.jpg",&img_width,&img_height,&img_nrChannels,0);
if(img_data == NULL){
std::cout<<"can not get image data"<<std::endl;
}
std::cout<<"image width-height: "<<img_width<<"-"<<img_height<<std::endl;
glGenTextures(1,&texture1);
glBindTexture(GL_TEXTURE_2D,texture1);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_S,GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_T,GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
glTexImage2D(GL_TEXTURE_2D,0,GL_RGB,img_width,img_height,0,GL_RGB,GL_UNSIGNED_BYTE,img_data);
glGenerateMipmap(GL_TEXTURE_2D);
stbi_image_free(img_data);
unsigned char *data2 = stbi_load("/Users/bytedance/Desktop/awesomeface.png",&img_width,&img_height,&img_nrChannels,0);
if(data2 == NULL){
std::cout<<"can not get image2 data"<<std::endl;
}
std::cout<<"image width-height: "<<img_width<<"-"<<img_height<<std::endl;
glGenTextures(1,&texture2);
glBindTexture(GL_TEXTURE_2D,texture2);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_S,GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_WRAP_T,GL_REPEAT);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
glTexImage2D(GL_TEXTURE_2D,0,GL_RGBA,img_width,img_height,0,GL_RGBA,GL_UNSIGNED_BYTE,data2);
glGenerateMipmap(GL_TEXTURE_2D);
stbi_image_free(data2);
}
之后得到你的shader相关需要进行采样的纹理对象的位置:
void ShaderManager::initLocationValue(bool screen)
{
glUseProgram(program);
if(!screen){
glUniform1i(glGetUniformLocation(program,"texture1"),0);
glUniform1i(glGetUniformLocation(program,"texture2"),1);
} else{
glUniform1i(glGetUniformLocation(program,"screenTexture"),0);
}
}
然后创建帧缓冲区,绑定纹理:
void ShaderManager::createFrameBuffer()
{
glGenFramebuffers(1,&framebuffer);
glBindFramebuffer(GL_FRAMEBUFFER,framebuffer);
glGenTextures(1,&texColorBuffer);
glBindTexture(GL_TEXTURE_2D,texColorBuffer);
glTexImage2D(GL_TEXTURE_2D,0,GL_RGB,800,600,0,GL_RGB,GL_UNSIGNED_BYTE,0);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MIN_FILTER,GL_LINEAR);
glTexParameteri(GL_TEXTURE_2D,GL_TEXTURE_MAG_FILTER,GL_LINEAR);
glFramebufferTexture2D(GL_FRAMEBUFFER,GL_COLOR_ATTACHMENT0,GL_TEXTURE_2D,texColorBuffer,0);
// unsigned int rbo;
// glGenRenderbuffers(1, &rbo);
// glBindRenderbuffer(GL_RENDERBUFFER, rbo);
// glRenderbufferStorage(GL_RENDERBUFFER, GL_DEPTH24_STENCIL8, 800, 600); // use a single renderbuffer object for both a depth AND stencil buffer.
// glFramebufferRenderbuffer(GL_FRAMEBUFFER, GL_DEPTH_STENCIL_ATTACHMENT, GL_RENDERBUFFER, rbo); // now actually attach it
glBindTexture(GL_TEXTURE_2D,0);
glBindFramebuffer(GL_FRAMEBUFFER, 0);
}
之后进行渲染:
这样就将这两张纹理混合后的结果保存到了帧缓冲区中
void ShaderManager::render()
{
glViewport(0,0,800,600);
glBindFramebuffer(GL_FRAMEBUFFER,framebuffer);
glActiveTexture(GL_TEXTURE0);
glBindTexture(GL_TEXTURE_2D,texture1);
glActiveTexture(GL_TEXTURE1);
glBindTexture(GL_TEXTURE_2D,texture2);
glUseProgram(program);
glClearColor(1.0,0.0,0.0,1.0);
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glBindVertexArray(VAO);
glBindBuffer(GL_ELEMENT_ARRAY_BUFFER,EBO);
glDrawElements(GL_TRIANGLES,6,GL_UNSIGNED_INT,0);
//glDrawArrays(GL_TRIANGLES,0,6);
glBindFramebuffer(GL_FRAMEBUFFER,0);
}
最后再渲染上屏:
ShaderManager *screen = new ShaderManager();
screen->initScreenData();
screen->loadShader(vs,fs);
screen->createProgram();
screen->render_to_screen(shaderManager->texColorBuffer);