一步步学OpenGL(一)

2020-04-18 03:41

一步步学OpenGL(一)

一 打开一个窗口

背景:

官方OpenGL的文档里并没有提供一个API来进行窗口的创建和操作。现在的Windows系统包含一个子系统将OpenGL上下文和Windows 系统绑定在一起从而对OpenGL提供支持。在X窗口系统中那个接口称作GLX。为了支持窗口Windows提供WGL(发音:Wiggle)接口,MacOS则有CGL。使用这些接口直接来创建窗口来显示图形通常非常麻烦,所以我们通常使用一个高水平的库来隐藏窗口的创建操作细节。这里使用的窗口库是GLUT(OpenGL utility library,freeGLUT是GLUT的开源版本,老GLUT早已停止更新),它提供了一个简化的API来操作窗口,以及支持事件处理,IO控制和其他一些功能。另外GLUT是一个跨平台的库所以移植性很好。和GLUT类似的其他替代库还有SDL和GLFW。

源代码详解

(1)glutInit(&argc, argue);

调用这个函数来初始化GLUT.参数可以直接引用command line的,而且有一些有用的选项,比如:‘-sync’和‘-gldebug’,可以禁掉X窗口的异步特征并分别自动检查和显示GL错误。

(2)glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA);

这里配置一些GLUT的选项设置。GLUT_DOUBLE在多数渲染结束后开启双缓冲机制(维护两个图像缓冲数据,屏幕显示一副图像时在后台同时绘制另一份图像缓冲数据,交替显示)和颜色缓冲。我们通常需要这两个设置,还有其他的选项设置后面会继续介绍。 (3)

glutInitWindowSize(960, 640); // 窗口尺寸 glutInitWindowPosition(200, 200); // 窗口位置 glutCreateWindow(\窗口标题

这几个函数的调用可以设置一些窗口参数并创建一个窗口。也可以定义窗口的标题。

(4)glutDisplayFunc(RenderScenceCB);

由于我们是在一个窗口系统中工作的,与运行的程序多数的交互是通过事件回调函数。GLUT针对与底层窗口系统的交互为我们提供了几个回调函数选项。这里我们先只用一个主回调来完成一帧图像的所有渲染工作。这个回调函数会不断地被GLUT内部循环调用。

(5)glClearColor(0.0f, 0.0f, 0.0f, 0.0f);

这个是我们在OpenGL中遇到的第一个状态(OpenGL是一个状态机)。OpenGL使用状态方案的原因是渲染是一个非常复杂的任务,不能仅仅通过一个函数接受几个参数来完成(一个合理设计的函数是不会接受大量的参数的)。对于渲染效果的设置我们需要定义shader着色器,buffer缓冲还有各种各样的flag标志变量。另外,我们也经常想保存一些相同的配置在多个渲染操作中使用(比如:如果我们从来不需要禁掉深度检测depth test,我们没必要在每一个渲染回调中来明确定义它)。这也是为什么多数的渲染操作配置都是通过在OpenGL

状态机中设置flag标志变量和值来完成,而且渲染回调本身通常也被局限于几个参数,参数解决需要绘制的定点数量和他们的偏移量。调用一个改变状态的函数后,具体的配置保持不变,直到下次再调用这个相同的函数再次改变状态和配置。上面的函数设置了当帧缓存(帧缓存后面还会介绍)清空后要使用的颜色值。颜色值有四个通道(RGBA),使用单位化的值0.0-1.0来表示。

(6)glutMainLoop();

这个函数调用传递指令给GLUT现在开始它的内部循环。在这个循环中它监听窗口系统中的事件并通过我们配置的回调传递出去。在我们这个例子中,GLUT将只会调用我们注册的那个display回调(RenderScenceCB),在这个回调函数中(RenderScenceCB)我们可以自定义代码来渲染这一帧的图像。 (7)

glClear(GL_COLOR_BUFFER_BIT); glutSwapBuffers();

在渲染函数中我们能做的就是清空帧缓存(使用上面定义的颜色,可以尝试任意改变颜色看效果)。第二个函数是告诉GLUT交换双缓冲机制中前后两个缓存的角色位置,也就是二者换班,后台的缓存放到前台显示,之前显示的缓存继续到后台开始另一帧的缓存工作。这样在下一个渲染回调循环中交换到当前的缓存将在屏幕上显示。

示例Demo

#include

#include //freeGLUT窗口库 /**

* 渲染回调函数 */

void RenderScenceCB(){ // 清空颜色缓存

glClear(GL_COLOR_BUFFER_BIT); // 交换前后缓存 glutSwapBuffers(); } /**

* 主函数 */

int main(int argc, char ** argv) {

// 初始化GLUT glutInit(&argc, argv);

// 显示模式:双缓冲、RGBA

glutInitDisplayMode(GLUT_DOUBLE | GLUT_RGBA);

// 窗口设置

glutInitWindowSize(480, 320); // 窗口尺寸 glutInitWindowPosition(100, 100); // 窗口位置 glutCreateWindow(\ // 窗口标题

// 开始渲染

glutDisplayFunc(RenderScenceCB);

// 缓存清空后的颜色值

glClearColor(0.0f, 0.0f, 0.0f, 0.0f);

// 通知开始GLUT的内部循环 glutMainLoop();

return 0; }

运行结果:

说明:

demo程序是原作者的代码,适用于多平台,由于在Mac上比较特殊,我在Mac上跑的时候发现GLEW有点问题暂时没解决,导致后面的三角形啥的在mac上没效果。orz??最后我放弃了在mac用opengl了,坑实在太多,装了双系统在windows上安装vs2013来跑原作者的源代码,就顺利多了。

所以强烈建议还是在visual studio上跑opengl吧!学习教程之前配置好开发环境,边看边跑代码效果最好,这里我的环境主要包括: 1.Visual Studio2013;

2.OpenGL,windows默认已内置了OpenGL可直接引入无需再安装(说是微软为了推广自己的DX只提供一个很低的版本,但我的默认就是最新版本,4.3呢!!!);

二 你好顶点

背景

这里要第一次开始使用GLEW(the OpenGL Extension Wrangler Library)库。GLEW可以帮助我们解决一些伴随OpenGL扩展库管理出现的一些头疼的问题,初始化之后,它会检索你平台中所有可用的扩展库,动态的加载并且可以通过简单引用一个头文件来使用。

在这个教程中,我们将第一次使用定点缓冲器对象(VBOs)。顾名思义,VBO是用来存储顶点的。我们试图在屏幕上显示的存在于3d世界中的物体,像一个怪物、城堡或者一个简单的旋转的立方体,都是通过连接一组顶点来实现的。VBOs是在GPU中加载顶点最有效的方式,他们是可以存储在视频内存的缓冲并且可迅速到达GPU处理,所以强烈推荐这种顶点加载方式。

这篇教程和下一篇教程是在这一系列中唯一依赖于固定功能管线的而不是可编程管线,事实上在这所有教程中也没有出现这两种管线的转换,我们只是依靠数据流经管线的方式。在接下来的教程中将会有关于管线的透彻学习,而现在已经足够可以理解在到光栅化(在屏幕上使用屏幕坐标画点、线、三角形等图元)之前,这些可见的顶点都有他们的XYZ坐标([-1.0,1.0]),光栅化程序将这些坐标映射到屏幕空间(例如:如果屏幕宽度是1024,那么X=-1.0就映射到0,X=1.0映射到1023)。最后,光栅化程序根据在draw call(下面代码中会讲到)中定义的拓扑结构来绘制这些图元。由于我们没有将任何shader着色器绑定到渲染管线上,我们的顶点也就没有经过任何变换。也就是说,我们只是给这些顶点一个给定范围的值来使他们可见。将X和Y坐标都设置为0可将顶点精确地至于两个坐标轴中间,也就是屏幕的中央。

安装GLEW:

GLEW可在官网下载:http://glew.sourceforge.net

大多数Linux发行版提供预先构建的包。在Ubuntu上可以通过下面的指令安装: apt-get install libglew1.6 libglew1.6-dev Mac上安装GLEW的教程(需要安装MacPort):http://blog.csdn.net/huyisu/article/details/42742379

VisualStudio安装GLEW:http://blog.csdn.net/xuguangsoft/article/details/8002375

源代码详解

(1)#include

GLEW库引入(一定要在GLUT引入之前引入,否则会编译错误),如果要引入其他OpenGL头文件,必须要注意将这个头文件放在前面。为了将项目与GLEW进行连接需要在makefile中添加‘-lGLEW’(Mac上这个在GLEW的安装教程上有说明,需要在Building Setting中设置).

(2)#include \

这个用于OpenGL的3d数学库可到网上自行下载,版本不一样可能变量名和变量初始化函数会不一样, 但使用方法都一样。也可以下载原作者的源码,里面也有这个头文件。 在这个教程中我们开始使用像向量这种辅助数据结构,并且慢慢我们会扩展这个头文件. (3)

GLenum res = glewInit(); if (res != GLEW_OK) {

fprintf(stderr, \ }

这里初始化GLEW并检查是否有错误,这个必须在GLUT初始化之后完成. (4)

Vector3f Vertices[1];

Vertices[0] = Vector3f(0.0f, 0.0f, 0.0f);

这里创建一个Vector3f结构的数组,并初始化XYZ坐标为0。这样使该点显示在屏幕中央。

(5)GLuint VBO;

这里在项目中定义一个全局的GLuint引用变量,来操作顶点缓冲器对象。后面会看到绝大多数OpenGL对象都是通过GLuint类型的变量来引用的.

(6)glGenBuffers(1, &VBO);

OpenGL定义了几个glGen*前缀的函数来产生不同类型的对象。它们通常有两个参数:第一个参数用来定义你想创建的对象的数量,第二个参数是一个GLuint变量的数组的地址,来存储分配给你的引用变量handles(要确保这个数组足够大来处理你的请求!)。以后对这个函数的调用将不会重复产生相同的handle对象,除非你先使用glDeleteBuffers删除他们。注意你不需要在buffer中定义你具体想要做的事情,将其一般化、通用化,具体的工作由下一步来完成。

(7)glBindBuffer(GL_ARRAY_BUFFER, VBO);

OpenGL使用handle的方式很独特,在很多API中handle可以提供给任何相关的函数并且具体的操作就是通过那个handle来操作完成,在OpenGL中我们需要将handle与一个目标的名称进行绑定,然后在该目标上执行命令。这些指令只会在与handle绑定的目标上生效直到另外有其他的对象跟这个handle绑定或者这个handle被置空。目标名GL_ARRAY_BUFFER意思是这个buffer将存储一个顶点的数组。另外一个有用的目标是GL_ELEMENT_ARRAY_BUFFER,这个的意思是这个buffer存储的是另一个buffer中顶点的标记。还有很多其他的目标,后面的教程中会看到。

(8)glBufferData(GL_ARRAY_BUFFER, sizeof(Vertices), Vertices, GL_STATIC_DRAW); 绑定了我们的对象之后,我们要往里面添加数据。这个回调函数取得我们之前绑定的目标名参数GL_ARRAY_BUFFER,还有数据的比特数参数,顶点数组的地址,还有一个表示这个数据模式的标志变量。因为我们不会去改变这个buffer的内容所以这里用了GL_STATIC_DRAW标志,相反的标志是GL_DYNAMIIC_DRAW, 这个只是给OpenGL的


一步步学OpenGL(一).doc 将本文的Word文档下载到电脑 下载失败或者文档不完整,请联系客服人员解决!

下一篇:张静中学中考历史复习资料5

相关阅读
本类排行
× 注册会员免费下载(下载后可以自由复制和排版)

马上注册会员

注:下载文档有可能“只有目录或者内容不全”等情况,请下载之前注意辨别,如果您已付费且无法下载或内容有问题,请联系我们协助你处理。
微信: QQ: