参考:http://blog.csdn.net/jnleec/article/details/8141863
参考:http://www.songho.ca/opengl/gl_tessellation.html
http://m.blog.csdn.net/blog/sangni007/9040257
http://blog.csdn.net/wind_hzx/article/details/11830425
环境:windows8.1 ,vs2013
一、简介
OpenGL中认为合法的多边形必须是凸多边形,凹多边形、自交多边形、带孔的多边形等非凸的多边形在OpenGL中绘制会出现出乎意料的结果。例如,在大多数系统中,只有多边形的凸包被填充,而在有些系统中,并非所有的凸包都被填充。OpenGL之所以对合法多边形类型做出限制,是为了更方便地提供能够对符合条件的多边形进行快速渲染的硬件。简单多边形可被快速地渲染,而复杂多边形难以快速检测出来。为了最大限度的提高性能,OpenGL假定多边形是简单的。
二、凹多边形的绘制
虽然在OpenGL中可以使用glBegin(GL_POLYGON)来画一个多边形,但是它只能实现简单的凸多边形。对于一些复杂的多边形,比如凹多边形,或者有实心有空心的多边形,OpenGL的glBegin(GL_POLYGON)就不能满足需求了。
通常可以采用一种叫做"分格化"的方法来画复杂的多边形。非凸多边形最简单的填充方法最简单的应该是GLU 网格化对象GLUtesselator(GLUT库或者libTess库)。要用分格化的方法画多边形,步骤如下:
1. gluNewTess(); //创建一个新的分格化对象
2. gluTessCallback(); //注册回调函数,完成分格化的一些操作,照着写就行了。
3. gluTessProperty(); //设置一些分格化的属性值,如环绕数和环绕规则,用来确定多边形的内部和外部
4. gluTessBeginPolygon(); //开始画多边形
gluTessBeginContour(tessobj);//设置多边形的边线 1
gluTessEndContour(tessobj);//结束设置边线1
gluTessBeginContour(tessobj);//,如果有边线2,设置多边形的边线 2
gluTessEndContour(tessobj);//结束设置边线2
5.gluTessEdnPolygon(); //结束画多边形
6. gluDeleteTess(); //删除分格化对象
绘制代码如下:
void drawGraphics(int type)
{
GLUtesselator * tessobj;
tessobj = gluNewTess();
//注册回调函数
gluTessCallback(tessobj, GLU_TESS_VERTEX, (void (CALLBACK *)())vertexCallback);
gluTessCallback(tessobj, GLU_TESS_BEGIN, (void (CALLBACK *)())beginCallback);
gluTessCallback(tessobj, GLU_TESS_END, (void (CALLBACK *)())endCallback);
gluTessCallback(tessobj, GLU_TESS_ERROR, (void (CALLBACK *)())errorCallback);
//gluTessCallback(tessobj, GLU_TESS_COMBINE, (void (CALLBACK *)())combineCallback);//多边型边自相交的情况下回调用回调函数
gluTessBeginPolygon(tessobj, NULL);
gluTessBeginContour(tessobj);//设置多边形的边线
gluTessVertex(tessobj, g_vertex[k][j], g_vertex[k][j]);
gluTessEndContour(tessobj);
gluTessEndPolygon(tessobj);
gluDeleteTess(tessobj);
}
回调函数:
//顶点的回调函数
void CALLBACK vertexCallback(GLvoid* vertex)
{
GLdouble* pt;
int numb;
pt = (GLdouble*)vertex;
glColor3f(0.8, 0.8, 0.8);
glVertex3d(pt[0],pt[1],pt[2]);
}
void CALLBACK beginCallback(GLenum type)
{
glBegin(type);
}
void CALLBACK endCallback()
{
glEnd();
}
void CALLBACK errorCallback(GLenum errorCode)
{
const GLubyte * estring;
//打印错误类型
estring = gluErrorString(errorCode);
fprintf(stderr, "Tessellation Error: %s/n", estring);
exit(0);
}
三、思考
当然也可以利用回调函数记录分格化的顶点和绘制类型,然后利用数组的绘制函数进行一次性的绘制,以提高绘制效率。