StapleGL
Header-only C++20 OpenGL wrapper
Loading...
Searching...
No Matches
teapot.cpp
Go to the documentation of this file.
1
10#define STAPLEGL_DEBUG
11#include "glad.h"
12#include "staplegl.hpp"
13#include <GLFW/glfw3.h>
14
15#include <algorithm>
16#include <array>
17#include <cstddef>
18#include <iostream>
19#include <span>
20#include <utility>
21
22// quick and dirty models as C headers.
23#include "box.h"
24#include "screen_quad.h"
25#include "teapot_data.h"
26
27// glm linear algebra library, NOT needed for staplegl, just for the example.
28#include "glm.hpp"
29#include "gtc/matrix_transform.hpp"
30#include "gtc/type_ptr.hpp"
31
32// stb_image in order to load up textures.
33#define STB_IMAGE_IMPLEMENTATION
34#include "stb_image.h"
35
36/*
37
38 IMPORTANT: in order for the program to correctly compile, go to
39 include/staplegl/modules/gl_functions.hpp and replace the placeholder
40 with the provided OpenGL loader ("glad.h").
41
42*/
43
44#define STAPLEGL_TEAPOT_LOW_MSAA
45
46#ifdef STAPLEGL_TEAPOT_LOW_MSAA
48constexpr auto MSAA_SAMPLES = 2;
49#else
51constexpr auto MSAA_SAMPLES = 8;
52#endif
53
54// glfw callbacks
55void framebuffer_size_callback(GLFWwindow* window, int width, int height);
56void processInput(GLFWwindow* window);
57
58// initial window size
59const int32_t SCR_WIDTH = 1600;
60const int32_t SCR_HEIGHT = 900;
61
62// global luminosity to be updated by keypresses
63float luminosity = 50.0F; // NOLINT
64
65// OpenGL debug callback
66void GLAPIENTRY
67MessageCallback(GLenum source [[maybe_unused]],
68 GLenum type,
69 GLuint id [[maybe_unused]],
70 GLenum severity,
71 GLsizei length [[maybe_unused]],
72 const GLchar* message,
73 const void* userParam [[maybe_unused]])
74{
75 // if (type == GL_DEBUG_TYPE_PERFORMANCE || type == GL_DEBUG_TYPE_OTHER) {
76 // return;
77 // }
78
79 // skip non-errors
80 if (severity == GL_DEBUG_SEVERITY_NOTIFICATION || type == GL_DEBUG_TYPE_PERFORMANCE || type == GL_DEBUG_TYPE_OTHER) {
81 return;
82 }
83
84 fprintf(stderr, "GL CALLBACK: %s type = 0x%x, severity = 0x%x,\nmessage = %s\n", // NOLINT
85 (type == GL_DEBUG_TYPE_ERROR ? "** GL ERROR **" : ""),
86 type, severity, message);
87
88 // print location of error
89 fprintf(stderr, "source = 0x%x, id = %d\n", source, id); // NOLINT
90}
91
92constexpr auto calc_pyramid_levels(uint32_t width, uint32_t height) -> uint32_t
93{
94 uint32_t levels = 1;
95 while (width > 2 && height > 2) {
96 width /= 2;
97 height /= 2;
98 levels++;
99 }
100 return levels;
101}
102
103// global aspect ratio, used for the projection matrix, please don't do this in
104// a real program.
105float aspect_ratio = static_cast<double>(SCR_WIDTH) / static_cast<double>(SCR_HEIGHT); // NOLINT
106
107auto main() -> int
108{
109
110 std::string_view const hello_message {
111 "Hello! This is a more complex example of staplegl usage, featuring the Utah Teapot model.\n"
112 "Press the U and D keys to increase (U) and decrease (D) the luminosity of the light source.\n"
113 "Play around with them to observe how the bloom effect changes."
114 };
115
116 std::cout << hello_message << std::endl;
117
118 // shorten layout declarations a bit.
119 using namespace staplegl::shader_data_type;
120
121 // glfw: initialize and configure
122 // ------------------------------
123 glfwInit();
124 glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 4);
125 glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 6);
126
127 glfwWindowHint(GLFW_SAMPLES, MSAA_SAMPLES); // MSAA
128 glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
129
130#ifdef __APPLE__
131 glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
132#endif
133
134 // glfw window creation
135 // --------------------
136 GLFWwindow* window = glfwCreateWindow(SCR_WIDTH, SCR_HEIGHT, "Utah Teapot", nullptr, nullptr);
137 if (window == nullptr) {
138 std::cout << "Failed to create GLFW window" << std::endl;
139 glfwTerminate();
140 return -1;
141 }
142 glfwMakeContextCurrent(window);
143 glfwSetFramebufferSizeCallback(window, framebuffer_size_callback);
144
145 // glad: load all OpenGL function pointers
146 // ---------------------------------------
147 if (gladLoadGLLoader(reinterpret_cast<GLADloadproc>(glfwGetProcAddress)) == 0) { // NOLINT
148 std::cout << "Failed to initialize GLAD" << std::endl;
149 return -1;
150 }
151
152 // enabe OpenGL features: depth test, MSAA, debug output, face culling
153 glEnable(GL_DEPTH_TEST);
154 glEnable(GL_DEBUG_OUTPUT);
155 glEnable(GL_CULL_FACE);
156 glEnable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
157
158#ifdef STAPLEGL_DEBUG
159 glDebugMessageCallback(MessageCallback, nullptr);
160 glDebugMessageControl(GL_DONT_CARE, GL_DONT_CARE, GL_DONT_CARE, 0, nullptr, GL_TRUE);
161 std::clog << "Printing OpenGL version info:" << std::endl;
162 std::clog << "OpenGL version: " << glGetString(GL_VERSION) << std::endl;
163 std::clog << "OpenGL vendor: " << glGetString(GL_VENDOR) << std::endl;
164 std::clog << "OpenGL renderer: " << glGetString(GL_RENDERER) << std::endl;
165#endif
166
167 // Set up all the shaders
168 staplegl::shader_program teapot_shader { "teapot_shader", "./shaders/teapot_shader.glsl" };
169 staplegl::shader_program skybox_shader { "skybox_shader", "./shaders/skybox_shader.glsl" };
170 staplegl::shader_program light_shader { "light_shader", "./shaders/light_shader.glsl" };
171 staplegl::shader_program tonemap_shader { "tone_mapping", "./shaders/tone_mapping.glsl" };
172 staplegl::shader_program downsample_shader { "downsample", "./shaders/downsample_shader.glsl" };
173 staplegl::shader_program upsample_shader { "upsample", "./shaders/upsample_shader.glsl" };
174 staplegl::shader_program passthrough_shader { "passthrough", "./shaders/passthrough_shader.glsl" };
175
176 skybox_shader.bind();
177 skybox_shader.upload_uniform1i("skybox", 0);
178
179 teapot_shader.bind();
180 teapot_shader.upload_uniform1i("environment", 0);
181
182 passthrough_shader.bind();
183 passthrough_shader.upload_uniform1i("scene", 1);
184
185 tonemap_shader.bind();
186 tonemap_shader.upload_uniform1i("scene", 1);
187 tonemap_shader.upload_uniform1i("bloom", 2);
188
189 downsample_shader.bind();
190 downsample_shader.upload_uniform1i("scene", 1);
191
192 upsample_shader.bind();
193 upsample_shader.upload_uniform1i("scene", 1);
194
195 // set up framebuffers and textures for HDR and bloom effect
196 std::array<staplegl::texture_2d, calc_pyramid_levels(SCR_WIDTH, SCR_HEIGHT)> pyramid_textures {};
197
198 staplegl::texture_2d const msaa_color {
199 std::span<const float> {},
202 .internal_format = GL_RGBA16F, .format = GL_RGBA, .datatype = GL_FLOAT },
204 .min_filter = GL_LINEAR, .mag_filter = GL_LINEAR, .clamping = GL_CLAMP_TO_EDGE },
205 MSAA
206 };
207
208 staplegl::texture_2d hdr_color {
209 std::span<const float> {},
212 .internal_format = GL_RGBA16F, .format = GL_RGBA, .datatype = GL_FLOAT },
214 .min_filter = GL_LINEAR, .mag_filter = GL_LINEAR, .clamping = GL_CLAMP_TO_EDGE }
215 };
216
217 // construct a texture pyramid, with each texture being half the size of the previous one.
218 // iteratively downsampling from screen resolution towards the bottom will allow us to
219 // spread the bloom effect over a wide area, with more details.
220 for (int i = 0; auto& tex : pyramid_textures) {
222 std::span<const float> {},
225 .internal_format = GL_RGBA16F, .format = GL_RGBA, .datatype = GL_FLOAT },
227 .min_filter = GL_LINEAR, .mag_filter = GL_LINEAR, .clamping = GL_CLAMP_TO_EDGE }
228 };
229 i++;
230 }
231
232 staplegl::framebuffer msaa_fbo {};
233 staplegl::framebuffer post_fbo {};
234
235 // set up vertex data (and buffer(s)) and configure vertex attributes
236 // ------------------------------------------------------------------
237
238 // Teapot model
239 staplegl::vertex_buffer_layout const layout_3P_3N { { u_type::vec3, "aPos" }, { u_type::vec3, "aNormal" } };
240
241 staplegl::vertex_buffer VBO { { teapot_vertices }, // std::span extracts the size from the C array, preventing decay.
243 VBO.set_layout(layout_3P_3N);
244
246 { teapot_indices } // capture the array again.
247 };
248
250
251 VAO.add_vertex_buffer(std::move(VBO));
252 VAO.set_index_buffer(std::move(EBO));
253 VAO.unbind(); // unbind to ensure that the VAO is not modified by accident.
254
255 // Cube model used for the skybox
256
257 staplegl::vertex_buffer_layout const layout_3P { { u_type::vec3, "aPos" } };
258
259 staplegl::vertex_buffer skybox_VBO { { skybox_vertices }, // std::span extracts the size from the C array, preventing decay.
261 skybox_VBO.set_layout(layout_3P); // we reuse the same layout as the teapot.
262
263 // the skybox has no indices per se, we generate them on the fly.
264 std::vector<unsigned int> skybox_indices(SKYBOX_VERTS / 3);
265 std::generate(skybox_indices.begin(), skybox_indices.end(),
266 [n = 0]() mutable { return n++; });
267 staplegl::index_buffer skybox_EBO { skybox_indices };
268
269 staplegl::vertex_array skybox_VAO;
270
271 skybox_VAO.add_vertex_buffer(std::move(skybox_VBO));
272 skybox_VAO.set_index_buffer(std::move(skybox_EBO));
273 skybox_VAO.unbind(); // another unbind for good measure.
274
275 // declare a simple model matrix, to be modified in the render loop.
276 glm::mat4 const model = glm::mat4(1.0F);
277
278 // screen quad for post-processing
279
280 staplegl::vertex_buffer_layout const layout_3P_2UV { { u_type::vec3, "aPos" }, { u_type::vec2, "aTexCoord" } };
281
282 staplegl::vertex_buffer quad_VBO { { quadVertices }, // std::span extracts the size from the C array, preventing decay.
284
285 quad_VBO.set_layout(layout_3P_2UV);
286 staplegl::vertex_array quad_VAO;
287
288 quad_VAO.add_vertex_buffer(std::move(quad_VBO));
289 quad_VAO.unbind();
290
291 /*
292
293 UNIFORMS:
294
295 we have three uniform blocks, one holding the camera data and the various matrices,
296 one holding the light data, and one holding the material data.
297
298 */
299
300 staplegl::vertex_buffer_layout const camera_block_layout {
301 { u_type::mat4, "projection" },
302 { u_type::mat4, "view" },
303 { u_type::mat4, "model" },
304 { u_type::vec4, "camera_pos" }
305 };
306
307 staplegl::uniform_buffer camera_block { camera_block_layout, 0 };
308
309 staplegl::vertex_buffer_layout const light_block_layout {
310 { u_type::vec4, "light_pos" },
311 { u_type::vec4, "light_color" },
312 { u_type::vec4, "light_attenuation" }, // 0 : constant, 1 : linear, 2 : quadratic, 3 : padding
313 { u_type::vec2, "light_intensities" } // 0 : diffuse, 1 : specular
314 };
315
316 staplegl::uniform_buffer light_block { light_block_layout, 1 };
317
318 const glm::vec4 light_pos { 1.0F, 1.0F, 10.0F, 1.0F };
319 const glm::vec4 light_color { 0.9333F, 0.5098, 0.9333F, 1.0F };
320
321 light_block.bind();
322 light_block.set_attribute_data(std::span { glm::value_ptr(light_pos), 4 }, "light_pos");
323 light_block.set_attribute_data(std::span { glm::value_ptr(light_color), 4 }, "light_color");
324 light_block.set_attribute_data(std::span { glm::value_ptr(glm::vec4(1.0F, 0.22F, 0.20F, 0.0F)), 4 }, "light_attenuation");
325 light_block.set_attribute_data(std::span { glm::value_ptr(glm::vec2(luminosity, 1.2F)), 2 }, "light_intensities");
326 light_block.unbind();
327
328 staplegl::vertex_buffer_layout const material_block_layout {
329 { u_type::vec4, "material_color" },
330 { u_type::float32, "material_shininess" },
331 { u_type::float32, "material_roughness" }
332 };
333 staplegl::uniform_buffer material_block { material_block_layout, 2 };
334
335 // teapot materials
336 const glm::vec4 teapot_color { 0.51F, 0.55F, 0.66F, 1.0F };
337 const float teapot_shininess = 32.0F;
338 const float teapot_roughness = 0.80F;
339
340 material_block.bind();
341 material_block.set_attribute_data(std::span { glm::value_ptr(teapot_color), 4 }, "material_color");
342 material_block.set_attribute_data(std::span { &teapot_shininess, 1 }, "material_shininess");
343 material_block.set_attribute_data(std::span { &teapot_roughness, 1 }, "material_roughness");
344 material_block.unbind();
345
346 // load up cubemap texture
347 // goes in order: right, left, top, bottom, front, back
348 std::array<std::string, 6> const faces {
349 "./assets/skybox/right.jpg", "./assets/skybox/left.jpg",
350 "./assets/skybox/top.jpg", "./assets/skybox/bottom.jpg",
351 "./assets/skybox/front.jpg", "./assets/skybox/back.jpg"
352 };
353
354 // prep for data transfer
355 std::array<std::span<std::byte>, 6> cube_data;
356
357 std::int32_t width = 0, height = 0, nrChannels = 0;
358 int i = 0;
359 for (auto& face : faces) {
360 auto* data = reinterpret_cast<std::byte*>( // NOLINT
361 stbi_load(face.c_str(), &width, &height, &nrChannels, 0));
362 if (data == nullptr) {
363 std::cout << "Failed to load texture" << std::endl;
364 return -1;
365 }
366 // pack data in array
367 cube_data[i++] = std::span<std::byte> { data, static_cast<size_t>(width * height * nrChannels) };
368 }
369
370 // create the cubemap texture
371 // srgb8 is used to ensure that the texture is correctly gamma corrected.
372 staplegl::cubemap const skybox {
373 cube_data, { width, height },
374 { .internal_format = GL_SRGB8, .format = GL_RGB, .datatype = GL_UNSIGNED_BYTE },
375 { .min_filter = GL_LINEAR, .mag_filter = GL_LINEAR, .clamping = GL_CLAMP_TO_EDGE },
376 true
377 };
378
379 for (auto& face : cube_data) {
380 stbi_image_free(face.data());
381 }
382
383 // set the texture to unit 0, so that the shader can find it.
384 skybox.set_unit(0);
385
386 glClearColor(0.F, 0.F, 0.F, 1.0F);
387
388 while (glfwWindowShouldClose(window) == 0) {
389 // input
390 // -----
391 processInput(window);
392
393 light_block.bind();
394 light_block.set_attribute_data(std::span { glm::value_ptr(glm::vec2(luminosity, 1.2F)), 2 }, "light_intensities");
395
396 // Rotate the camera around the teapot.
397 const auto time = static_cast<float>(glfwGetTime());
398 const float radius = 4.0F;
399 const float slow_factor = 0.25F;
400
401 const float camX = sin(time * slow_factor) * radius;
402 const float camZ = cos(time * slow_factor) * radius;
403 const float camY = sin(time * slow_factor) + 0.5F; // slightly over the teapot.
404
405 // gen camera vector and make it look at the teapot (center of the screen).
406 glm::vec4 camera_pos { camX, camY, camZ, 1.0F };
407
408 // gen the view and projection matrices.
409 glm::mat4 view = glm::lookAt(glm::vec3(camera_pos), glm::vec3(0.0F, 0.0F, 0.0F),
410 glm::vec3(0.0F, 1.0F, 0.0F));
411 glm::mat4 projection = glm::perspective(glm::radians(90.0F), aspect_ratio, 0.01F, 100.0F);
412
413 // send the data to the GPU through the uniform block.
414 camera_block.bind();
415 camera_block.set_attribute_data(
416 std::span { glm::value_ptr(camera_pos), 4 }, "camera_pos");
417 camera_block.set_attribute_data(std::span { glm::value_ptr(view), 16 }, "view");
418 camera_block.set_attribute_data(std::span { glm::value_ptr(projection), 16 }, "projection");
419
420 // first off, we bind the framebuffer and start drawing to our HDR texture.
421 msaa_fbo.bind();
422 msaa_fbo.set_texture(msaa_color);
423 msaa_fbo.set_renderbuffer(
424 { SCR_WIDTH,
425 SCR_HEIGHT }, // get a renderbuffer of the same size as the screen.
427 MSAA); // set the same samples as the color texture.
428
429 if (!msaa_fbo.assert_completeness()) [[unlikely]] {
430 std::cerr << "Framebuffer not complete, line: " << __LINE__ << std::endl;
431 return EXIT_FAILURE; // check for completeness
432 }
433
434 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); // clear the framebuffer
435
436 /*
437
438 Rendering with MSAA
439
440 */
441
442 // prep to render the skybox
443 skybox_VAO.bind();
444 skybox_shader.bind();
445
446 // make the skybox huge and centered on the camera.
447 // in a more complex scene you would have it follow the camera.
448 glm::mat4 skybox_mat = glm::scale(glm::mat4(1.0F), glm::vec3(50.0F, 50.0F, 50.0F));
449 camera_block.set_attribute_data(std::span { glm::value_ptr(skybox_mat), 16 }, "model");
450
451 // disable depth writing so that the skybox is always drawn behind everything else.
452 // we do not need to invert culling due to the skybox's indices being wound in the opposite direction.
453 glDepthMask(GL_FALSE);
454 glDrawElements(GL_TRIANGLES, skybox_VAO.index_data().count(), GL_UNSIGNED_INT, nullptr);
455 glDepthMask(GL_TRUE);
456
457 // draw the blue light source
458 // we use the same VAO as the skybox, but we change the shader and the model matrix.
459
460 glm::mat4 light_mat = glm::translate(glm::mat4(1.0F), glm::vec3(light_pos));
461 camera_block.set_attribute_data(std::span { glm::value_ptr(light_mat), 16 }, "model");
462
463 light_shader.bind();
464 // since the skybox has an inverted winding order, we need to invert culling
465 // (we want to see the cube from outside now), so the effect is not cancelled out.
466 glFrontFace(GL_CW);
467 glDrawElements(GL_TRIANGLES, skybox_VAO.index_data().count(), GL_UNSIGNED_INT, nullptr);
468 glFrontFace(GL_CCW); // on a real program, use better models :)
469
470 // draw the teapot
471
472 // The teapot model matrix flips it upside-down, so that it is rendered correctly.
473 glm::mat4 model_mat = glm::scale(model, glm::vec3(1.0F, -1.0F, 1.0F));
474 camera_block.set_attribute_data(std::span { glm::value_ptr(model_mat), 16 }, "model");
475
476 VAO.bind();
477 teapot_shader.bind();
478
479 glDrawElements(GL_TRIANGLES, VAO.index_data().count(), GL_UNSIGNED_INT, nullptr);
480
481 /*
482
483 Resolve MSAA to HDR framebuffer
484
485 */
486
487 post_fbo.bind();
488 post_fbo.set_texture(hdr_color);
489
491
492 post_fbo.bind();
493 post_fbo.set_renderbuffer({ 0, 0 }, staplegl::fbo_attachment::NONE);
494
495 /*
496
497 Post-processing
498
499 */
500
501 // bind screen-quad, every draw call will now simply
502 // transfer content from one texture to another.
503
504 quad_VAO.bind();
505
506 // copy stuff to the bloom pyramid
507 passthrough_shader.bind();
508 hdr_color.set_unit(1);
509 post_fbo.set_texture(pyramid_textures[0], 0);
510
511 glDrawArrays(GL_TRIANGLE_STRIP, 0, 4); // copy the scene to the lowest level of the pyramid
512
513 downsample_shader.bind();
514
515 // work our way down the levels of the pyramid, downsampling the texture each time.
516 for (size_t i = 0; i < pyramid_textures.size() - 1; i++) {
517 // downsample from texture i to texture i + 1
518
519 auto& draw_source = pyramid_textures[i];
520 auto& draw_target = pyramid_textures[i + 1];
521 auto const& t_res = draw_target.get_resolution();
522
523 draw_source.set_unit(1);
524
525 post_fbo.set_texture(draw_target, 0);
526 post_fbo.set_viewport(
527 { t_res.width,
528 t_res.height });
529
530 downsample_shader.upload_uniform2f(
531 "uResolution", static_cast<float>(t_res.width),
532 static_cast<float>(t_res.height));
533
534 glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
535 }
536
537 // now every level of the pyramid is filled with a downsampled version of the scene.
538 // we can now upsample the pyramid and add the results together to get the bloom effect.
539
540 // blending for Bloom
541 glEnable(GL_BLEND);
542 glBlendFunc(GL_ONE, GL_ONE);
543 glBlendEquation(GL_FUNC_ADD);
544
545 upsample_shader.bind();
546
547 for (int i = pyramid_textures.size() - 1; i > 0; i--) {
548 // upsample from texture i to texture i - 1
549
550 auto& draw_source = pyramid_textures[i];
551 auto& draw_target = pyramid_textures[i - 1];
552
553 draw_source.set_unit(1);
554
555 post_fbo.set_texture(draw_target, 0);
556 post_fbo.set_viewport(
557 { draw_target.get_resolution().width,
558 draw_target.get_resolution().height });
559 glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
560 }
561 glDisable(GL_BLEND);
562
563 // After upscaling the pyramid, we have the bloom effect in the lowest level of the pyramid.
564 // we can combine it with the original scene to get the final result.
565
566 // HDR post-processing
568 glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);
569
570 tonemap_shader.bind();
571 // bind the pyramid texture and the original scene
572 hdr_color.set_unit(1);
573 pyramid_textures[0].set_unit(2);
574
575 glDrawArrays(GL_TRIANGLE_STRIP, 0, 4);
576
577 glfwSwapBuffers(window);
578 glfwPollEvents();
579 }
580
581 // no need to de-allocate anything as staplegl handles all the OpenGL objects in a RAII fashion.
582
583 // glfw: terminate, clearing all previously allocated GLFW resources.
584 // ------------------------------------------------------------------
585 glfwTerminate();
586 return 0;
587}
588
589// process all input: query GLFW whether relevant keys are pressed/released this
590// frame and react accordingly
591// ---------------------------------------------------------------------------------------------------------
592void processInput(GLFWwindow* window)
593{
594 if (glfwGetKey(window, GLFW_KEY_ESCAPE) == GLFW_PRESS) {
595 glfwSetWindowShouldClose(window, 1);
596 }
597 if (glfwGetKey(window, GLFW_KEY_U) == GLFW_PRESS) {
598 luminosity += 0.5F;
599 }
600 if (glfwGetKey(window, GLFW_KEY_D) == GLFW_PRESS) {
601 luminosity = std::max(0.0F, luminosity - 0.5F);
602 }
603 if (glfwGetKey(window, GLFW_KEY_X) == GLFW_PRESS) {
604 // toggle mesh wireframe
605 static bool wireframe = false;
606 wireframe = !wireframe;
607 glPolygonMode(GL_FRONT_AND_BACK, wireframe ? GL_LINE : GL_FILL);
608 }
609}
610
611// glfw: whenever the window size changed (by OS or user resize) this callback
612// function executes
613// ---------------------------------------------------------------------------------------------
614void framebuffer_size_callback(GLFWwindow* /*window*/, int width, int height)
615{
616 // make sure the viewport matches the new window dimensions; note that width
617 // and height will be significantly larger than specified on retina displays.
618 glViewport(0, 0, width, height);
619 aspect_ratio = static_cast<float>(width) / static_cast<float>(height);
620}
const int32_t SCR_WIDTH
Definition batches.cpp:67
void processInput(GLFWwindow *window)
Definition batches.cpp:239
void framebuffer_size_callback(GLFWwindow *window, int width, int height)
Definition batches.cpp:246
void GLAPIENTRY MessageCallback(GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *message, const void *userParam)
Definition batches.cpp:50
auto main() -> int
Definition batches.cpp:70
const int32_t SCR_HEIGHT
Definition batches.cpp:68
Cube map texture wrapper.
Definition cubemap.hpp:33
Framebuffer Object (FBO) wrapper.
static void bind_default()
Bind the default framebuffer.
static void transfer_data(framebuffer const &src, framebuffer const &dst, resolution res)
Transfer the contents of a framebuffer to another.
Element Buffer Object (EBO) wrapper.
Shader program class.
Definition shader.hpp:77
void bind() const
Bind the shader program.
Definition shader.hpp:367
2D texture wrapper.
Definition texture.hpp:97
Uniform Buffer Object (UBO) wrapper.
void bind() const
Bind the uniform to the OpenGL context.
Vertex Array Object (VAO) wrapper.
void bind() const
Bind the vertex array object.
void set_index_buffer(index_buffer &&ibo)
Set the index buffer object.
static void unbind()
Unbind the vertex array object.
constexpr auto index_data() -> index_buffer &
Get the index buffer object.
auto add_vertex_buffer(vertex_buffer &&vbo) -> vertex_array::iterator_t
Add a vertex buffer to the vertex array object.
Vertex Buffer Object (VBO) wrapper.
void set_layout(const vertex_buffer_layout &layout)
Set the layout object.
StapleGL, a C++20 wrapper for OpenGL.
A struct that represents an image's dimensions.
Definition utility.hpp:28
OpenGL texture details relating to the color format and data type.
Definition texture.hpp:39
std::int32_t internal_format
Definition texture.hpp:40
OpenGL texture details relating to filtering and clamping of the image.
Definition texture.hpp:49
std::int32_t min_filter
Definition texture.hpp:50
float luminosity
Definition teapot.cpp:63
float aspect_ratio
Definition teapot.cpp:105
constexpr auto MSAA_SAMPLES
Definition teapot.cpp:48
constexpr auto MSAA
Definition teapot.cpp:47
constexpr auto calc_pyramid_levels(uint32_t width, uint32_t height) -> uint32_t
Definition teapot.cpp:92