OpenGL模板缓冲区与模板测试

转载 2015年07月10日 16:01:31

原文地址:http://www.blogjava.net/qileilove/archive/2014/01/23/409269.html

帧缓冲区有许多缓冲区构成,这些缓冲区大致分为:

    颜色缓冲区:用于绘图的缓冲区,它包含了颜色索引或者RGBA颜色数据。
     深度缓冲区:存储每个像素的深度值,当启动深度测试时,片段像素深度值和深度缓冲区深度值进行比较,决定片段哪些像素点数据可以替换到颜色缓冲区中。
  模板缓冲区:就像使用纸板和喷漆一样精确的混图一样,当启动模板测试时,通过模板测试的片段像素点会被替换到颜色缓冲区中,从而显示出来,未通过的则不会保存到颜色缓冲区中,从而达到了过滤的功能。
  累积缓冲区:累积缓冲区允许你把渲染到颜色缓冲区的值,拷贝到累积缓冲区。在多次拷贝操作到累积缓冲区时,可以用不同方式的把颜色缓冲区内容和当前累积缓冲区的内容进行重复混合

  模板测试
  模板测试只有存在模板缓冲区的情况下进行,模板测试把像素存储在模板缓冲区的点与一个参考值进行比较(glStencilFunc),根据测试结果,对模板缓冲区的值进行响应的修改glStencilOp
  void glStencilFunc (GLenum func, GLint ref, GLuint mask);
  func:GL_NEVER 从来不能通过
  GL_ALWAYS 永远可以通过(默认值)
  GL_LESS 小于参考值可以通过
  GL_LEQUAL 小于或者等于可以通过
  GL_EQUAL 等于通过
  GL_GEQUAL 大于等于通过
  GL_GREATER 大于通过
  GL_NOTEQUAL 不等于通过
  ref: 参考值
  mask:掩码,书上说模板测试只在哪些对应为1的位上进行。(不是很确定具体作用)
  举例:glStencilFunc (GL_LESS, 1.0, 1.0);模板缓冲区对应的像素点的值如果小于1.0,则通过模板测试
  void glStencilOp (GLenum fail, GLenum zfail, GLenum zpass);
  fail模板测试未通过时该如何变化;zfail表示模板测试通过,但深度测试未通过时该如何变化;zpass表示模板测试和深度测试或者未执行深度测试均通过时该如何变化
  GL_KEEP(不改变,这也是默认值)
  GL_ZERO(回零)
  GL_REPLACE(使用测试条件中的设定值来代替当前模板值)
  GL_INCR(增加1,但如果已经是最大值,则保持不变),
  GL_INCR_WRAP(增加1,但如果已经是最大值,则从零重新开始)
  GL_DECR(减少1,但如果已经是零,则保持不变),
  GL_DECR_WRAP(减少1,但如果已经是零,则重新设置为最大值)
  GL_INVERT(按位取反)
  未启用模板缓冲区

glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT);
glLoadIdentity();
glTranslatef(0, 0, -20);
glColor3f(1.0f,1.0f,1.0f);
dRadius = 5.0*(sqrt(2.0)/2.0);
glBegin(GL_LINE_STRIP);
for (dAngel=0;dAngel<380.0;dAngel+=0.1)
{
glVertex2d(dRadius*cos(dAngel),dRadius*sin(dAngel));
dRadius*=1.003;
}
glEnd();
glColor3f(1.0f,0.0f,0.0f);
glRectf(-5,-5,5,5);

 

初始清除背景填充颜色为蓝色
  以上这部分代码可以用如下3张图表示绘制的过程
  开启模板缓冲区
void init()
{
glClearColor(0,0,1.0,0);
glClearStencil(0);
glClearDepth(1.0f);
glEnable(GL_STENCIL_TEST);
}
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);//1
glLoadIdentity();
glTranslatef(0, 0, -20);
glStencilFunc(GL_NEVER,0x0,0xFF);  //2
glStencilOp(GL_INCR,GL_INCR,GL_INCR);//3
glColor3f(1.0f,1.0f,1.0f);
dRadius = 5.0*(sqrt(2.0)/2.0);
glBegin(GL_LINE_STRIP);
for (dAngel=0;dAngel<380.0;dAngel+=0.1)
{
glVertex2d(dRadius*cos(dAngel),dRadius*sin(dAngel));
dRadius*=1.003;
}
glEnd();
glStencilFunc(GL_NOTEQUAL,0x1,0xFF);  //4
glStencilOp(GL_KEEP,GL_KEEP,KEEP); //5
glColor3f(1.0f,0.0f,0.0f);
glRectf(-5,-5,5,5);


  当执行到1处,3个缓冲区都被清空
  颜色缓冲区:每个像素点颜色都是蓝色
  深度缓冲区:每个像素点深度都是1.0
  模板缓冲区:每个像素点模板值都是0
  执行到2,3处,模板测试条件是从不通过测试,如果不通过测试结果是模板值+1
  接着应用模板测试进行绘制一组点,由于模板测试条件是从不通过测试,所以颜色缓冲器值不会变化,但是绘制的点对应的像素点的模板值变为1,此时
  颜色缓冲区:每个像素点颜色都是蓝色
  深度缓冲区:每个像素点深度都是1.0
  模板缓冲区:点数组对应的模板值是1,其他区域像素点的模板值还是0
  执行到4,5处,模板测试条件是模板值不一定1则通过测试,如果不通过测试结果是模板值+1
   接着应用刚才的模板测试进行绘制一个(-5,-5,5,5)的矩形,在这个矩形区域内,像素点的模板值分为2中,值为1的是上1步的点数组。值为0的是 上一步非的点数组像素点。那个根据模板测试条件,模板值为0的像素点通过测试,可以进行替换颜色缓冲区的值(替换成红色),模板值为0的像素点不能通过测 试,因此不能改变颜色缓冲区的值
  颜色缓冲区:(-5,-5,5,5)区域内 模板值为0的像素点为红色,其他区域都为蓝色
  深度缓冲区:每个像素点深度都是1.0
  模板缓冲区:点数组对应的模板值是1,其他区域像素点的模板值还是0
模板查询
  可以用glGetInteger函数获取与模板相关的参数值
#include "header.h"
float dRadius =0;
float dAngel;
float aspect=0;
void init()
{
glClearColor(0,0,1.0,0);
glClearStencil(0);
glClearDepth(1.0f);
glEnable(GL_STENCIL_TEST);
}
void display()
{
glClear(GL_COLOR_BUFFER_BIT|GL_DEPTH_BUFFER_BIT|GL_STENCIL_BUFFER_BIT);
glLoadIdentity();
glTranslatef(0, 0, -20);
////glStencilFunc(GL_ALWAYS, 0,0x00);
glStencilFunc(GL_NEVER,0x0,0xFF);
glStencilOp(GL_INCR,GL_INCR,GL_INCR);
glColor3f(1.0f,1.0f,1.0f);
dRadius = 5.0*(sqrt(2.0)/2.0);
glBegin(GL_LINE_STRIP);
for (dAngel=0;dAngel<380.0;dAngel+=0.1)
{
glVertex2d(dRadius*cos(dAngel),dRadius*sin(dAngel));
dRadius*=1.003;
}
glEnd();
glStencilFunc(GL_NOTEQUAL,0x1,0xFF);
glStencilOp(GL_INCR,GL_INCR,GL_INCR);//
glColor3f(1.0f,0.0f,0.0f);
glRectf(-5,-5,5,5);
glutSwapBuffers();
}
void reshape(int w, int h)
{
glViewport(0,0,w,h);
aspect = (w*1.0)/h;
glMatrixMode(GL_PROJECTION);
glLoadIdentity();
gluPerspective(60, aspect, 1, 100);
glMatrixMode(GL_MODELVIEW);
glLoadIdentity();
}
int main(int argc, char** argv)
{
glutInit(&argc, argv);
//glutInitDisplayMode(GLUT_DOUBLE|GLUT_RGBA|GLUT_STENCIL);
glutInitDisplayMode(GLUT_DOUBLE|GLUT_RGBA|GLUT_STENCIL|GLUT_DEPTH);
glutInitWindowPosition(200,200);
glutInitWindowSize(600,600);
glutCreateWindow("模板缓冲区与模板测试");
glewInit();
init();
glutReshapeFunc(reshape);
glutDisplayFunc(display);
glutMainLoop();
return 0;
}


OPenGL模板缓冲区示例程序

模板测试是把像素存储在模板缓冲区的值与一个参考值进行比较。根据测试的结果,对模板缓冲区中得这个值进行相应的修改。 Note:模板测试只有在存在模板缓冲区的情况下才会执行,如果不存在模板缓冲区,模板测...
  • danshiming
  • danshiming
  • 2017年01月09日 17:11
  • 406

OpenGL: 模板缓冲区

相信大家有些人对OpenGL的模板缓冲区不是很理解,包括我最开始也是,OpenGL的模板缓冲区其实就是采用过滤的技术来控制那些颜色可以绘制,那些不能进行绘制。这里的过滤技术也就是我们的一个控制方法,主...
  • Augusdi
  • Augusdi
  • 2014年03月04日 14:01
  • 5580

OpenGL学习二十九:模板缓冲区与模板测试

帧缓冲区有许多缓冲区构成,这些缓冲区大致分为: 颜色缓冲区:用于绘图的缓冲区,它包含了颜色索引或者RGBA颜色数据。 深度缓冲区:存储每个像素的深度值,当启动深度测试时,片段像素深度值和深度缓...
  • tiankefeng19850520
  • tiankefeng19850520
  • 2014年01月22日 09:15
  • 670

OpenGL中的缓冲区

OpenGL中的缓冲区 颜色缓冲区          OpenGL在绘制图元时,先是在一个缓冲区中完成渲染,然后再把渲染结果交换到屏幕上。我们把这两个缓冲区称为前颜色缓冲区(屏幕)和后颜色缓冲区。...
  • Haohan_Meng
  • Haohan_Meng
  • 2014年05月07日 22:06
  • 8377

OpenGL深入探索——缓冲区对象(*BO)

转载自:http://blog.csdn.net/afei198409/article/details/8267300 缓冲区对象 许多OpenGL操作都向OpenGL发送一大块数据,例如向它传递需...
  • panda1234lee
  • panda1234lee
  • 2016年05月31日 15:51
  • 911

OpenGL缓冲区总结

帧缓冲区   几乎每个图形程序的重要目标之一都是在屏幕上绘制图形。屏幕是由一个矩形像素数组组成的,每个像素都可以在图像的某一个点上显示一个某种颜色的微小方块。在光栅化阶段(包括纹理和雾)之后,数据就...
  • u014800094
  • u014800094
  • 2016年11月04日 15:05
  • 237

OpenGL--帧缓冲区

理论基础 1,帧缓冲区(显存):是由像素组成的二维数组,每一个存储单元对应屏幕上的一个像素,整个帧缓冲对应一帧图像即当前屏幕画面。帧缓冲通常包括:颜色缓冲,深度缓冲,模板缓冲和累积缓冲。这些缓冲区可...
  • u010223072
  • u010223072
  • 2015年04月27日 11:05
  • 3919

OpenGL: 模板缓冲区的理解

引用:http://blog.163.com/danshiming@126/blog/static/109412748201632895137788/ 这几天看《OPenGL 编程指南》中的模板...
  • lizhen37031
  • lizhen37031
  • 2017年12月19日 18:20
  • 17

Opengl 单双缓冲区区别

转自http://blog.csdn.net/mfcappwizard/article/details/6965617 单缓冲,实际上就是将所有的绘图指令在窗口上执行,就是直接在窗口上绘图,这样的绘...
  • qq_28453735
  • qq_28453735
  • 2016年03月20日 10:43
  • 1375

OpenGL: 模板缓冲区

相信大家有些人对OpenGL的模板缓冲区不是很理解,包括我最开始也是,OpenGL的模板缓冲区其实就是采用过滤的技术来控制那些颜色可以绘制,那些不能进行绘制。这里的过滤技术也就是我们的一个控制方法,主...
  • kalongkaya
  • kalongkaya
  • 2015年07月01日 15:46
  • 233
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:OpenGL模板缓冲区与模板测试
举报原因:
原因补充:

(最多只允许输入30个字)