outsourced application handling
This commit is contained in:
parent
5762d02b48
commit
71fdb2c053
|
@ -16,5 +16,6 @@ else (SDL2_FOUND)
|
||||||
set (SDL2_LIBRARIES SDL2-static SDL2main)
|
set (SDL2_LIBRARIES SDL2-static SDL2main)
|
||||||
endif (SDL2_FOUND)
|
endif (SDL2_FOUND)
|
||||||
|
|
||||||
# Include sub-projects.
|
# Include sub-projects
|
||||||
|
add_subdirectory ("lib")
|
||||||
add_subdirectory ("src")
|
add_subdirectory ("src")
|
||||||
|
|
11
lib/CMakeLists.txt
Normal file
11
lib/CMakeLists.txt
Normal file
|
@ -0,0 +1,11 @@
|
||||||
|
|
||||||
|
add_library(nm_utils STATIC
|
||||||
|
"Window.cpp"
|
||||||
|
)
|
||||||
|
|
||||||
|
target_include_directories(nm_utils PUBLIC ${SDL2_INCLUDE_DIRS} ${CMAKE_CURRENT_SOURCE_DIR})
|
||||||
|
target_link_libraries(nm_utils PUBLIC ${SDL2_LIBRARIES})
|
||||||
|
|
||||||
|
if(MSVC)
|
||||||
|
target_compile_definitions(nm_utils PUBLIC _CRT_SECURE_NO_WARNINGS)
|
||||||
|
endif()
|
150
lib/Window.cpp
Normal file
150
lib/Window.cpp
Normal file
|
@ -0,0 +1,150 @@
|
||||||
|
#include "Window.hpp"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
#include <SDL.h>
|
||||||
|
|
||||||
|
void Window::Launch()
|
||||||
|
{
|
||||||
|
SDL_ShowWindow(window);
|
||||||
|
|
||||||
|
while (!shouldClose)
|
||||||
|
{
|
||||||
|
HandleEvents();
|
||||||
|
Update();
|
||||||
|
Render();
|
||||||
|
}
|
||||||
|
|
||||||
|
SDL_HideWindow(window);
|
||||||
|
}
|
||||||
|
|
||||||
|
Window::Window(int width, int height, const std::string& title)
|
||||||
|
{
|
||||||
|
window = nullptr;
|
||||||
|
renderer = nullptr;
|
||||||
|
|
||||||
|
window = SDL_CreateWindow(
|
||||||
|
title.c_str(),
|
||||||
|
SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED,
|
||||||
|
width, height,
|
||||||
|
SDL_WINDOW_HIDDEN
|
||||||
|
);
|
||||||
|
if (window == nullptr)
|
||||||
|
{
|
||||||
|
char errbuf[512];
|
||||||
|
SDL_GetErrorMsg(errbuf, 512);
|
||||||
|
|
||||||
|
std::cerr << "Failed to create SDL Window: " << std::endl
|
||||||
|
<< errbuf << std::endl;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_ACCELERATED);
|
||||||
|
if (renderer == nullptr)
|
||||||
|
{
|
||||||
|
char errbuf[512];
|
||||||
|
SDL_GetErrorMsg(errbuf, 512);
|
||||||
|
|
||||||
|
std::cerr << "Failed to create SDL Renderer: " << std::endl
|
||||||
|
<< errbuf << std::endl;
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
SDL_SetRenderDrawBlendMode(renderer, SDL_BLENDMODE_BLEND);
|
||||||
|
|
||||||
|
startOfLastFrame = std::chrono::steady_clock::now();
|
||||||
|
}
|
||||||
|
|
||||||
|
Window::Window(const Window& other)
|
||||||
|
{
|
||||||
|
*this = other;
|
||||||
|
}
|
||||||
|
|
||||||
|
Window& Window::operator=(const Window& other)
|
||||||
|
{
|
||||||
|
const char* title = SDL_GetWindowTitle(other.window);
|
||||||
|
int width, height;
|
||||||
|
SDL_GetWindowSize(other.window, &width, &height);
|
||||||
|
|
||||||
|
window = SDL_CreateWindow(
|
||||||
|
title,
|
||||||
|
SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED,
|
||||||
|
width, height,
|
||||||
|
SDL_WINDOW_HIDDEN
|
||||||
|
);
|
||||||
|
|
||||||
|
if (window == nullptr)
|
||||||
|
{
|
||||||
|
char errbuf[512];
|
||||||
|
SDL_GetErrorMsg(errbuf, 512);
|
||||||
|
|
||||||
|
std::cerr << "Failed to create SDL Window: " << std::endl
|
||||||
|
<< errbuf << std::endl;
|
||||||
|
|
||||||
|
throw std::runtime_error("");
|
||||||
|
}
|
||||||
|
|
||||||
|
renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_ACCELERATED);
|
||||||
|
if (renderer == nullptr)
|
||||||
|
{
|
||||||
|
char errbuf[512];
|
||||||
|
SDL_GetErrorMsg(errbuf, 512);
|
||||||
|
|
||||||
|
std::cerr << "Failed to create SDL Renderer: " << std::endl
|
||||||
|
<< errbuf << std::endl;
|
||||||
|
|
||||||
|
throw std::runtime_error("");
|
||||||
|
}
|
||||||
|
|
||||||
|
startOfLastFrame = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
Window::~Window()
|
||||||
|
{
|
||||||
|
SDL_DestroyRenderer(renderer);
|
||||||
|
SDL_DestroyWindow(window);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Window::HandleEvents()
|
||||||
|
{
|
||||||
|
SDL_Event event;
|
||||||
|
while (SDL_PollEvent(&event))
|
||||||
|
{
|
||||||
|
switch (event.type)
|
||||||
|
{
|
||||||
|
case SDL_WINDOWEVENT:
|
||||||
|
{
|
||||||
|
switch (event.window.event)
|
||||||
|
{
|
||||||
|
case SDL_WINDOWEVENT_CLOSE:
|
||||||
|
shouldClose = true;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
} break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Window::Update()
|
||||||
|
{
|
||||||
|
double dt = std::chrono::duration_cast<std::chrono::duration<double>>(std::chrono::steady_clock::now() - startOfLastFrame).count();
|
||||||
|
startOfLastFrame = std::chrono::steady_clock::now();
|
||||||
|
|
||||||
|
OnUpdate(dt);
|
||||||
|
}
|
||||||
|
|
||||||
|
void Window::Render()
|
||||||
|
{
|
||||||
|
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255);
|
||||||
|
SDL_RenderClear(renderer);
|
||||||
|
|
||||||
|
OnRender(renderer);
|
||||||
|
|
||||||
|
if (renderer == nullptr)
|
||||||
|
throw std::runtime_error("Client window left renderer in an invalid state");
|
||||||
|
|
||||||
|
SDL_RenderPresent(renderer);
|
||||||
|
}
|
37
lib/Window.hpp
Normal file
37
lib/Window.hpp
Normal file
|
@ -0,0 +1,37 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <string>
|
||||||
|
#include <chrono>
|
||||||
|
|
||||||
|
struct SDL_Renderer;
|
||||||
|
struct SDL_Window;
|
||||||
|
|
||||||
|
class Window
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
void Launch();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
Window(int width, int height, const std::string& title);
|
||||||
|
Window(const Window& other);
|
||||||
|
Window& operator=(const Window& other);
|
||||||
|
~Window();
|
||||||
|
|
||||||
|
|
||||||
|
virtual void OnUpdate(double dt) {}
|
||||||
|
virtual void OnRender(SDL_Renderer* renderer) {}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void HandleEvents();
|
||||||
|
void Update();
|
||||||
|
void Render();
|
||||||
|
|
||||||
|
protected:
|
||||||
|
SDL_Window* window;
|
||||||
|
|
||||||
|
private:
|
||||||
|
SDL_Renderer* renderer;
|
||||||
|
|
||||||
|
bool shouldClose = false;
|
||||||
|
std::chrono::steady_clock::time_point startOfLastFrame;
|
||||||
|
};
|
|
@ -1,103 +0,0 @@
|
||||||
#include "Application.hpp"
|
|
||||||
|
|
||||||
#include <iostream>
|
|
||||||
#include <SDL.h>
|
|
||||||
|
|
||||||
Application::Application(int width, int height, const char* title)
|
|
||||||
{
|
|
||||||
window = SDL_CreateWindow(
|
|
||||||
title,
|
|
||||||
SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED,
|
|
||||||
width, height,
|
|
||||||
SDL_WINDOW_SHOWN
|
|
||||||
);
|
|
||||||
if (window == nullptr)
|
|
||||||
{
|
|
||||||
char errbuf[512];
|
|
||||||
SDL_GetErrorMsg(errbuf, 512);
|
|
||||||
|
|
||||||
std::cerr << "Failed to create SDL Window: " << std::endl
|
|
||||||
<< errbuf << std::endl;
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Let's just pretend we're the first application in this program to create a renderer :) what could go wrong
|
|
||||||
renderer = SDL_CreateRenderer(window, -1, SDL_RENDERER_ACCELERATED);
|
|
||||||
if (renderer == nullptr)
|
|
||||||
{
|
|
||||||
char errbuf[512];
|
|
||||||
SDL_GetErrorMsg(errbuf, 512);
|
|
||||||
|
|
||||||
std::cerr << "Failed to create SDL Renderer: " << std::endl
|
|
||||||
<< errbuf << std::endl;
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
SDL_SetRenderDrawBlendMode(renderer, SDL_BLENDMODE_BLEND);
|
|
||||||
|
|
||||||
field = new FluidField(60);
|
|
||||||
before = std::chrono::steady_clock::now();
|
|
||||||
}
|
|
||||||
|
|
||||||
Application::~Application()
|
|
||||||
{
|
|
||||||
delete field;
|
|
||||||
|
|
||||||
SDL_DestroyRenderer(renderer); // Let's just destroy this renderer regardless of other applications in this program :) what could go wrong
|
|
||||||
SDL_DestroyWindow(window);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Application::Launch()
|
|
||||||
{
|
|
||||||
if (renderer == nullptr || window == nullptr)
|
|
||||||
{
|
|
||||||
throw std::runtime_error("Can't launch application. Window or Renderer is in invalid state.");
|
|
||||||
}
|
|
||||||
|
|
||||||
while (!shouldClose)
|
|
||||||
{
|
|
||||||
HandleEvents();
|
|
||||||
Update();
|
|
||||||
Render();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Application::HandleEvents()
|
|
||||||
{
|
|
||||||
static SDL_Event event;
|
|
||||||
|
|
||||||
while (SDL_PollEvent(&event))
|
|
||||||
{
|
|
||||||
if (event.type == SDL_WINDOWEVENT)
|
|
||||||
{
|
|
||||||
switch (event.window.event)
|
|
||||||
{
|
|
||||||
case SDL_WINDOWEVENT_CLOSE:
|
|
||||||
shouldClose = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void Application::Update()
|
|
||||||
{
|
|
||||||
double frametime = std::chrono::duration_cast<std::chrono::duration<double>>(std::chrono::steady_clock::now() - before).count();
|
|
||||||
before = std::chrono::steady_clock::now();
|
|
||||||
|
|
||||||
field->VelocityStep(0.002, frametime);
|
|
||||||
field->DensityStep(0.0005, frametime);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Application::Render()
|
|
||||||
{
|
|
||||||
SDL_SetRenderDrawColor(renderer, 0, 0, 0, 255);
|
|
||||||
SDL_RenderClear(renderer);
|
|
||||||
|
|
||||||
|
|
||||||
field->Draw(renderer, {0, 0, 1000, 1000});
|
|
||||||
|
|
||||||
SDL_RenderPresent(renderer);
|
|
||||||
}
|
|
|
@ -1,32 +0,0 @@
|
||||||
#pragma once
|
|
||||||
|
|
||||||
#include <chrono>
|
|
||||||
#include "FluidField.hpp"
|
|
||||||
|
|
||||||
// Forward Declarations
|
|
||||||
struct SDL_Window;
|
|
||||||
struct SDL_Renderer;
|
|
||||||
|
|
||||||
// Should be a singleton but who cares
|
|
||||||
class Application
|
|
||||||
{
|
|
||||||
public:
|
|
||||||
Application(int width, int height, const char* title);
|
|
||||||
~Application();
|
|
||||||
|
|
||||||
void Launch();
|
|
||||||
|
|
||||||
private:
|
|
||||||
void HandleEvents();
|
|
||||||
void Update();
|
|
||||||
void Render();
|
|
||||||
|
|
||||||
private:
|
|
||||||
SDL_Window* window = nullptr;
|
|
||||||
SDL_Renderer* renderer = nullptr;
|
|
||||||
|
|
||||||
std::chrono::steady_clock::time_point before;
|
|
||||||
bool shouldClose = false;
|
|
||||||
|
|
||||||
FluidField* field;
|
|
||||||
};
|
|
|
@ -4,10 +4,10 @@
|
||||||
cmake_minimum_required (VERSION 3.8)
|
cmake_minimum_required (VERSION 3.8)
|
||||||
|
|
||||||
# Add source to this project's executable.
|
# Add source to this project's executable.
|
||||||
add_executable (EulerFluid "main.cpp" "Application.hpp" "Application.cpp" "VectorField.hpp" "VectorField.cpp" "FluidField.hpp" "FluidField.cpp")
|
add_executable (EulerFluid "main.cpp" "EulerFluid.hpp" "EulerFluid.cpp" "VectorField.hpp" "VectorField.cpp" "FluidField.hpp" "FluidField.cpp")
|
||||||
|
|
||||||
target_include_directories(EulerFluid PUBLIC ${SDL2_INCLUDE_DIRS})
|
target_include_directories(EulerFluid PUBLIC nm_utils)
|
||||||
target_link_libraries(EulerFluid PUBLIC ${SDL2_LIBRARIES})
|
target_link_libraries(EulerFluid PRIVATE nm_utils)
|
||||||
|
|
||||||
if(MSVC)
|
if(MSVC)
|
||||||
target_compile_definitions(EulerFluid PUBLIC _CRT_SECURE_NO_WARNINGS)
|
target_compile_definitions(EulerFluid PUBLIC _CRT_SECURE_NO_WARNINGS)
|
||||||
|
|
26
src/EulerFluid/EulerFluid.cpp
Normal file
26
src/EulerFluid/EulerFluid.cpp
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
#include "EulerFluid.hpp"
|
||||||
|
|
||||||
|
#include <iostream>
|
||||||
|
#include <SDL.h>
|
||||||
|
|
||||||
|
EulerFluid::EulerFluid(int width, int height, const char* title) :
|
||||||
|
Window::Window(width, height, title)
|
||||||
|
{
|
||||||
|
field = new FluidField(60);
|
||||||
|
}
|
||||||
|
|
||||||
|
EulerFluid::~EulerFluid()
|
||||||
|
{
|
||||||
|
delete field;
|
||||||
|
}
|
||||||
|
|
||||||
|
void EulerFluid::OnUpdate(double dt)
|
||||||
|
{
|
||||||
|
field->VelocityStep(0.002, dt);
|
||||||
|
field->DensityStep(0.0005, dt);
|
||||||
|
}
|
||||||
|
|
||||||
|
void EulerFluid::OnRender(SDL_Renderer* renderer)
|
||||||
|
{
|
||||||
|
field->Draw(renderer, {0, 0, 1000, 1000});
|
||||||
|
}
|
18
src/EulerFluid/EulerFluid.hpp
Normal file
18
src/EulerFluid/EulerFluid.hpp
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include "Window.hpp"
|
||||||
|
#include "FluidField.hpp"
|
||||||
|
|
||||||
|
class EulerFluid : public Window
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
EulerFluid(int width, int height, const char* title);
|
||||||
|
~EulerFluid();
|
||||||
|
|
||||||
|
private:
|
||||||
|
void OnUpdate(double dt) override;
|
||||||
|
void OnRender(SDL_Renderer* renderer) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
FluidField* field;
|
||||||
|
};
|
|
@ -1,8 +1,10 @@
|
||||||
#include "Application.hpp"
|
#include "EulerFluid.hpp"
|
||||||
|
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
int main(int argc, char** argv)
|
int main(int argc, char** argv)
|
||||||
{
|
{
|
||||||
Application* app = new Application(1000, 1000, "Euler Fluid Simulation");
|
EulerFluid* app = new EulerFluid(1000, 1000, "Euler Fluid Simulation");
|
||||||
app->Launch();
|
app->Launch();
|
||||||
|
|
||||||
delete app;
|
delete app;
|
||||||
|
|
Loading…
Reference in a new issue