OpenGL学习(2)——绘制三角形
在创建窗口的基础上,添加代码实现三角形的绘制。
声明和定义变量
在屏幕上绘制一个三角形需要的变量有:
- 三角形的三个顶点坐标;
- Vertex Buffer Object 将顶点数据存储在GPU的内存中;
- Vertex Array Object存储对顶点属性的配置和与顶点属性相关的VBO。在需要绘制的对象数量和顶点属性很多的情况下,VAO的使用能够大大减小工作量;
- Vertex Shader将顶点作为输入,对顶点坐标进行变换并输出。在编写Vertex Shader源码时,要将顶点的非齐次坐标变换成齐次坐标,只需要添加w分量即可;
- Fragment Shader计算三角形对应像素点的颜色,为了方便,将像素点颜色全部设置成(1, 0.5, 0.2);
- Shader Program由多个Shader链接后得到。
float vertices[] = {-0.5f, -0.5f, 0.0f,
0.5f, -0.5f, 0.0f,
0.0f, 0.5f, 0.0f};
unsigned int VBO;
unsigned int VAO;
int vertexShader;
const char *vertexShaderSource = "#version 330 core\n"
"layout (location = 0) in vec3 Pos;"
"void main()"
"{gl_Position = vec4(Pos.x, Pos.y, Pos.z, 1.0f);}";
int fragmentShader;
const char *fragmentShaderSource = "#version 330 core\n"
"out vec4 fragColor;"
"void main()"
"{fragColor = vec4(1.0f, 0.5f, 0.2f, 1.0f);}";
int shaderProgram;
绘制一个对象的过程略繁琐,可以拆分成几个部分:
- 创建Vertex Shader并编译源码;
- 创建Fragment Shader并编译源码;
- 链接Vertex Shader和Fragment Shader得到Shader Program;
- 创建VAO
- 创建VBO
- 绑定和配置顶点属性指针
创建Vertex Shader并编译源码
调用glCreateShader函数创建一个Shader对象,传递参数GL_VERTEX_SHADER使该对象为Vertex Shader对象。
调用glShaderSource函数将Vertex Shader源码附加到Shader对象上。
调用glCompileShader函数编译源码。
vertexShader = glCreateShader(GL_VERTEX_SHADER);
glShaderSource(vertexShader, 1, &vertexShaderSource, NULL);
glCompileShader(vertexShader);
创建Fragment Shader并编译源码
和上一步的区别在于传递参数不同。
fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
glShaderSource(fragmentShader, 1, &fragmentShaderSource, NULL);
glCompileShader(fragmentShader);
链接Shader
调用glCreateProgram函数创建Shader Program对象。
调用两次glAttachShader函数将Shader附加到Shader Program对象上。
调用glLinkProgr函数进行链接。
shaderProgram = glCreateProgram();
glAttachShader(shaderProgram, vertexShader);
glAttachShader(shaderProgram, fragmentShader);
glLinkProgram(shaderProgram);
链接成功后调用glDeleteShader函数移除Shader对象,释放被占用的资源。
glDeleteShader(vertexShader);
glDeleteShader(fragmentShader);
创建VAO
glGenVertexArrays函数用来创建VAO,并生成对象ID。第一个参数指定需要创建的VAO数量。
glGenVertexArrays(1, &VAO);
创建VBO
和创建VAO类似,调用glGenBuffers函数创建VBO。
glGenBuffers(1, &VBO);
绑定和配置顶点属性指针
绑定VAO
调用glBindVertexArray函数绑定VAO,接下来对顶点属性的使能、顶点属性指针的配置和相应的VBO,都将存储在这个VAO中。
需要解绑当前VAO时,将参数设置为0即可。
glBindVertexArray(VAO);
配置VBO
VBO的Buffer类型是GL_ARRAY_BUFFER,通过调用glBindBuffer函数绑定VBO和GL_ARRAY_BUFFER。
之后,当再次调用GL_ARRAY_BUFFER,便可完成对VBO的配置。glBufferData函数能够把顶点数据复制到Buffer内存中供GPU使用。
glBindBuffer(GL_ARRAY_BUFFER, VBO);
glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);
配置顶点属性指针
当顶点数据被复制到Buffer内存中之后,还需要配置对顶点数据的解析。就是告诉GPU,Vertex Shader的哪一个顶点属性,对应着Buffer中的哪一部分数据。
在编写Vertex Shader源码时,只定义了一个顶点属性:位置,该属性的索引为0,因此设置glVertexAttribPointer的第一个参数为0。
每一个顶点数据有x,y,z三个维度,用来表示该顶点的位置,因此glVertexAttribPointer的第二个参数为3。
第三个参数指定顶点数据的类型。第四个参数设置是否需要将顶点数据标准化,即映射到[-1, 1]。
第五个参数指定同一顶点属性的相邻数据之间的步长,这里为3个float类型的长度。
第六个参数指定某一顶点属性下第一个数据的起始位置。
最后调用glEnableVertexAttribArray函数使能该顶点属性。
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3*sizeof(float), (void*)0);
glEnableVertexAttribArray(0);
配置完成后,解绑VAO和VBO
glBindBuffer(GL_ARRAY_BUFFER, 0);
glBindVertexArray(0);
渲染循环
在while循环中添加三个函数,激活Shader Program,绑定VAO并在视口中绘制三角形。
glUseProgram(shaderProgram);
glBindVertexArray(VAO);
glDrawArrays(GL_TRIANGLES, 0, 3);
╮(╯▽╰)╭画个三角形真不容易。。。
完整代码如下:
#include <iostream>
#include <glad/glad.h>
#include <GLFW/glfw3.h>
using namespace std;
/*
void frambuffer_size_callback(GLFWwindow *window, int width, int height)
{
glViewport(0, 0, width, height);
}
*/
void processInput(GLFWwindow* window)
{
//check if ESCAPE is pressed
if(glfwGetKey(window, GLFW_KEY_ESCAPE) == GLFW_PRESS)
glfwSetWindowShouldClose(window, true);
}
const unsigned int window_width = 800;
const unsigned int window_height = 600;
//coordnate (x,y,z) of vertices
float vertices[] = {-0.5f, -0.5f, 0.0f,
0.5f, -0.5f, 0.0f,
0.0f, 0.5f, 0.0f};
//vertex buffer object(VBO)
unsigned int VBO;
//vertex array object(VAO)
unsigned int VAO;
//vertext shader
int vertexShader;
const char *vertexShaderSource = "#version 330 core\n"
"layout (location = 0) in vec3 Pos;"
"void main()"
"{gl_Position = vec4(Pos.x, Pos.y, Pos.z, 1.0f);}";
//fragment shader
int fragmentShader;
const char *fragmentShaderSource = "#version 330 core\n"
"out vec4 fragColor;"
"void main()"
"{fragColor = vec4(1.0f, 0.5f, 0.2f, 1.0f);}";
//shader program
int shaderProgram;
int main()
{
//initialize GLFW
if(!glfwInit())
return -1;
//configure GLFW
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
glfwWindowHint(GLFW_OPENGL_PROFILE,GLFW_OPENGL_CORE_PROFILE);
//creat a window object
GLFWwindow *window = glfwCreateWindow(window_width, window_height, "OpenGL_Demo", NULL, NULL);
if (window == NULL){
cout << "Failed to create GLFW window" << endl;
glfwTerminate();
return -1;
}
glfwMakeContextCurrent(window);
//initialize GLAD to manage function pointers for OpenGL
if (!gladLoadGLLoader((GLADloadproc)glfwGetProcAddress)){
cout << "Failed to initialize GLAD" << endl;
return -1;
}
//set width and height of Viewport
glViewport(0, 0, window_width, window_height);
//glfwSetFramebufferSizeCallback(window, frambuffer_size_callback);
//compile vertex shader
vertexShader = glCreateShader(GL_VERTEX_SHADER);
glShaderSource(vertexShader, 1, &vertexShaderSource, NULL);
glCompileShader(vertexShader);
//check if compilation of vertex shader is successful
int success;
char infoLog[512];
glGetShaderiv(vertexShader, GL_COMPILE_STATUS, &success);
if (!success){
glGetShaderInfoLog(vertexShader, 512, NULL, infoLog);
cout << "ERROR::VERTEXSHADER::COMPILATION_FAILED\n" << infoLog << endl;
}
else cout << "VERTEXSHADER_COMPILATION_SUCCESS" << endl;
//cmpile fragment shader
fragmentShader = glCreateShader(GL_FRAGMENT_SHADER);
glShaderSource(fragmentShader, 1, &fragmentShaderSource, NULL);
glCompileShader(fragmentShader);
//check if compilation of fragment shader is successful
glGetShaderiv(fragmentShader, GL_COMPILE_STATUS, &success);
if (!success){
glGetShaderInfoLog(fragmentShader, 512, NULL, infoLog);
cout << "ERROR::FRAGMENTSHADER::COMPILATION_FAILED\n" << infoLog << endl;
}
else cout << "FRAGMENTSHADER_COMPILATION_SUCCESS" << endl;
//link shader program
shaderProgram = glCreateProgram();
glAttachShader(shaderProgram, vertexShader);
glAttachShader(shaderProgram, fragmentShader);
glLinkProgram(shaderProgram);
//check if linking is successful
glGetProgramiv(shaderProgram, GL_LINK_STATUS, &success);
if(!success){
glGetProgramInfoLog(shaderProgram, 512, NULL, infoLog);
cout << "ERROR::LINKING_FAILED\n" << infoLog << endl;
}
else cout << "LINKING_SUCCESS" << endl;
//clear resource of shader objects
glDeleteShader(vertexShader);
glDeleteShader(fragmentShader);
//generate vertex array objext
glGenVertexArrays(1, &VAO);
//generate vertex buffer object
glGenBuffers(1, &VBO);
glBindVertexArray(VAO);
glBindBuffer(GL_ARRAY_BUFFER, VBO);
glBufferData(GL_ARRAY_BUFFER, sizeof(vertices), vertices, GL_STATIC_DRAW);
//link vertex attributes
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 3*sizeof(float), (void*)0);
glEnableVertexAttribArray(0);
glBindBuffer(GL_ARRAY_BUFFER, 0);
glBindVertexArray(0);
//render loop
while(!glfwWindowShouldClose(window)){
processInput(window);
glClearColor(0.2f, 0.3f, 0.3f, 1.0f);
glClear(GL_COLOR_BUFFER_BIT);
glUseProgram(shaderProgram);
glBindVertexArray(VAO);
glDrawArrays(GL_TRIANGLES, 0, 3);
//glBindVertexArray(0);
glfwSwapBuffers(window);
glfwPollEvents();
}
//clear resource
glfwTerminate();
return 0;
}
OpenGL学习(2)——绘制三角形的更多相关文章
- Android OpenGL 入门示例----绘制三角形和正方形
Android上对OpenGl的支持是无缝的,所以才有众多3D效果如此逼真的游戏,在Camera的一些流程中也有用到GLSurfaceView的情况.本文记录OpenGL在Android上的入门级示例 ...
- Linux OpenGL 实践篇-3 绘制三角形
本次实践是绘制两个三角形,重点理解顶点数组对象和OpenGL缓存的使用. 顶点数组对象 顶点数组对象负责管理一组顶点属性,顶点属性包括位置.法线.纹理坐标等. OpenGL缓存 OpenGL缓存实质上 ...
- iOS OpenGL ES简单绘制三角形
OpenGL 是用于2D/3D图形编程的一套基于C语言的统一接口. windows,Linux,Unix上均可兼容. OpenGL ES 是在OpenGL嵌入式设备上的版本, android/iOS ...
- OpenGL学习(2)——绘制三角形(补)
对上一篇的补充,通过绘制三角形来完成矩形的绘制.此外,完成章节后练习. 绘制矩形 一个矩形由两个三角形组成,因此绘制矩形需要绘制两个三角形,一共6个顶点,其中2个顶点重复画了两次. 为了减小开销,仅储 ...
- OpenGL学习进程(11)第八课:颜色绘制的详解
本节是OpenGL学习的第八个课时,下面将详细介绍OpenGL的颜色模式,颜色混合以及抗锯齿. (1)颜色模式: OpenGL支持两种颜色模式:一种是RGBA,一种是颜色索引模式. R ...
- OpenGL学习进程(10)第七课:四边形绘制与动画基础
本节是OpenGL学习的第七个课时,下面以四边形为例介绍绘制OpenGL动画的相关知识: (1)绘制几种不同的四边形: 1)四边形(GL_QUADS) OpenGL的GL_QUADS图 ...
- OpenGL学习进程(4)第二课:绘制图形
本节是OpenGL学习的第二个课时,下面介绍如何用点和线来绘制图形: (1)用点的坐标来绘制矩形: #include <GL/glut.h> void display(void) ...
- 1.opengl绘制三角形
顶点数组对象:Vertex Array Object,VAO,用于存储顶点状态配置信息,每当界面刷新时,则通过VAO进行绘制. 顶点缓冲对象:Vertex Buffer Object,VBO,通过VB ...
- Android OpenGL ES(十)绘制三角形Triangle .
三角形为OpenGL ES支持的面,同样创建一个DrawTriangle Activity,定义6个顶点使用三种不同模式来绘制三角形: float vertexArray[] = { -0.8f, - ...
随机推荐
- 通过C/C++基于http下载文件
简介 Windows系统如何通过C/C++下载互联网上的文件呢?这里笔者给大家演示一个最简单的方法,利用Windows提供的urlmon库,可以快速实现文件下载的简单实例. 注:本文内容部分参考了&l ...
- 第 14 章 结构和其他数据形式(伸缩型数组成员C99)
伸缩型数组成员C99 声明一个伸缩型数组成员的规则: 1.伸缩型数组成员必须是结构的最后一个成员: 2.结构中必须至少有一个成员: 3.伸缩数组的方括号是空的. 示例 struct flex { in ...
- 如何使用EditPlus批量删除 带有某个字符的一行
比如以下五行,我要将带有英文字母a的一行全部批量删除1234551243243123aa244123123981232137aa 2013-04-11 19:32 提问者采纳 我这里是英文版, ...
- c++ 远程连接局域网内 数据库,并进行操作
首先尝试利用Windows自带的dos命令窗口操作数据库:参考见https://jingyan.baidu.com/article/3aed632e19b5e8701080918f.html 1.搜索 ...
- BZOJ2079:[POI2010]Guilds(乱搞)
Description Zy皇帝面临一个严峻的问题,两个互相抵触的贸易团体,YYD工会和FSR工会,他们在同一时间请求在王国各个城市开办自己的办事处.这里有n个城市,其中有一些以双向马路相连,这两个工 ...
- Winfrom 使用WCF 实现双工通讯
实现双工通讯主要分三步. 通信接口的定义: 被调用接口的实现 双工通道的建立 请先引用DLL(CSDN的代码编辑器真尼玛蛋疼) 整个解决方案的结构 1.通信接口的定义: 服务端调用客户端接口IServ ...
- 为什么web3 1.0 的接口有personal_*和eth_*的,两者有什么不同
看https://github.com/ethereum/EIPs/pull/712 Why personal_* namespace instead of eth_* namespace? I be ...
- OpenCV——staturate_cast、掩模操作
saturate_cast<>()模板函数,用于溢出保护 //大致的原理如下 ) data=; elseif(data>) data=; 掩模操作:https://blog.csdn ...
- JS时间轴效果(类似于qq空间时间轴效果)
在上一家公司写了一个时间轴效果,今天整理了下,感觉有必要写一篇博客出来 给大家分享分享 当然代码还有很多不足的地方,希望大家多指点指点下,此效果类似于QQ空间或者人人网空间时间轴效果,当时也是为了需求 ...
- Python2.7-tarfile
tarfile模块,读写 tar 压缩文件,包括用 gzip 或是 bz2 压缩的文件(如tar.bz2.tar.gz),一般使用 TarFile 类完成操作 1.模块方法 tarfile.is_ta ...