Framebuffer.h 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114
  1. #ifndef FRAMEBUFFER_H
  2. #define FRAMEBUFFER_H
  3. #include <iostream>
  4. #include "utils/ArrayList.h"
  5. #include "utils/Size.h"
  6. #include "wrapper/Texture.h"
  7. template<int N>
  8. class Framebuffer final {
  9. ArrayList<Texture, N> textures;
  10. GLuint buffer;
  11. public:
  12. template<typename... Args>
  13. Framebuffer(const TextureFormat& a, Args&&... args) : buffer(0) {
  14. const int size = sizeof...(args) + 1;
  15. TextureFormat init[size] = {a, args...};
  16. static_assert(N == size,
  17. "framebuffer size and amount of arguments do not match");
  18. for(int i = 0; i < N; i++) {
  19. textures.add(init[i]);
  20. textures[i].setClampWrap();
  21. if(init[i].linear) {
  22. textures[i].setLinearFilter();
  23. }
  24. }
  25. }
  26. ~Framebuffer() {
  27. glDeleteFramebuffers(1, &buffer);
  28. }
  29. Framebuffer(const Framebuffer&) = delete;
  30. Framebuffer(Framebuffer&&) = delete;
  31. Framebuffer& operator=(const Framebuffer&) = delete;
  32. Framebuffer& operator=(Framebuffer&&) = delete;
  33. bool init(const Size& size) {
  34. glGenFramebuffers(1, &buffer);
  35. glBindFramebuffer(GL_FRAMEBUFFER, buffer);
  36. ArrayList<GLenum, N> attachments;
  37. for(Texture& t : textures) {
  38. t.setData(size.width, size.height);
  39. if(t.format.depth) {
  40. glFramebufferTexture2D(GL_FRAMEBUFFER, GL_DEPTH_ATTACHMENT,
  41. GL_TEXTURE_2D, t.texture, 0);
  42. } else {
  43. GLenum c = GL_COLOR_ATTACHMENT0 + attachments.getLength();
  44. glFramebufferTexture2D(GL_FRAMEBUFFER, c, GL_TEXTURE_2D,
  45. t.texture, 0);
  46. attachments.add(c);
  47. }
  48. }
  49. glDrawBuffers(attachments.getLength(), attachments.begin());
  50. return hasError();
  51. }
  52. void bindAndClear() {
  53. glBindFramebuffer(GL_FRAMEBUFFER, buffer);
  54. glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT |
  55. GL_STENCIL_BUFFER_BIT);
  56. }
  57. void bindTextureTo(int index, int textureUnit) const {
  58. textures[index].bindTo(textureUnit);
  59. }
  60. void resize(const Size& size) {
  61. for(Texture& t : textures) {
  62. t.setData(size.width, size.height);
  63. }
  64. }
  65. private:
  66. bool hasError() const {
  67. GLenum error = glCheckFramebufferStatus(GL_FRAMEBUFFER);
  68. if(error == GL_FRAMEBUFFER_COMPLETE) {
  69. return false;
  70. }
  71. switch(error) {
  72. case GL_FRAMEBUFFER_UNDEFINED:
  73. std::cout << "undefined framebuffer\n";
  74. return true;
  75. case GL_FRAMEBUFFER_INCOMPLETE_ATTACHMENT:
  76. std::cout << "incomplete framebuffer attachment\n";
  77. return true;
  78. case GL_FRAMEBUFFER_INCOMPLETE_MISSING_ATTACHMENT:
  79. std::cout << "incomplete missing framebuffer attachment\n";
  80. return true;
  81. case GL_FRAMEBUFFER_INCOMPLETE_DRAW_BUFFER:
  82. std::cout << "incomplete framebuffer draw buffer\n";
  83. return true;
  84. case GL_FRAMEBUFFER_INCOMPLETE_READ_BUFFER:
  85. std::cout << "incomplete framebuffer read buffer\n";
  86. return true;
  87. case GL_FRAMEBUFFER_UNSUPPORTED:
  88. std::cout << "unsupported framebuffer\n";
  89. return true;
  90. case GL_FRAMEBUFFER_INCOMPLETE_MULTISAMPLE:
  91. std::cout << "incomplete framebuffer multisample\n";
  92. return true;
  93. case GL_FRAMEBUFFER_INCOMPLETE_LAYER_TARGETS:
  94. std::cout << "incomplete framebuffer layer targets\n";
  95. return true;
  96. }
  97. return "unknown error";
  98. }
  99. };
  100. #endif