Initial commit
This commit is contained in:
parent
b8564d1d94
commit
7570369ea4
4
.gitignore
vendored
Normal file
4
.gitignore
vendored
Normal file
|
@ -0,0 +1,4 @@
|
||||||
|
.vs
|
||||||
|
out
|
||||||
|
|
||||||
|
*.json
|
16
CMakeLists.txt
Normal file
16
CMakeLists.txt
Normal file
|
@ -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")
|
|
@ -1 +1,6 @@
|
||||||
# BezierCurves
|
# 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
|
92
src/BezierCurve.cpp
Normal file
92
src/BezierCurve.cpp
Normal file
|
@ -0,0 +1,92 @@
|
||||||
|
#include "BezierCurve.hpp"
|
||||||
|
|
||||||
|
#include <math.h>
|
||||||
|
#include <iostream>
|
||||||
|
|
||||||
|
#include <SDL.h>
|
||||||
|
|
||||||
|
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<Uint32> 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());
|
||||||
|
}
|
||||||
|
}
|
39
src/BezierCurve.hpp
Normal file
39
src/BezierCurve.hpp
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#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<Vertex*, 4> vertices;
|
||||||
|
std::array<Line*, 4> lines;
|
||||||
|
|
||||||
|
std::vector<SDL_FPoint> bezier;
|
||||||
|
|
||||||
|
bool inProgress = true;
|
||||||
|
};
|
31
src/CMakeLists.txt
Normal file
31
src/CMakeLists.txt
Normal file
|
@ -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/ $<TARGET_FILE_DIR:beziercurves>
|
||||||
|
)
|
||||||
|
endif()
|
12
src/UI/IDrawable.hpp
Normal file
12
src/UI/IDrawable.hpp
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
struct SDL_Renderer;
|
||||||
|
|
||||||
|
class IDrawable
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
virtual void Draw(SDL_Renderer* renderer) = 0;
|
||||||
|
|
||||||
|
protected:
|
||||||
|
IDrawable() = default;
|
||||||
|
};
|
14
src/UI/Line.cpp
Normal file
14
src/UI/Line.cpp
Normal file
|
@ -0,0 +1,14 @@
|
||||||
|
#include "Line.hpp"
|
||||||
|
|
||||||
|
#include <SDL.h>
|
||||||
|
|
||||||
|
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);
|
||||||
|
}
|
19
src/UI/Line.hpp
Normal file
19
src/UI/Line.hpp
Normal file
|
@ -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;
|
||||||
|
};
|
27
src/UI/Vertex.cpp
Normal file
27
src/UI/Vertex.cpp
Normal file
|
@ -0,0 +1,27 @@
|
||||||
|
#include "Vertex.hpp"
|
||||||
|
|
||||||
|
#include <SDL.h>
|
||||||
|
|
||||||
|
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;
|
||||||
|
}
|
26
src/UI/Vertex.hpp
Normal file
26
src/UI/Vertex.hpp
Normal file
|
@ -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;
|
||||||
|
};
|
126
src/main.cpp
Normal file
126
src/main.cpp
Normal file
|
@ -0,0 +1,126 @@
|
||||||
|
#include <iostream>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include <SDL.h>
|
||||||
|
|
||||||
|
#include "BezierCurve.hpp"
|
||||||
|
|
||||||
|
#undef main
|
||||||
|
|
||||||
|
std::vector<BezierCurve*> 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;
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in a new issue