OpenGL(三)创建窗口

前言:走到第三节了,发现GLFW库用不了,原来前面的内容没有把GLFW库的头文件放入Include文件夹下,导致找不到GLFW头文件,从而无法找到glfw3.h头文件,于是又把前面两节内容加以完善,总算是补上了,现在继续第三节的内容,冲鸭!!!


在main函数中实例化GLFW窗口

#include <glad/glad.h>
#include <GLFW/glfw3.h>

int main() {
    //初始化GLFW
    glfwInit();

    //配置GLFW====================
    //glfwWindowHint(参数1,参数2)
    //参数1:选项的名称。可以从很多以GLFW_开头的枚举值中选择
    //参数2:接受一个整型,用来设置这个选项的值
    //设置版本号为3.3版本
    glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR,3);//设置主板本号为3
    glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR,3);//设置副版本号为3
    glfwWindowHint(GLFW_OPENGL_PROFILE,GLFW_OPENGL_CORE_PROFILE);//模式:核心模式(Core-profile)

    //如果你使用的是Mac OS X系统,你还需要解除下面这行代码的注释
    //glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
    return 0;
}
  • 该函数的所有的选项以及对应的值都可以在 GLFW’s window handling 这篇文档中找到。如果你现在编译你的cpp文件会得到大量的 undefined reference (未定义的引用)错误,也就是说你并未顺利地链接GLFW库。
    注意:请确认您的系统支持OpenGL3.3或更高版本,否则此应用有可能会崩溃或者出现不可预知的错误。如果想要查看OpenGL版本的话,在Linux上运行glxinfo,或者在Windows上使用其它的工具(例如OpenGL Extension Viewer)。如果你的OpenGL版本低于3.3,检查一下显卡是否支持OpenGL 3.3+(不支持的话你的显卡真的太老了),并更新你的驱动程序,有必要的话请更新显卡。

创建一个窗口对象

这个窗口对象存放了所有和窗口相关的数据,而且会被GLFW的其他函数频繁地用到。

#include <iostream>
    //创建窗口对象=================
    //glfwCreateWindow(参数1,参数2,参数3,参数4,参数5)
    //参数1:窗口宽
    //参数2:窗口高
    //参数3:窗口标题(名字)
    //参数4,参数5暂时忽略
    //返回值类型:GLFWwindow*
    GLFWwindow* window = glfwCreateWindow(800,600,"LearnOpenGL",NULL,NULL);
    //创建窗口对象失败
    if (window == NULL)
    {
        //打印错误信息
        std::cout << "Failed to create GLFW window" << std::endl;//需要引用iostream头文件
        //终止GLFW
        glfwTerminate();
        return -1;
    }
    //将窗口的上下文设置为当前线程的主上下文
    glfwMakeContextCurrent(window);

GLAD

GLAD是用来管理OpenGL的函数指针的,所以在调用任何OpenGL的函数之前我们需要初始化GLAD。

    //GLAD========================
    //加载系统相关的OpenGL函数指针地址的函数,它根据我们编译的系统定义了正确的函数。
    //gladLoadGLLoader(参数1)
    //参数1:GLADloadproc类型的函数指针地址。这里我们通过GLFW中的glfwGetProcAddress属性获取了函数指针的的地址
    if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
    {
        //加载函数地址失败
        //打印错误信息
        std::cout << "Failed to initialize GLAD" << std::endl;
        return -1;
    }

设置OpenGL渲染窗口(视口)的尺寸大小

    //设置视口(渲染窗口)的尺寸大小============
    //glViewport(参数1,参数2,参数3,参数4)
    //参数1,参数2:渲染窗口左下角的位置
    //参数3,参数4:渲染窗口的宽度和高度(单位:像素)
    glViewport(0, 0, 800, 600);
  • 我们实际上也可以将视口的维度设置为比GLFW的维度小,这样子之后所有的OpenGL渲染将会在一个更小的窗口中显示,这样子的话我们也可以将一些其它元素显示在OpenGL视口之外。
    注意:OpenGL幕后使用glViewport中定义的位置和宽高进行2D坐标的转换,将OpenGL中的位置坐标转换为你的屏幕坐标。例如,OpenGL中的坐标(-0.5, 0.5)有可能(最终)被映射为屏幕中的坐标(200,450)。注意,处理过的OpenGL坐标范围只为-1到1,因此我们事实上将(-1到1)范围内的坐标映射到(0, 800)和(0, 600)。

  • 当用户改变窗口的大小的时候,视口也应该被调整。我们可以对窗口注册一个回调函数(Callback Function),它会在每次窗口大小被调整的时候被调用。这个回调函数的原型如下:

  void framebuffer_size_callback(GLFWwindow* window, int width, int height);
  • 这个帧缓冲大小函数需要一个GLFWwindow作为它的第一个参数,以及两个整数表示窗口的新维度。每当窗口改变大小,GLFW会调用这个函数并填充相应的参数供你处理。
void framebuffer_size_callback(GLFWwindow* window, int width, int height)
{
    glViewport(0, 0, width, height);
}
  • 我们还需要注册这个函数,告诉GLFW我们希望每当窗口调整大小的时候调用这个函数:
glfwSetFramebufferSizeCallback(window, framebuffer_size_callback);
  • 当窗口被第一次显示的时候framebuffer_size_callback也会被调用。对于视网膜(Retina)显示屏,width和height都会明显比原输入值更高一点。
  • 我们还可以将我们的函数注册到其它很多的回调函数中。比如说,我们可以创建一个回调函数来处理手柄输入变化,处理错误消息等。
  • 我们会在创建窗口之后,渲染循环初始化之前注册这些回调函数。

渲染循环

为了让应用程序不只绘制一个图像之后就立即退出并关闭窗口并且能够在主动关闭它之前不断绘制图像并能够接受用户输入。因此,我们需要在程序中添加一个while循环,我们可以把它称之为渲染循环(Render Loop),它能在我们让GLFW退出前一直保持运行。下面几行的代码就实现了一个简单的渲染循环:

    //渲染循环================================
    //glfwWindowShouldClose(参数1)
    //此函数每次循环的开始前检查一次GLFW是否被要求退出,
    //如果是的话该函数返回true,渲染循环结束,之后关闭应用程序。
    //参数1:检测对象
    while (!glfwWindowShouldClose(window))
    {
        //交换颜色缓冲(它是一个储存着GLFW窗口每一个像素颜色值的大缓冲),
        //它在这一迭代中被用来绘制,并且将会作为输出显示在屏幕上。
        glfwSwapBuffers(window);
        //检查有没有触发什么事件(比如键盘输入、鼠标移动等)、更新窗口状态,
        //并调用对应的回调函数(可以通过回调方法手动设置)。
        glfwPollEvents();
    }
  • 双缓冲(Double Buffer)
    为什么使用双缓冲?
    应用程序使用单缓冲绘图时可能会存在图像闪烁的问题。 这是因为生成的图像不是一下子被绘制出来的,而是按照从左到右,由上而下逐像素地绘制而成的。最终图像不是在瞬间显示给用户,而是通过一步一步生成的,这会导致渲染的结果很不真实。为了规避这些问题,我们应用双缓冲渲染窗口应用程序。
    什么是双缓冲?
    前缓冲保存着最终输出的图像,它会在屏幕上显示;而所有的的渲染指令都会在后缓冲上绘制。当所有的渲染指令执行完毕后,我们交换(Swap)前缓冲和后缓冲,这样图像就立即呈显出来,之前提到的不真实感就消除了。

正确释放/删除之前的分配的所有资源

当渲染循环结束后我们需要正确释放/删除之前的分配的所有资源。我们可以在main函数的最后调用glfwTerminate函数来完成。

    //终止GLFW,释放所有资源====================
    glfwTerminate();

测试

  • 点击本地Windows调试器


  • 可以看到如下效果:


  • 完整代码:

#include <glad/glad.h>
#include <GLFW/glfw3.h>
#include <iostream>

int main() {
    //初始化GLFW
    glfwInit();

    //配置GLFW====================
    //glfwWindowHint(参数1,参数2)
    //参数1:选项的名称。可以从很多以GLFW_开头的枚举值中选择
    //参数2:接受一个整型,用来设置这个选项的值
    //设置版本号为3.3版本
    glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR,3);//设置主板本号为3
    glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR,3);//设置副版本号为3
    glfwWindowHint(GLFW_OPENGL_PROFILE,GLFW_OPENGL_CORE_PROFILE);//模式:核心模式(Core-profile)

    //如果你使用的是Mac OS X系统,你还需要解除下面这行代码的注释
    //glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);

    //创建窗口对象=================
    //glfwCreateWindow(参数1,参数2,参数3,参数4,参数5)
    //参数1:窗口宽
    //参数2:窗口高
    //参数3:窗口标题(名字)
    //参数4,参数5暂时忽略
    //返回值类型:GLFWwindow*
    GLFWwindow* window = glfwCreateWindow(800,600,"LearnOpenGL",NULL,NULL);
    //创建窗口对象失败
    if (window == NULL)
    {
        //打印错误信息
        std::cout << "Failed to create GLFW window" << std::endl;//需要引用iostream头文件
        //终止GLFW
        glfwTerminate();
        return -1;
    }
    //将窗口的上下文设置为当前线程的主上下文
    glfwMakeContextCurrent(window);

    //GLAD========================
    //加载系统相关的OpenGL函数指针地址的函数,它根据我们编译的系统定义了正确的函数。
    //gladLoadGLLoader(参数1)
    //参数1:GLADloadproc类型的函数指针地址。这里我们通过GLFW中的glfwGetProcAddress属性获取了函数指针的的地址
    if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress))
    {
        //加载函数地址失败
        //打印错误信息
        std::cout << "Failed to initialize GLAD" << std::endl;
        return -1;
    }

    //设置视口(渲染窗口)的尺寸大小============
    //glViewport(参数1,参数2,参数3,参数4)
    //参数1,参数2:渲染窗口左下角的位置
    //参数3,参数4:渲染窗口的宽度和高度(单位:像素)
    glViewport(0, 0, 800, 600);

    //渲染循环================================
    //glfwWindowShouldClose(参数1)
    //此函数每次循环的开始前检查一次GLFW是否被要求退出,
    //如果是的话该函数返回true,渲染循环结束,之后关闭应用程序。
    //参数1:检测对象
    while (!glfwWindowShouldClose(window))
    {
        //交换颜色缓冲(它是一个储存着GLFW窗口每一个像素颜色值的大缓冲),
        //它在这一迭代中被用来绘制,并且将会作为输出显示在屏幕上。
        glfwSwapBuffers(window);
        //检查有没有触发什么事件(比如键盘输入、鼠标移动等)、更新窗口状态,
        //并调用对应的回调函数(可以通过回调方法手动设置)。
        glfwPollEvents();
    }

    //终止GLFW,释放所有资源====================
    glfwTerminate();

    return 0;
}
  • 或者你可以参考LearnOpenGL官方网站的源码:传送门
  • 如果程序编译有问题,请先检查连接器选项是否正确,IDE中是否导入了正确的目录(前面教程解释过)。并且请确认你的代码是否正确,直接对照上面提供的源代码就行了。如果还是有问题,欢迎评论,我或者其他人可能会帮助你的。

总结:此节内容,我们学习了如何创建一个窗口,其中包括了初始化GLFW、配置GLFW、创建窗口对象、GLAD加载函数指针地址、设置视口大小、渲染循环、释放分配的内存。呼~ 好累=-=,下次见~

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容