randomwalk/main.cpp

154 lines
4.1 KiB
C++
Raw Normal View History

2024-08-03 02:42:49 +02:00
#include <SDL2/SDL.h>
#include <chrono>
2024-08-03 02:42:49 +02:00
#include <memory>
#include <print>
2024-08-03 03:23:16 +02:00
#include <random>
2024-08-03 02:42:49 +02:00
namespace {
template <typename SDLT, auto deletef>
struct sdl_wrapper_t : std::unique_ptr<SDLT, decltype(deletef)> {
explicit sdl_wrapper_t(SDLT* p)
: std::unique_ptr<SDLT, decltype(deletef)>{p, deletef} {}
2024-08-03 02:42:49 +02:00
operator SDLT*() { return std::unique_ptr<SDLT, decltype(deletef)>::get(); }
};
2024-08-03 03:23:16 +02:00
using window_t = sdl_wrapper_t<SDL_Window, SDL_DestroyWindow>;
using surface_t = sdl_wrapper_t<SDL_Surface, SDL_FreeSurface>;
using renderer_t = sdl_wrapper_t<SDL_Renderer, SDL_DestroyRenderer>;
2024-08-03 22:06:57 +02:00
auto operator+(SDL_Point a, SDL_Point b) {
return SDL_Point{.x = a.x + b.x, .y = a.y + b.y};
}
auto operator<=>(const SDL_Point& a, const SDL_Point& b) {
auto y_comp = a.x <=> b.x;
if (y_comp != std::strong_ordering::equal) { return y_comp; }
return a.y <=> b.y;
}
2024-08-03 03:23:16 +02:00
2024-08-03 22:06:57 +02:00
auto operator==(const SDL_Point& a, const SDL_Point& b) {
return a.x == b.x && a.y == b.y;
2024-08-03 03:23:16 +02:00
}
// constexpr int width = 1000;
// constexpr int height = 1000;
constexpr int width = 3440;
constexpr int height = 1300;
2024-08-03 02:42:49 +02:00
2024-08-03 22:40:14 +02:00
template <auto EF>
struct [[nodiscard("give this a name so SDL_Quit is called at the end"
)]] Defer {
Defer() = default;
Defer(const Defer&) = delete;
Defer(Defer&&) = delete;
auto operator=(Defer&&) = delete;
auto operator=(const Defer&) = delete;
~Defer() { EF(); }
};
using clock = std::chrono::steady_clock;
auto poll_events(bool& continu) {
for (SDL_Event e; SDL_PollEvent(&e);) {
if (e.type == SDL_QUIT) { continu = false; }
}
}
} // namespace
2024-08-03 02:42:49 +02:00
int main() {
if (SDL_Init(SDL_INIT_VIDEO) < 0) {
std::println("fuck");
return EXIT_FAILURE;
}
2024-08-03 22:40:14 +02:00
Defer<SDL_Quit> defer_SDL_Quit;
window_t window{SDL_CreateWindow(
"random walk", SDL_WINDOWPOS_UNDEFINED, SDL_WINDOWPOS_UNDEFINED, width,
height, 0
2024-08-03 22:40:14 +02:00
)};
if (!window) {
std::println(
"Window could not be created! SDL_Error: {}", SDL_GetError()
);
return EXIT_FAILURE;
}
2024-08-03 02:42:49 +02:00
2024-08-03 22:40:14 +02:00
// owned by window
auto* window_surface = SDL_GetWindowSurface(window);
if (!window_surface) { return EXIT_FAILURE; }
if (window_surface->format->palette) { return EXIT_FAILURE; }
2024-08-03 23:38:19 +02:00
std::println(
"got surface with {} bits per pixel",
window_surface->format->BitsPerPixel
);
if (!(window_surface->format->format & SDL_PIXELFORMAT_RGBA32)) {
return EXIT_FAILURE;
}
2024-08-03 22:40:14 +02:00
SDL_Point pos{.x = width / 2, .y = height / 2};
const SDL_Point directions[] = {
{.x = -1, .y = 0}, {.x = 1, .y = 0}, {.x = 0, .y = -1}, {.x = 0, .y = 1}
};
std::mt19937_64 rne(std::random_device{}());
std::uniform_int_distribution dist(0, 3);
2024-08-03 23:38:19 +02:00
// aarrggbb??
constexpr Uint32 bg_color = 0xffdbd7c0;
constexpr Uint32 walk_color = 0xff0d0f14;
SDL_LockSurface(window_surface);
for (int i = 0; i < window_surface->w * window_surface->h; ++i) {
static_cast<Uint32*>(window_surface->pixels)[i] = bg_color;
}
SDL_UnlockSurface(window_surface);
auto batchsize = 100.0;
constexpr auto frametime_target = std::chrono::milliseconds{10};
2024-08-03 22:40:14 +02:00
const auto start_time = clock::now();
auto next_poll_events_time = start_time;
auto frame_start = start_time - frametime_target;
clock::duration frame_time = frametime_target;
for (bool continu = true; continu;) {
{
// measure frame_time
auto now = clock::now();
frame_time = now - frame_start;
frame_start = now;
}
batchsize = batchsize * (0.9 + 0.1 * frametime_target / frame_time);
2024-08-03 23:38:19 +02:00
SDL_LockSurface(window_surface);
for (auto step = 0z; step < int(batchsize); ++step) {
SDL_Point newpoint;
do {
newpoint = pos + directions[dist(rne)];
} while (newpoint.x < 0 or newpoint.x >= width or newpoint.y < 0 or
newpoint.y >= height);
pos = newpoint;
static_cast<Uint32*>(window_surface->pixels
)[pos.x + pos.y * window_surface->w] = walk_color;
}
2024-08-03 23:38:19 +02:00
SDL_UnlockSurface(window_surface);
SDL_UpdateWindowSurface(window);
2024-08-03 02:42:49 +02:00
const auto now = clock::now();
if (now >= next_poll_events_time) {
next_poll_events_time = now + std::chrono::milliseconds{200};
poll_events(continu);
std::println(
"last frame_time={} batchsize={}", frame_time, batchsize
);
2024-08-03 02:42:49 +02:00
}
}
// leaking memory, nothing I can do
}