颜色缓冲区
OpenGL时,先是在一个缓冲区中完毕渲染,然后再把渲染结果交换到屏幕上。
我们把这两个缓冲区称为前颜色缓冲区(屏幕)和后颜色缓冲区。在默认情况下,OpenGL命令是在后颜色缓冲区进行渲染的。当然。也能够直接在前颜色缓冲区中进行渲染。
若要在前颜色缓冲区中进行渲染,第一种方法是直接告诉OpenGL希望在前颜色缓冲区中进行画图,能够调用以下这个函数来实现这个目的:
void glDrawBuffer(Glenum mode);
假设參数mode指定为GL_FRONT。OpenGL就会在前颜色缓冲区中进行渲染;
假设參数mode指定为GL_BACK,那么渲染将在后颜色缓冲区中进行。
在前颜色缓冲区进行渲染的另外一种方法是在OpenGL被初始化时简单地不要求进行双缓冲区渲染。进行单缓冲区渲染时。假设希望把渲染结果实际绘制到屏幕上,须要调用glFlush()或glFinsh(),这点很重要。
OpenGL实现除了支持单纯的前颜色缓冲区和后颜色缓冲区之外。还支持其它模式,如用于立体渲染的左和右缓冲区以及辅助缓冲区。
深度缓冲区
与颜色缓冲区不同的是,深度缓冲区中所填充的是深度值而不是颜色值。
为了启用深度缓冲区进行深度測试,仅仅须要调用:
glEnable(GL_DEPTH_TEST)。
另外。即使深度缓冲区未被启用,假设深度缓冲区被创建,OpenGL也会把全部写入到颜色缓冲区的颜色片段相应的深度值写入到深度缓冲区中。可是,假设我们希望在进行深度測试时暂时禁止把值写入到深度缓冲区,我们能够使用函数:
void glDepthMask(GLboolean mask);
把GL_FALSE作为參数,经禁止写入深度值,但并不禁止用已经写入到深度缓冲区的值进行深度測试。
把GL_TRUE作为參数。能够又一次启用深度缓冲区的写入。同一时候,这也是默认的设置。
裁剪測试
OpenGL同意在窗体中指定一个裁剪矩形。让渲染仅仅在这个区域内进行。
在默认情况下。裁剪矩形就是窗体的大小。不会进行裁剪測试。
我们能够打开裁剪測试:
glEnable(GL_SCISSOR_TEST)。
在窗体内部运行渲染的那个区域称为裁剪框(scissor box),它使用以下这个函数以窗体坐标的形式指定:
void glScissor(Glint x, Glint y, GLsizei width, GLsizei height);
以下的程序清除了3次颜色缓冲区。每次在清除之前都指定一个更小的裁剪框。形成一组重叠的着色矩形。如图所看到的:
/* 程序清单 3-12
* 2014/5/7
*/
#include <glut.h>
#include <math.h>
// 设置渲染状态
void SetupRC()
{
// 设置清除窗体的颜色(黑色背景)
glClearColor(0.0f, 0.0f, 0.0f, 1.0f);
// 设置画图颜色为绿色
glColor3f(0.0f, 1.0f, 0.0f);
}
// 绘制场景(显示回调函数)
void RenderScene()
{
// 设置清除窗体的颜色(蓝色背景)
glClearColor(0.0f, 0.0f, 1.0f, 0.0f);
glClear(GL_COLOR_BUFFER_BIT);
// 把裁剪框设置为一个更小的红色子区域
glClearColor(1.0f, 0.0f, 0.0f, 0.0f);
glScissor(100, 100, 600, 400);
glEnable(GL_SCISSOR_TEST);
glClear(GL_COLOR_BUFFER_BIT);
// 把裁剪框设置为一个更小的绿色子区域
glClearColor(0.0f, 1.0f, 0.0f, 0.0f);
glScissor(200, 200, 400, 200);
glClear(GL_COLOR_BUFFER_BIT);
// 关闭裁剪測试,以便进行下一次渲染
glDisable(GL_SCISSOR_TEST);
glutSwapBuffers();
}
// 当窗体大小改变时由GLUT函数库调用
void ChangeSize(GLsizei w, GLsizei h)
{
// 范围
GLfloat nRange = 100.0f;
// 纵横比
GLfloat aspectRatio;
// 防止被0所除
if (0== h){
h = 1;
}
// 依据窗体大小设置视口
glViewport(0, 0, w, h);
// 选择投影矩阵。并重置坐标系统
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
// 计算窗体的纵横比(像素比)
aspectRatio = (GLfloat) w / (GLfloat)h;
// 定义裁剪区域(依据窗体的纵横比,并使用正投影)
if (w<=h) {//宽 <高
glOrtho(-nRange, nRange,-nRange /aspectRatio, nRange / aspectRatio, -nRange, nRange);
} else{//宽 >高
glOrtho(-nRange * aspectRatio,nRange *aspectRatio, -nRange, nRange,-nRange, nRange);
}
// 选择模型视图矩阵。并重置坐标系统
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
}
int main(int argc,char *argv[])
{
// 传递命令行參数,并对GLUT函数库进行初始化
glutInit(&argc, argv);
// 设置创建窗体时的显示模式(双缓冲区、RGB颜色模式)
glutInitDisplayMode(GLUT_DOUBLE |GLUT_RGB);
// 设置窗体的初始大小
glutInitWindowSize(800, 600);
// 创建窗体
glutCreateWindow("Bounce");
// 设置显示回调函数
glutDisplayFunc(RenderScene);
// 设置当窗体的大小发生变化时的回调函数
glutReshapeFunc(ChangeSize);
// 设置渲染状态
SetupRC();
// 启动GLUT框架的执行。一经调用便不再返回,直到程序终止
glutMainLoop();
return0;
}
版权声明:本文博主原创文章,博客,未经同意不得转载。