1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
// Copyright (C) 2021 Emilio J. Padrón
// Released as Free Software under the X11 License
// https://spdx.org/licenses/X11.html
//
// Strongly inspired by movingtri.cpp in OpenGL Superbible
// https://github.com/openglsuperbible
#include <GL/glew.h>
#include <GLFW/glfw3.h>
#include <stdio.h>
#include <math.h>
int gl_width = 640;
int gl_height = 480;
void glfw_window_size_callback(GLFWwindow* window, int width, int height);
void processInput(GLFWwindow *window);
void render(double);
GLuint shader_program = 0; // shader program to set render pipeline
GLuint vao = 0; // Vertext Array Object to set input data
int main() {
// start GL context and O/S window using the GLFW helper library
if (!glfwInit()) {
fprintf(stderr, "ERROR: could not start GLFW3\n");
return 1;
}
// glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
// glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
// glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
// glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
GLFWwindow* window = glfwCreateWindow(gl_width, gl_height, "My moving triangle", NULL, NULL);
if (!window) {
fprintf(stderr, "ERROR: could not open window with GLFW3\n");
glfwTerminate();
return 1;
}
glfwSetWindowSizeCallback(window, glfw_window_size_callback);
glfwMakeContextCurrent(window);
// start GLEW extension handler
// glewExperimental = GL_TRUE;
glewInit();
// get version info
const GLubyte* vendor = glGetString(GL_VENDOR); // get vendor string
const GLubyte* renderer = glGetString(GL_RENDERER); // get renderer string
const GLubyte* glversion = glGetString(GL_VERSION); // version as a string
const GLubyte* glslversion = glGetString(GL_SHADING_LANGUAGE_VERSION); // version as a string
printf("Vendor: %s\n", vendor);
printf("Renderer: %s\n", renderer);
printf("OpenGL version supported %s\n", glversion);
printf("GLSL version supported %s\n", glslversion);
printf("Starting viewport: (width: %d, height: %d)\n", gl_width, gl_height);
// Vertex Shader
const char* vertex_shader =
"#version 130\n"
"in vec3 v_pos;"
"in vec3 offset;"
"void main() {"
" gl_Position = vec4(v_pos, 1.0) + vec4(offset, 0.0);"
"}";
// Fragment Shader
const char* fragment_shader =
"#version 130\n"
"out vec4 frag_col;"
"void main() {"
" frag_col = vec4(0.54, 0.73, 0.1, 1.0);"
"}";
// Shaders compilation
GLuint vs = glCreateShader(GL_VERTEX_SHADER);
glShaderSource(vs, 1, &vertex_shader, NULL);
glCompileShader(vs);
GLuint fs = glCreateShader(GL_FRAGMENT_SHADER);
glShaderSource(fs, 1, &fragment_shader, NULL);
glCompileShader(fs);
// Create program, attach shaders to it and link it
shader_program = glCreateProgram();
glAttachShader(shader_program, fs);
glAttachShader(shader_program, vs);
glLinkProgram(shader_program);
// Release shader objects
glDeleteShader(vs);
glDeleteShader(fs);
// Vertex Array Object
glGenVertexArrays(1, &vao);
glBindVertexArray(vao);
// Triangle to be rendered (NDC)
float points[] = {
0.0f, 0.5f, 0.0f,
0.5f, -0.5f, 0.0f,
-0.5f, -0.5f, 0.0f
};
// Vertex Buffer Object (for vertex coordinates)
GLuint vbo = 0;
glGenBuffers(1, &vbo);
glBindBuffer(GL_ARRAY_BUFFER, vbo);
glBufferData(GL_ARRAY_BUFFER, sizeof(points), points, GL_STATIC_DRAW);
// Vertex attributes
// 0: vertex position (x, y, z)
// 1: offset (offx, offy, offz) -> in render loop (each iter. update)
glVertexAttribPointer(0, 3, GL_FLOAT, GL_FALSE, 0, NULL);
glEnableVertexAttribArray(0);
// Unbind vbo (it was conveniently registered by VertexAttribPointer)
glBindBuffer(GL_ARRAY_BUFFER, 0);
// Unbind vao
glBindVertexArray(0);
// Render loop
while(!glfwWindowShouldClose(window)) {
processInput(window);
render(glfwGetTime());
glfwSwapBuffers(window);
glfwPollEvents();
}
glfwTerminate();
return 0;
}
void render(double currentTime) {
glClear(GL_COLOR_BUFFER_BIT);
glViewport(0, 0, gl_width, gl_height);
glUseProgram(shader_program);
glBindVertexArray(vao);
GLfloat attrib_offset[] = { (float)sin(currentTime) * 0.5f,
(float)cos(currentTime) * 0.6f,
0.0f };
glVertexAttrib3fv(1, attrib_offset);
glDrawArrays(GL_TRIANGLES, 0, 3);
}
void processInput(GLFWwindow *window) {
if(glfwGetKey(window, GLFW_KEY_ESCAPE) == GLFW_PRESS)
glfwSetWindowShouldClose(window, 1);
}
// Callback function to track window size and update viewport
void glfw_window_size_callback(GLFWwindow* window, int width, int height) {
gl_width = width;
gl_height = height;
printf("New viewport: (width: %d, height: %d)\n", width, height);
}