From 5e53c905a8d6652c066bc4024a07fbddb5d2041a Mon Sep 17 00:00:00 2001 From: Lauchmelder Date: Tue, 21 Dec 2021 00:51:18 +0100 Subject: [PATCH] added wireframe cube --- CMakeLists.txt | 3 +- src/Application.cpp | 57 +++------------ src/Application.hpp | 6 +- src/CMakeLists.txt | 9 ++- src/Cuboid.cpp | 95 +++++++++++++++++++++++++ src/Cuboid.hpp | 24 +++++++ src/Util.hpp | 3 + src/backend/ObjectManager.hpp | 71 ++++++++++++++++++ src/{ => backend}/Shader.cpp | 14 ++++ src/{ => backend}/Shader.hpp | 4 ++ src/{ => backend}/VertexArrayObject.cpp | 4 +- src/{ => backend}/VertexArrayObject.hpp | 2 +- 12 files changed, 233 insertions(+), 59 deletions(-) create mode 100644 src/Cuboid.cpp create mode 100644 src/Cuboid.hpp create mode 100644 src/Util.hpp create mode 100644 src/backend/ObjectManager.hpp rename src/{ => backend}/Shader.cpp (83%) rename src/{ => backend}/Shader.hpp (87%) rename src/{ => backend}/VertexArrayObject.cpp (93%) rename src/{ => backend}/VertexArrayObject.hpp (97%) diff --git a/CMakeLists.txt b/CMakeLists.txt index 7be59e4..c815a89 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -19,7 +19,8 @@ if(NOT GLM_FOUND) message(STATUS "Could not find GLM on system, including from source instead") add_subdirectory("vendor/glm") - set(GLM_INCLUDE_DIRS glm::glm_static) + set(GLM_INCLUDE_DIRS glm) + set(GLM_LIBRARIES glm) endif() # Include sub-projects. diff --git a/src/Application.cpp b/src/Application.cpp index f4c12d8..4220efa 100644 --- a/src/Application.cpp +++ b/src/Application.cpp @@ -8,6 +8,9 @@ Application::~Application() { + if (cube != nullptr) + delete cube; + if (window != nullptr) { glfwDestroyWindow(window); @@ -20,7 +23,7 @@ Application::~Application() void Application::Init(int width, int height, const std::string& title) { // Initialize GLFW - if(window == nullptr) + if (window == nullptr) glfwInit(); GLFWmonitor* monitor = glfwGetPrimaryMonitor(); @@ -59,9 +62,10 @@ void Application::Init(int width, int height, const std::string& title) } glViewport(0, 0, mode->width, mode->height); + glEnable(GL_DEPTH_TEST); // Register GLFW callbacks - glfwSetFramebufferSizeCallback(window, + glfwSetFramebufferSizeCallback(window, [](GLFWwindow* window, int width, int height) { glViewport(0, 0, width, height); @@ -79,49 +83,7 @@ void Application::Init(int width, int height, const std::string& title) } ); - - // TODO: Remove, this should probably be done elsewhere - model = VAOFactory::Produce( - { - -0.5f, -0.5f, - 0.0f, 0.5f, - 0.5f, -0.5f - }, - { - 0, 1, 2 - }, - { - { 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), (void*)0 } - } - ); - - shader = ShaderFactory::Produce( - R"( - #version 460 core - - layout (location = 0) in vec2 aPos; - - void main() - { - gl_Position = vec4(aPos, 0.0, 1.0); - } - )", - R"( - #version 460 core - - out vec4 FragColor; - - void main() - { - FragColor = vec4(0.5f, 0.0f, 0.8f, 1.0f); - } - )" - ); - - if (!shader->Good()) - { - throw std::runtime_error("Shader creation failed"); - } + cube = new Cuboid(); } void Application::Launch() @@ -131,10 +93,9 @@ void Application::Launch() glfwPollEvents(); glClearColor(0.0f, 0.0f, 0.0f, 1.0f); - glClear(GL_COLOR_BUFFER_BIT); + glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); - shader->Use(); - model->Render(); + cube->Render(); glfwSwapBuffers(window); } diff --git a/src/Application.hpp b/src/Application.hpp index 13ad921..032466a 100644 --- a/src/Application.hpp +++ b/src/Application.hpp @@ -1,8 +1,7 @@ #pragma once #include -#include "Shader.hpp" -#include "VertexArrayObject.hpp" +#include "Cuboid.hpp" struct GLFWwindow; @@ -35,6 +34,5 @@ public: private: GLFWwindow* window = nullptr; - Shader shader = nullptr; - VertexArrayObject model = nullptr; + Cuboid* cube; }; \ No newline at end of file diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index 13a5ae8..efab66d 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -1,15 +1,18 @@ add_executable(visualizer "main.cpp" "Application.cpp" - "VertexArrayObject.cpp" "Shader.cpp") + "backend/VertexArrayObject.cpp" "backend/Shader.cpp" "Cuboid.cpp") target_sources(visualizer PUBLIC ${CMAKE_SOURCE_DIR}/vendor/glad/src/glad.c ) -target_include_directories(visualizer PUBLIC +target_include_directories(visualizer PRIVATE ${GLFW3_INCLUDE_DIRS} ${GLM_INCLUDE_DIRS} ${CMAKE_SOURCE_DIR}/vendor/glad/include ) -target_link_libraries(visualizer PUBLIC ${GLFW3_INCLUDE_DIRS}) \ No newline at end of file +target_link_libraries(visualizer PRIVATE + ${GLFW3_LIBRARIES} + ${GLM_LIBRARIES} +) \ No newline at end of file diff --git a/src/Cuboid.cpp b/src/Cuboid.cpp new file mode 100644 index 0000000..9f7003b --- /dev/null +++ b/src/Cuboid.cpp @@ -0,0 +1,95 @@ +#include "Cuboid.hpp" + +#include + +#include +#include + +#include "backend/ObjectManager.hpp" +#include "Util.hpp" + +Cuboid::Cuboid() +{ + vao = VAOManager::GetInstance().Get(CUBOID_ID); + if (vao == nullptr) + { + vao = VAOFactory::Produce( + { + -1.0f, -1.0f, -1.0f, + 1.0f, -1.0f, -1.0f, + 1.0f, 1.0f, -1.0f, + -1.0f, 1.0f, -1.0f, + -1.0f, -1.0f, 1.0f, + 1.0f, -1.0f, 1.0f, + 1.0f, 1.0f, 1.0f, + -1.0f, 1.0f, 1.0f + }, + { + 0, 1, 1, 2, 2, 3, 3, 0, // Front + 4, 5, 5, 6, 6, 7, 7, 4, // Back + 0, 4, 1, 5, 2, 6, 3, 7 + }, + { + { 3, GL_FLOAT, GL_FALSE, 3 * sizeof(float), (void*)0 } + } + ); + + VAOManager::GetInstance().Register(CUBOID_ID, vao); + } + + shader = ShaderManager::GetInstance().Get(CUBOID_ID); + if (shader == nullptr) + { + shader = ShaderFactory::Produce( + R"( + #version 460 core + + layout (location = 0) in vec3 aPos; + + uniform mat4 model;msp + uniform mat4 view; + uniform mat4 perspective; + + void main() + { + gl_Position = perspective * view * model * vec4(aPos, 1.0f); + } + )", + R"( + #version 460 core + + out vec4 FragColor; + + void main() + { + FragColor = vec4(1.0f, 1.0f, 1.0f, 1.0f); + } + )" + ); + + if (!shader->Good()) + { + throw std::runtime_error("Shader creation failed"); + } + + ShaderManager::GetInstance().Register(CUBOID_ID, shader); + } + + model = glm::mat4(1.0); + + view = glm::lookAt(glm::vec3(2.0f, 2.0f, -3.5f), glm::vec3(0.0f), glm::vec3(0.0f, 1.0f, 0.0f)); + + perspective = glm::perspective(glm::radians(100.0f), 16.0f / 9.0f, 0.01f, 100.0f); +} + +void Cuboid::Render() +{ + glPolygonMode(GL_FRONT_AND_BACK, GL_LINE); + + shader->Use(); + shader->SetUniform("model", model); + shader->SetUniform("view", view); + shader->SetUniform("perspective", perspective); + + vao->Render(GL_LINES); +} diff --git a/src/Cuboid.hpp b/src/Cuboid.hpp new file mode 100644 index 0000000..2ddd019 --- /dev/null +++ b/src/Cuboid.hpp @@ -0,0 +1,24 @@ +#pragma once + +#include +#include "backend/VertexArrayObject.hpp" +#include "backend/Shader.hpp" + +/** + * A cuboid that sits at a position and expands into all + * three spatial directions + */ +class Cuboid +{ +public: + Cuboid(); + + void Render(); + +private: + VertexArrayObject vao; + Shader shader; + + // TODO: Remove view and projection matrices from cube class + glm::mat4 model, view, perspective; +}; \ No newline at end of file diff --git a/src/Util.hpp b/src/Util.hpp new file mode 100644 index 0000000..2efc9f5 --- /dev/null +++ b/src/Util.hpp @@ -0,0 +1,3 @@ +#pragma once + +#define CUBOID_ID 0x1 \ No newline at end of file diff --git a/src/backend/ObjectManager.hpp b/src/backend/ObjectManager.hpp new file mode 100644 index 0000000..f8bf050 --- /dev/null +++ b/src/backend/ObjectManager.hpp @@ -0,0 +1,71 @@ +#pragma once + +#include +#include + +/** + * Some objects should only exist once but be available to multiple objects (e.g. multiple + * models sharing the same VAO and shaders. Any object can register objects here and other + * objects can then retrieve them if they have the ID. + * + * Objects are stored and returned as shared_ptr's, so even if a part of the program deletes + * an object from the manager, any part of the program that has the objects stored in them will + * not break and still work. But realistically this shouldn't happen in the first place. + * As a consequence, even if no objects are using an object stored in here it will continue to + * exist. + */ +template +class ObjectManager +{ +public: + static ObjectManager& GetInstance() + { + static ObjectManager instance; + return instance; + } + +public: + ObjectManager(const ObjectManager&) = delete; + void operator=(const ObjectManager&) = delete; + + /** + * Add new (existing) object to manager + */ + inline void Register(unsigned int id, std::shared_ptr obj) + { + objects.insert(std::make_pair(id, obj)); + } + + /** + * Remove object from manager + */ + inline void Delete(unsigned int id) + { + objects.erase(id); + } + + /** + * Retrieve object from manager + */ + inline std::shared_ptr Get(unsigned int id) + { + auto it = objects.find(id); + + if (it == objects.end()) + return nullptr; + + return it->second; + } + +private: + ObjectManager() {} + +private: + std::map> objects; +}; + +class AbstractShader; +class AbstractVertexArrayObject; + +typedef ObjectManager VAOManager; +typedef ObjectManager ShaderManager; \ No newline at end of file diff --git a/src/Shader.cpp b/src/backend/Shader.cpp similarity index 83% rename from src/Shader.cpp rename to src/backend/Shader.cpp index b400ceb..9a1ccd4 100644 --- a/src/Shader.cpp +++ b/src/backend/Shader.cpp @@ -1,8 +1,13 @@ #include "Shader.hpp" #include + +#include #include +#define IMPLEMENT_UNIFORM_FUNCTION(type, func) \ +inline + AbstractShader::AbstractShader(const std::string& vertexShader, const std::string& fragmentShader) : id(0) { @@ -72,3 +77,12 @@ void AbstractShader::Use() { glUseProgram(id); } + +void AbstractShader::SetUniform(const std::string& name, const glm::mat4& value) +{ + GLint location = glGetUniformLocation(id, name.c_str()); + if (location == -1) + return; + + glUniformMatrix4fv(location, 1, GL_FALSE, glm::value_ptr(value)); +} \ No newline at end of file diff --git a/src/Shader.hpp b/src/backend/Shader.hpp similarity index 87% rename from src/Shader.hpp rename to src/backend/Shader.hpp index 015d5e1..98b135c 100644 --- a/src/Shader.hpp +++ b/src/backend/Shader.hpp @@ -1,5 +1,7 @@ #pragma once +#include + #include #include @@ -15,6 +17,8 @@ public: inline bool Good() { return id != 0; } void Use(); + void SetUniform(const std::string& name, const glm::mat4& value); + private: unsigned int id; }; diff --git a/src/VertexArrayObject.cpp b/src/backend/VertexArrayObject.cpp similarity index 93% rename from src/VertexArrayObject.cpp rename to src/backend/VertexArrayObject.cpp index 5524534..72e7fed 100644 --- a/src/VertexArrayObject.cpp +++ b/src/backend/VertexArrayObject.cpp @@ -10,13 +10,13 @@ AbstractVertexArrayObject::~AbstractVertexArrayObject() glDeleteVertexArrays(1, &vao); } -void AbstractVertexArrayObject::Render() +void AbstractVertexArrayObject::Render(GLenum mode) { assert(vao != 0); glBindVertexArray(vao); // GLenum result = glGetError(); - glDrawElements(GL_TRIANGLES, indexCount, GL_UNSIGNED_INT, 0); + glDrawElements(mode, indexCount, GL_UNSIGNED_INT, 0); } AbstractVertexArrayObject::AbstractVertexArrayObject(const VertexArray& vertices, const IndexArray& indices, const Layout& layout, Usage usage) : diff --git a/src/VertexArrayObject.hpp b/src/backend/VertexArrayObject.hpp similarity index 97% rename from src/VertexArrayObject.hpp rename to src/backend/VertexArrayObject.hpp index c602f22..3b644ca 100644 --- a/src/VertexArrayObject.hpp +++ b/src/backend/VertexArrayObject.hpp @@ -34,7 +34,7 @@ public: AbstractVertexArrayObject(const VertexArray& vertices, const IndexArray& indices, const Layout& layout, Usage usage); ~AbstractVertexArrayObject(); - void Render(); + void Render(unsigned int mode = 4); private: unsigned int vao, vbo, ebo;