From 7570369ea49abd503e16f5124c84d039550b9f7d Mon Sep 17 00:00:00 2001 From: Robert Date: Tue, 22 Sep 2020 21:56:44 +0200 Subject: [PATCH] Initial commit --- .gitignore | 4 ++ CMakeLists.txt | 16 ++++++ README.md | 7 ++- src/BezierCurve.cpp | 92 +++++++++++++++++++++++++++++++ src/BezierCurve.hpp | 39 ++++++++++++++ src/CMakeLists.txt | 31 +++++++++++ src/UI/IDrawable.hpp | 12 +++++ src/UI/Line.cpp | 14 +++++ src/UI/Line.hpp | 19 +++++++ src/UI/Vertex.cpp | 27 ++++++++++ src/UI/Vertex.hpp | 26 +++++++++ src/main.cpp | 126 +++++++++++++++++++++++++++++++++++++++++++ 12 files changed, 412 insertions(+), 1 deletion(-) create mode 100644 .gitignore create mode 100644 CMakeLists.txt create mode 100644 src/BezierCurve.cpp create mode 100644 src/BezierCurve.hpp create mode 100644 src/CMakeLists.txt create mode 100644 src/UI/IDrawable.hpp create mode 100644 src/UI/Line.cpp create mode 100644 src/UI/Line.hpp create mode 100644 src/UI/Vertex.cpp create mode 100644 src/UI/Vertex.hpp create mode 100644 src/main.cpp diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..32dcdee --- /dev/null +++ b/.gitignore @@ -0,0 +1,4 @@ +.vs +out + +*.json \ No newline at end of file diff --git a/CMakeLists.txt b/CMakeLists.txt new file mode 100644 index 0000000..d4402d9 --- /dev/null +++ b/CMakeLists.txt @@ -0,0 +1,16 @@ +# CMakeList.txt : Top-level CMake project file, do global configuration +# and include sub-projects here. +# +cmake_minimum_required (VERSION 3.8) + +project ("Bezier Curves") + +set(CMAKE_CXX_STANDARD 17) + +find_package(SDL2 REQUIRED) +if(NOT ${SDL2_FOUND}) + message(FATAL_ERROR "Could not locate SDL2. Aborting.") +endif() + +# Include sub-projects. +add_subdirectory ("src") diff --git a/README.md b/README.md index 2de52fb..705f4f0 100644 --- a/README.md +++ b/README.md @@ -1 +1,6 @@ -# BezierCurves \ No newline at end of file +# Bezier Curves + +I was trying to understand how Bézier Curves are created/calculated, +so I created this program to draw them. + +Graphics done in SDL2 \ No newline at end of file diff --git a/src/BezierCurve.cpp b/src/BezierCurve.cpp new file mode 100644 index 0000000..9c9454d --- /dev/null +++ b/src/BezierCurve.cpp @@ -0,0 +1,92 @@ +#include "BezierCurve.hpp" + +#include +#include + +#include + +Uint32 BinomCoeff(int n, int k) +{ + return (tgamma(n+1) / (tgamma(k+1) * tgammal(n-k+1))); +} + +BezierCurve::BezierCurve() +{ + vertices.fill(nullptr); + lines.fill(nullptr); +} + +void BezierCurve::AddVertex(int x, int y) +{ + vertices[numVertices++] = new Vertex(x, y); + if (numVertices > 1) + lines[numLines++] = new Line(vertices[numVertices - 2], vertices[numVertices - 1]); +} + +void BezierCurve::Confirm() +{ + if (numLines > 1) + { + lines[numLines] = new Line(vertices[numLines], vertices[0]); + numLines++; + } + + inProgress = false; + ConstructBezier(0.05); +} + +void BezierCurve::SetPreviewVertex(int x, int y) +{ + if (vertices[numVertices] == nullptr) + vertices[numVertices] = new Vertex(x, y, Vertex::Cursor); + else + vertices[numVertices]->SetPos(x, y); +} + +void BezierCurve::ConstructBezier(float resolution) +{ + int grade = numVertices - 1; + if (vertices[grade] != nullptr && inProgress) + grade++; + + if (grade < 1) + return; + + bezier.clear(); + + std::vector binoms; + for (int i = 0; i <= grade; i++) + binoms.push_back(BinomCoeff(grade, i)); + + SDL_FPoint vert; + + for (float t = 0; t <= 1.f; t += resolution) + { + vert = { 0, 0 }; + for (int i = 0; i <= grade; i++) + { + float f = binoms[i] * pow(t, i) * pow(1 - t, grade - i); + vert.x += f * vertices[i]->x; + vert.y += f * vertices[i]->y; + } + bezier.push_back(vert); + } + + vert = { (float)vertices[grade]->x, (float)vertices[grade]->y }; + bezier.push_back(vert); +} + +void BezierCurve::Draw(SDL_Renderer* renderer) +{ + for (int i = 0; i < numVertices; i++) + vertices[i]->Draw(renderer); + + for (int i = 0; i < numLines; i++) + lines[i]->Draw(renderer); + + if (bezier.size() != 0) + { + SDL_SetRenderDrawColor(renderer, Color[0], Color[1], Color[2], Color[3]); + SDL_RenderDrawLinesF(renderer, &bezier[0], bezier.size()); + } +} diff --git a/src/BezierCurve.hpp b/src/BezierCurve.hpp new file mode 100644 index 0000000..42241f0 --- /dev/null +++ b/src/BezierCurve.hpp @@ -0,0 +1,39 @@ +#pragma once + +#include +#include + +#include "UI/Vertex.hpp" +#include "UI/Line.hpp" + +struct SDL_Renderer; +struct SDL_FPoint; + +class BezierCurve : public IDrawable +{ +public: + inline static unsigned char Color[4] = { 20, 230, 20, 255 }; +public: + BezierCurve(); + + void AddVertex(int x, int y); + void Confirm(); + + void SetPreviewVertex(int x, int y); + + uint8_t GetVertexCount() const { return numVertices; } + + void ConstructBezier(float resolution); + + void Draw(SDL_Renderer* renderer) override; + +private: + uint8_t numVertices = 0; + uint8_t numLines = 0; + std::array vertices; + std::array lines; + + std::vector bezier; + + bool inProgress = true; +}; \ No newline at end of file diff --git a/src/CMakeLists.txt b/src/CMakeLists.txt new file mode 100644 index 0000000..8d1ee94 --- /dev/null +++ b/src/CMakeLists.txt @@ -0,0 +1,31 @@ + +add_executable(beziercurves + "main.cpp" + "UI/IDrawable.hpp" "UI/Vertex.hpp" "UI/Vertex.cpp" "BezierCurve.hpp" "BezierCurve.cpp" "UI/Line.hpp" "UI/Line.cpp") + +if(WIN32) +target_include_directories(beziercurves PRIVATE + SDL2::SDL2 +) +else() +target_include_directories(beziercurves PRIVATE + ${SDL2_INCLUDE_DIRS} +) +endif() + + +if(WIN32) +target_link_libraries(beziercurves PRIVATE + SDL2::SDL2 +) +else() +target_link_libraries(beziercurves PRIVATE + ${SDL2_LIBRARIES} m +) +endif() + +if(WIN32) + add_custom_command(TARGET beziercurves POST_BUILD + COMMAND ${CMAKE_COMMAND} -E copy_directory ${SDL2_DIR}/../bin/ $ + ) +endif() \ No newline at end of file diff --git a/src/UI/IDrawable.hpp b/src/UI/IDrawable.hpp new file mode 100644 index 0000000..222afca --- /dev/null +++ b/src/UI/IDrawable.hpp @@ -0,0 +1,12 @@ +#pragma once + +struct SDL_Renderer; + +class IDrawable +{ +public: + virtual void Draw(SDL_Renderer* renderer) = 0; + +protected: + IDrawable() = default; +}; \ No newline at end of file diff --git a/src/UI/Line.cpp b/src/UI/Line.cpp new file mode 100644 index 0000000..0e27115 --- /dev/null +++ b/src/UI/Line.cpp @@ -0,0 +1,14 @@ +#include "Line.hpp" + +#include + +Line::Line(const Vertex* a, const Vertex* b) : + a(a), b(b) +{ +} + +void Line::Draw(SDL_Renderer* renderer) +{ + SDL_SetRenderDrawColor(renderer, Color[0], Color[1], Color[2], Color[3]); + SDL_RenderDrawLine(renderer, a->x, a->y, b->x, b->y); +} diff --git a/src/UI/Line.hpp b/src/UI/Line.hpp new file mode 100644 index 0000000..fe6a324 --- /dev/null +++ b/src/UI/Line.hpp @@ -0,0 +1,19 @@ +#pragma once + +#include "IDrawable.hpp" +#include "Vertex.hpp" + +class Line : public IDrawable +{ +public: + inline static unsigned char Color[4] = { 200, 200, 200, 255 }; + +public: + Line(const Vertex* a, const Vertex* b); + + void Draw(SDL_Renderer* renderer) override; + +private: + const Vertex* a; + const Vertex* b; +}; \ No newline at end of file diff --git a/src/UI/Vertex.cpp b/src/UI/Vertex.cpp new file mode 100644 index 0000000..d681e38 --- /dev/null +++ b/src/UI/Vertex.cpp @@ -0,0 +1,27 @@ +#include "Vertex.hpp" + +#include + +Vertex::Vertex(unsigned int x, unsigned int y, unsigned char* color) : + rect(new SDL_Rect), + col(color) +{ + SetPos(x, y); +} + +void Vertex::Draw(SDL_Renderer* renderer) +{ + SDL_SetRenderDrawColor(renderer, col[0], col[1], col[2], col[3]); + SDL_RenderFillRect(renderer, rect); +} + +void Vertex::SetPos(int x, int y) +{ + this->x = x; + this->y = y; + + rect->x = x - Size / 2; + rect->y = y - Size / 2; + rect->w = Size; + rect->h = Size; +} diff --git a/src/UI/Vertex.hpp b/src/UI/Vertex.hpp new file mode 100644 index 0000000..618e6f1 --- /dev/null +++ b/src/UI/Vertex.hpp @@ -0,0 +1,26 @@ +#pragma once + +#include "IDrawable.hpp" + +struct SDL_Rect; + +class Vertex : public IDrawable +{ +public: + inline static unsigned char Color[4] = { 230, 20, 20, 255 }; + inline static unsigned char Cursor[4] = { 230, 20, 230, 255 }; + inline static unsigned int Size = 10; + +public: + Vertex(unsigned int x, unsigned int y, unsigned char* color = Color); + + void Draw(SDL_Renderer* renderer) override; + void SetPos(int x, int y); + + unsigned int x, y; + +private: + SDL_Rect* rect; + + unsigned char* col; +}; \ No newline at end of file diff --git a/src/main.cpp b/src/main.cpp new file mode 100644 index 0000000..911ba88 --- /dev/null +++ b/src/main.cpp @@ -0,0 +1,126 @@ +#include +#include + +#include + +#include "BezierCurve.hpp" + +#undef main + +std::vector curves; + +void AddVertex(int x, int y, BezierCurve** curve); + +int main(int argc, char** argv) +{ + SDL_Init(SDL_INIT_VIDEO); + + SDL_Window* window = SDL_CreateWindow("Bezier Curves", + SDL_WINDOWPOS_CENTERED, SDL_WINDOWPOS_CENTERED, + 800, 800, + SDL_WINDOW_SHOWN); + + if (window == nullptr) + { + std::cerr << "Could not create window" << std::endl; + return 1; + } + + SDL_Renderer* renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_SOFTWARE); + if (renderer == nullptr) + { + std::cerr << "Could not create renderer" << std::endl; + return 1; + } + + SDL_Event e; + int mouseX = 0, mouseY = 0; + SDL_GetMouseState(&mouseX, &mouseY); + + BezierCurve* currentCurve = nullptr; + + Vertex cursor(mouseX, mouseY, Vertex::Cursor); + + bool isOpen = true; + while (isOpen) + { + while (SDL_PollEvent(&e)) + { + if(e.type == SDL_WINDOWEVENT) + { + switch (e.window.event) + { + case SDL_WINDOWEVENT_CLOSE: + isOpen = false; + break; + } + } + + switch (e.type) + { + case SDL_MOUSEMOTION: + SDL_GetMouseState(&mouseX, &mouseY); + cursor.SetPos(mouseX, mouseY); + if (currentCurve != nullptr) + { + currentCurve->SetPreviewVertex(mouseX, mouseY); + currentCurve->ConstructBezier(0.05); + } + break; + + case SDL_MOUSEBUTTONDOWN: + if (e.button.button == SDL_BUTTON_LEFT) + AddVertex(mouseX, mouseY, ¤tCurve); + if (e.button.button == SDL_BUTTON_RIGHT) + { + if (currentCurve == nullptr) + break; + + if (currentCurve->GetVertexCount() > 1) + { + currentCurve->Confirm(); + curves.push_back(currentCurve); + } + + currentCurve = nullptr; + } + break; + } + } + + SDL_SetRenderDrawColor(renderer, 0, 0, 0, 0); + SDL_RenderClear(renderer); + + for (BezierCurve* curve : curves) + curve->Draw(renderer); + + if(currentCurve != nullptr) + currentCurve->Draw(renderer); + + cursor.Draw(renderer); + + SDL_RenderPresent(renderer); + } + + SDL_DestroyRenderer(renderer); + SDL_DestroyWindow(window); + + SDL_Quit(); + + return 0; +} + +void AddVertex(int x, int y, BezierCurve** curve) +{ + if (*curve == nullptr) + (*curve) = new BezierCurve; + + (*curve)->AddVertex(x, y); + + if ((*curve)->GetVertexCount() == 4) + { + (*curve)->Confirm(); + curves.push_back(*curve); + *curve = nullptr; + } +} \ No newline at end of file