diff --git a/CMakeLists.txt b/CMakeLists.txt index d5c6e2b..8551207 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -5,6 +5,8 @@ cmake_minimum_required (VERSION 3.8) project ("Visualizer") +set(CMAKE_CXX_STANDARD 17) + set(VENDOR_DIR ${CMAKE_SOURCE_DIR}/vendor) find_package(GLFW3) diff --git a/backend/include/backend/BoundingBox.hpp b/backend/include/backend/BoundingBox.hpp new file mode 100644 index 0000000..fe7ff0e --- /dev/null +++ b/backend/include/backend/BoundingBox.hpp @@ -0,0 +1,15 @@ +#pragma once + +struct Rect +{ + float x, y; + float w, h; +}; + +struct BoundingBox +{ + float x, y, z; + float w, h, d; +}; + +typedef BoundingBox BBox; \ No newline at end of file diff --git a/backend/include/backend/Camera.hpp b/backend/include/backend/Camera.hpp index 1a6e614..210bd3c 100644 --- a/backend/include/backend/Camera.hpp +++ b/backend/include/backend/Camera.hpp @@ -13,6 +13,11 @@ public: void SetScale(const glm::vec3&) = delete; void Scale(const glm::vec3&) = delete; + inline void LookAt(const glm::vec3& target) + { + transformation = glm::lookAt(position, target, glm::vec3(0.0f, 1.0f, 0.0f)); + } + inline const glm::mat4& GetView() const { return transformation; @@ -40,6 +45,11 @@ public: { projection = glm::perspective(glm::radians(fov), aspect, zNear, zFar); } + + inline void Update(float fov, float aspect, float zNear, float zFar) + { + projection = glm::perspective(glm::radians(fov), aspect, zNear, zFar); + } }; class OrthogonalCamera : public CameraBase @@ -49,4 +59,9 @@ public: { projection = glm::ortho(left, right, bottom, top, zNear, zFar); } + + inline void Update(float left, float right, float bottom, float top, float zNear, float zFar) + { + projection = glm::ortho(left, right, bottom, top, zNear, zFar); + } }; \ No newline at end of file diff --git a/backend/include/backend/Drawable.hpp b/backend/include/backend/Drawable.hpp index a46f7b4..88f5540 100644 --- a/backend/include/backend/Drawable.hpp +++ b/backend/include/backend/Drawable.hpp @@ -23,7 +23,7 @@ public: Drawable(const Drawable& other) = delete; void operator=(const Drawable& other) = delete; - virtual void InitializeShader(const CameraBase& camera) const = 0; + virtual void PreRender(const CameraBase& camera) const { }; void Draw(const CameraBase& camera) const; void SetPrimitiveType(PrimitiveType type); diff --git a/backend/src/Drawable.cpp b/backend/src/Drawable.cpp index 1d3a33c..9e75add 100644 --- a/backend/src/Drawable.cpp +++ b/backend/src/Drawable.cpp @@ -3,7 +3,7 @@ void Drawable::Draw(const CameraBase& camera) const { shader->Use(); - InitializeShader(camera); + PreRender(camera); vao->Render(static_cast(type)); } diff --git a/backend/src/Transformable.cpp b/backend/src/Transformable.cpp index ae1a4d4..860d29f 100644 --- a/backend/src/Transformable.cpp +++ b/backend/src/Transformable.cpp @@ -1,7 +1,7 @@ #include "backend/Transformable.hpp" Transformable::Transformable() : - position(0.0f), scale(1.0f), orientation(0.0, 0.0, 0.0, 1.0) + position(0.0f), scale(1.0f), orientation(glm::vec3(0.0, 0.0, 0.0)) { CalculateTransformationMatrix(); } @@ -41,6 +41,11 @@ void Transformable::SetRotation(const glm::vec3& axis, float angle) void Transformable::SetRotation(const glm::vec3& eulerAngles) { + /*orientation = glm::quat(0.0f, 0.0f, 0.0f, 1.0f); + orientation = glm::rotate(orientation, eulerAngles.x, glm::vec3(1.0f, 0.0f, 0.0f)); + orientation = glm::rotate(orientation, eulerAngles.y, glm::vec3(0.0f, 1.0f, 0.0f)); + orientation = glm::rotate(orientation, eulerAngles.z, glm::vec3(0.0f, 0.0f, 1.0f));*/ + orientation = glm::quat(eulerAngles); CalculateTransformationMatrix(); } @@ -71,7 +76,7 @@ void Transformable::Scale(const glm::vec3& factor) void Transformable::CalculateTransformationMatrix() { transformation = glm::mat4(1.0f); - transformation = glm::translate(transformation, -position); - transformation *= glm::toMat4(orientation); + transformation = glm::translate(transformation, position); + transformation *= glm::mat4(orientation); transformation = glm::scale(transformation, scale); } diff --git a/src/Application.cpp b/src/Application.cpp index e4a63ec..ed43e04 100644 --- a/src/Application.cpp +++ b/src/Application.cpp @@ -19,8 +19,8 @@ Application::~Application() ImGui_ImplGlfw_Shutdown(); ImGui::DestroyContext(); - if (grid != nullptr) - delete grid; + if (plot != nullptr) + delete plot; if (window != nullptr) { @@ -90,9 +90,8 @@ void Application::Init(int width, int height, const std::string& title) WindowData* data = (WindowData*)glfwGetWindowUserPointer(window); float aspectRatio = (float)width / (float)height; - *(data->camera) = Camera(100.0f, aspectRatio); - data->camera->Move(glm::vec3(0.0f, 0.0f, -4.0f)); - *(data->orthoCam) = OrthogonalCamera(-3.0f * aspectRatio, 3.0f * aspectRatio, -3.0f, 3.0f); + data->camera->Update(100.0f, aspectRatio, 0.01f, 100.0f); + data->orthoCam->Update(-3.0f * aspectRatio, 3.0f * aspectRatio, -3.0f, 3.0f, -100.0f, 100.0f); glViewport(0, 0, width, height); } @@ -123,13 +122,23 @@ void Application::Init(int width, int height, const std::string& title) float aspectRatio = (float)windowWidth / (float)windowHeight; camera = Camera(100.0f, aspectRatio); - camera.Move(glm::vec3(0.0f, 0.0f, 4.0f)); + camera.Move(glm::vec3(0.0f, 4.0f, -4.0f)); + camera.LookAt(glm::vec3(0.0f)); orthoCam = OrthogonalCamera(-3.0f * aspectRatio, 3.0f * aspectRatio, -3.0f, 3.0f); + orthoCam.Move(glm::vec3(0.0f, 4.0f, -4.0f)); + orthoCam.LookAt(glm::vec3(0.0f)); activeCamera = &camera; - grid = new Grid(glm::vec2(5.0f, 4.0f), 25, 20); + plot = new Plot3D({ -glm::two_pi(), -glm::two_pi(), -1.5f, glm::two_pi(), glm::two_pi(), 1.5f }, 0.5f, 0.1f, + [](float x, float y) + { + return (cos(x) + cos(y)) * 0.5f; + // return cos(x); + } + ); + cubePosition = glm::vec3(0.0f); cubeOrientation = glm::vec3(0.0f); cubeScale = glm::vec3(1.0f); @@ -139,7 +148,7 @@ void Application::Init(int width, int height, const std::string& title) glfwWindowHint(GLFW_SAMPLES, 4); - glEnable(GL_CULL_FACE); + // glEnable(GL_CULL_FACE); glEnable(GL_MULTISAMPLE); } @@ -149,9 +158,9 @@ void Application::Launch() { glfwPollEvents(); - grid->SetPosition(cubePosition); - grid->SetRotation(cubeOrientation); - grid->SetScale(cubeScale); + plot->SetPosition(cubePosition); + plot->SetRotation(cubeOrientation); + plot->SetScale(cubeScale); glClearColor(0.0f, 0.0f, 0.0f, 1.0f); glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT); @@ -160,14 +169,14 @@ void Application::Launch() ImGui_ImplGlfw_NewFrame(); ImGui::NewFrame(); - grid->Draw(*activeCamera); + plot->Draw(*activeCamera); ImGui::Begin("Debug"); - if (ImGui::CollapsingHeader("Grid")) + if (ImGui::CollapsingHeader("Plot")) { ImGui::SliderFloat3("Position", &(cubePosition[0]), -2.0f, 2.0f); - ImGui::SliderFloat3("Orientation", &(cubeOrientation[0]), 0.0f, glm::two_pi()); + ImGui::SliderFloat3("Orientation", &(cubeOrientation[0]), -glm::pi(), glm::pi()); ImGui::SliderFloat3("Scale", &(cubeScale[0]), 0.0f, 2.0f); } diff --git a/src/Application.hpp b/src/Application.hpp index 0890cce..556d194 100644 --- a/src/Application.hpp +++ b/src/Application.hpp @@ -2,7 +2,7 @@ #include #include "backend/Camera.hpp" -#include "Grid.hpp" +#include "Plot3D.hpp" struct GLFWwindow; @@ -47,5 +47,5 @@ private: CameraBase* activeCamera; glm::vec3 cubeOrientation, cubePosition, cubeScale; - Grid* grid; + Plot3D* plot; }; \ No newline at end of file diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt index da507de..d7c25b5 100644 --- a/src/CMakeLists.txt +++ b/src/CMakeLists.txt @@ -1,6 +1,6 @@ add_executable(visualizer - "main.cpp" "Application.cpp" "Cuboid.cpp" - "Grid.cpp") + "main.cpp" "Application.cpp" + "Plot3D.cpp") target_sources(visualizer PUBLIC ${VENDOR_DIR}/imgui/backends/imgui_impl_opengl3.cpp diff --git a/src/Cuboid.cpp b/src/Cuboid.cpp deleted file mode 100644 index d7d8ac4..0000000 --- a/src/Cuboid.cpp +++ /dev/null @@ -1,87 +0,0 @@ -#include "Cuboid.hpp" - -#include - -#include -#include - -#include "backend/ObjectManager.hpp" -#include "backend/Camera.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; - 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); - } - - type = PrimitiveType::Lines; -} - -void Cuboid::InitializeShader(const CameraBase& camera) const -{ - shader->SetUniform("model", transformation); - shader->SetUniform("view", camera.GetView()); - shader->SetUniform("perspective", camera.GetProjection()); -} diff --git a/src/Cuboid.hpp b/src/Cuboid.hpp deleted file mode 100644 index a0493f9..0000000 --- a/src/Cuboid.hpp +++ /dev/null @@ -1,18 +0,0 @@ -#pragma once - -#include -#include "backend/Drawable.hpp" -#include "backend/Transformable.hpp" - -/** - * A cuboid that sits at a position and expands into all - * three spatial directions - */ -class Cuboid : - public Drawable, public Transformable -{ -public: - Cuboid(); - - void InitializeShader(const CameraBase& camera) const override; -}; \ No newline at end of file diff --git a/src/Grid.cpp b/src/Grid.cpp deleted file mode 100644 index 4018f2a..0000000 --- a/src/Grid.cpp +++ /dev/null @@ -1,102 +0,0 @@ -#include "Grid.hpp" - -#include -#include - -#include "backend/Camera.hpp" -#include "backend/ObjectManager.hpp" -#include "Util.hpp" - -Grid::Grid(const glm::vec2& size, unsigned int linesAlongWidth, unsigned int linesAlongHeight) -{ - std::vector vertices; - std::vector indices; - - glm::vec2 halfSize = size / 2.0f; - - // Unoptimized memory usage - for (unsigned int x = 0; x <= linesAlongWidth; x++) - { - vertices.push_back(-halfSize.x + (float)x * size.x / (float)linesAlongWidth); - vertices.push_back(-halfSize.y); - - vertices.push_back(-halfSize.x + (float)x * size.x / (float)linesAlongWidth); - vertices.push_back(halfSize.y); - - indices.push_back(2 * x); - indices.push_back(2 * x + 1); - } - - unsigned int lastIndex = indices.back() + 1; - for (unsigned int y = 0; y <= linesAlongHeight; y++) - { - vertices.push_back(-halfSize.x); - vertices.push_back(-halfSize.y + (float)y * size.y / (float)linesAlongHeight); - - vertices.push_back(halfSize.x); - vertices.push_back(-halfSize.y + (float)y * size.y / (float)linesAlongHeight); - - indices.push_back(lastIndex + 2 * y); - indices.push_back(lastIndex + 2 * y + 1); - } - - vao = VAOFactory::Produce(vertices, indices, - { - { 2, GL_FLOAT, GL_FALSE, 2 * sizeof(float), (void*)0 } - } - ); - - shader = ShaderManager::GetInstance().Get(GRID_ID); - if (shader == nullptr) - { - shader = ShaderFactory::Produce( - R"( - #version 460 core - - layout (location = 0) in vec2 aPos; - - uniform mat4 model; - uniform mat4 view; - uniform mat4 perspective; - - void main() - { - gl_Position = perspective * view * model * vec4(aPos, 0.0f, 1.0f); - } - )", - R"( - #version 460 core - - out vec4 FragColor; - - uniform vec4 gridColor; - - void main() - { - FragColor = gridColor; - } - )" - ); - - if (!shader->Good()) - { - throw std::runtime_error("Shader creation failed"); - } - - ShaderManager::GetInstance().Register(GRID_ID, shader); - } - - type = PrimitiveType::Lines; -} - -void Grid::InitializeShader(const CameraBase& camera) const -{ - if(glm::dot(GetPosition() - camera.GetPosition(), glm::rotate(glm::inverse(GetQuaternion()), normal)) < 0.0f) - shader->SetUniform("gridColor", glm::vec4(0.5f)); - else - shader->SetUniform("gridColor", glm::vec4(0.0f)); - - shader->SetUniform("model", transformation); - shader->SetUniform("view", camera.GetView()); - shader->SetUniform("perspective", camera.GetProjection()); -} diff --git a/src/Grid.hpp b/src/Grid.hpp deleted file mode 100644 index f0fc37e..0000000 --- a/src/Grid.hpp +++ /dev/null @@ -1,16 +0,0 @@ -#pragma once - -#include "backend/Drawable.hpp" -#include "backend/Transformable.hpp" - -class Grid : - public Transformable, public Drawable -{ -public: - Grid(const glm::vec2& size, unsigned int linesAlongWidth, unsigned int linesAlongHeight); - - void InitializeShader(const CameraBase& camera) const override; - -private: - glm::vec3 normal = glm::vec3(0.0f, 0.0f, 1.0f); -}; \ No newline at end of file diff --git a/src/Plot3D.cpp b/src/Plot3D.cpp new file mode 100644 index 0000000..71d3380 --- /dev/null +++ b/src/Plot3D.cpp @@ -0,0 +1,209 @@ +#include "Plot3D.hpp" + +#include +#include +#include + +#include "backend/Camera.hpp" +#include "backend/ObjectManager.hpp" +#include "Util.hpp" + +inline float Map(const glm::vec2& from, const glm::vec2& to, float val) +{ + return (val - from.x) * (to.y - to.x) / (from.y - from.x) + to.x; +} + +Plot3D::Plot3D(const BBox& domainAndRange, float scale, float resolution, PlottableFunction func) +{ + // magic epsilon + if (resolution < 0.0001f) return; + + std::vector vertices; + std::vector indices; + + std::vector> functionValues; + unsigned int sliceLength = 0; + unsigned int slices = 0; + + // Intervals of the domain and (desired) range + glm::vec2 xInterval(domainAndRange.x, domainAndRange.w); + glm::vec2 yInterval(domainAndRange.y, domainAndRange.h); + glm::vec2 zInterval(domainAndRange.z, domainAndRange.d); + + // Bounding box of the graph in 3D space (centered at (0, 0, 0) + glm::vec2 xBounds = glm::vec2(-0.5f, 0.5f) * ((xInterval.y - xInterval.x) * scale); + glm::vec2 yBounds = glm::vec2(-0.5f, 0.5f) * ((yInterval.y - yInterval.x) * scale); + + // Calculate function values + unsigned int index = 0; + float minFunctionValue = std::numeric_limits::max(); + float maxFunctionValue = std::numeric_limits::min(); + + for (float y = yInterval.x; y <= yInterval.y; y += resolution) + { + sliceLength = 0; + + for (float x = xInterval.x; x <= xInterval.y; x += resolution) + { + if (std::abs(x) < 0.05f && std::abs(y) < 0.05f) + volatile int sjdks = 3; + + float val = func(x, y); + std::optional pointIndex; + + // If function value is in bbox + if (val >= zInterval.x && val <= zInterval.y) + { + vertices.push_back(Map(xInterval, xBounds, x)); + vertices.push_back(val); // Will be corrected later! + vertices.push_back(Map(yInterval, yBounds, y)); + vertices.push_back(0.0f); + + minFunctionValue = std::min(minFunctionValue, val); + maxFunctionValue = std::max(maxFunctionValue, val); + + pointIndex = index; + index++; + } + + functionValues.push_back(pointIndex); + sliceLength++; + } + + slices++; + } + + zInterval = glm::vec2(minFunctionValue, maxFunctionValue); + glm::vec2 zBounds = glm::vec2(-0.5f, 0.5f) * ((zInterval.y - zInterval.x) * scale); + + for (unsigned int i = 3; i < vertices.size(); i += 4) + { + vertices[i] = Map(zInterval, glm::vec2(0.0f, 1.0f), vertices[i - 2]); + vertices[i - 2] = Map(zInterval, zBounds, vertices[i - 2]); + } + + /* + * This is what the desired meshing should look like + * +y + * CurrentSlice o---o---o---o---o---o---o---o x o ^ + * | / \ | / | / | / \ | \ / | | + * PreviousSlice o x o---o---o x x o---o---o +---> +x + * + * + * Possible fragment constellations + * +---------+---------+---------+---------+---------+---------+ + * | 2---1 | o---o | o---o | x o | o---o | o x | + * | | | | | / | | | / | / | | \ | | | \ | + * | 3---4 | o---o | o x | o---o | x o | o---o | + * +---------+---------+---------+---------+---------+---------+ + * + * 123, 134, 124, 234 + * + * o = Function value inside bbox + * x = Function value outside bbox + * + * By default OpenGL treats triangles defined in couter-clockwise order as "front facing". So the algorithm + * should construct the triangles in that way (even if we're not culling faces, just to not have any funny bugs in the future) + * + * The algorithm works as follows: + * 1. Pick a patch of 4 points from the value list, and label them like in the diagram above + * 2. Attempt to connect the 123 constellation. + * 3. Attempt to connect the next constellation, until none are left + * -> Success: Go to 1 + * -> Failure: Go to 3 + */ + + struct Constellation { + unsigned int i, j, k; + }; + + std::array constellations = { + Constellation {0, 1, 2}, + Constellation {0, 2, 3}, + Constellation {0, 1, 3}, + Constellation {1, 2, 3} + }; + std::array*, 4> points; + + for (unsigned int y = 1; y < slices; y++) + { + for (unsigned int x = 1; x < sliceLength; x++) + { + points = { + &functionValues[y * sliceLength + x], + &functionValues[y * sliceLength + (x - 1)], + &functionValues[(y - 1) * sliceLength + (x - 1)], + &functionValues[(y - 1) * sliceLength + x], + }; + + int matches = 0; + for (Constellation& constellation : constellations) + { + // This constellation doesnt match + if (!(points[constellation.i]->has_value() && points[constellation.j]->has_value() && points[constellation.k]->has_value())) + continue; + + indices.push_back(points[constellation.i]->value()); + indices.push_back(points[constellation.j]->value()); + indices.push_back(points[constellation.k]->value()); + + matches++; + if (matches == 2) + break; + } + } + } + + vao = VAOFactory::Produce(vertices, indices, + { + { 3, GL_FLOAT, GL_FALSE, sizeof(float) * 4, (void*)0 }, + { 1, GL_FLOAT, GL_FALSE, sizeof(float) * 4, (void*)(3 * sizeof(float))} + } + ); + + shader = ShaderManager::GetInstance().Get(PLOT3D_ID); + if (shader == nullptr) + { + shader = ShaderFactory::Produce( + R"( + #version 440 core + + layout (location = 0) in vec3 aPosition; + layout (location = 1) in float aU; + + out float u; + + uniform mat4 model; + uniform mat4 view; + uniform mat4 projection; + + void main() + { + u = aU; + gl_Position = projection * view * model * vec4(aPosition, 1.0f); + } + )", + R"( + #version 440 core + + in float u; + + out vec4 FragColor; + + void main() + { + FragColor = vec4(1.0f - u, 0.0f, u, 1.0f); + } + )" + ); + + ShaderManager::GetInstance().Register(PLOT3D_ID, shader); + } +} + +void Plot3D::PreRender(const CameraBase& camera) const +{ + shader->SetUniform("model", transformation); + shader->SetUniform("view", camera.GetView()); + shader->SetUniform("projection", camera.GetProjection()); +} diff --git a/src/Plot3D.hpp b/src/Plot3D.hpp new file mode 100644 index 0000000..979a6a3 --- /dev/null +++ b/src/Plot3D.hpp @@ -0,0 +1,22 @@ +#pragma once + +#include + +#include "backend/Drawable.hpp" +#include "backend/Transformable.hpp" +#include "backend/BoundingBox.hpp" + +typedef std::function PlottableFunction; + +class Plot3D : + public Transformable, public Drawable +{ +public: + Plot3D(const BBox& domainAndRange, float scale, float resolution, PlottableFunction func); + +private: + void PreRender(const CameraBase& camera) const override; + +private: + glm::vec2 range; +}; \ No newline at end of file diff --git a/src/Util.hpp b/src/Util.hpp index aece120..bb48cbd 100644 --- a/src/Util.hpp +++ b/src/Util.hpp @@ -1,4 +1,3 @@ #pragma once -#define CUBOID_ID 0x1 -#define GRID_ID 0x2 \ No newline at end of file +#define PLOT3D_ID 0x1 \ No newline at end of file