snake_2024/include/Map.hpp

113 lines
2.6 KiB
C++
Raw Normal View History

#pragma once
#include <deque>
#include <optional>
2024-07-22 21:22:52 +03:00
#include "Canvas.hpp"
#include "Utils.hpp"
#include "Vec.hpp"
class Map : public GameObject {
struct Food {
size_t gen;
Veci pos;
int weight;
bool eaten = false;
};
public:
2024-07-22 21:22:52 +03:00
struct Config {
double gen_interval;
size_t food_exists_gens;
size_t gen_food_count;
Veci size;
int min_food_weight;
int max_food_weight;
Color food_color;
};
Map(Config config) : config_(config) {}
void act(float dt) override {
time_from_last_gen_ += dt;
2024-07-22 21:22:52 +03:00
if (time_from_last_gen_ > config_.gen_interval) {
time_from_last_gen_ -= config_.gen_interval;
generate();
}
}
int eat(Veci pos, int64_t dist) {
size_t eaten_weight = 0;
for (auto &food : food_) {
if ((pos - food.pos).len_sq() < dist * dist and not food.eaten) {
food.eaten = true;
eaten_weight += food.weight;
}
}
return eaten_weight;
}
void draw(Veci offset) const override {
2024-07-22 21:22:52 +03:00
for (const auto &food : food_) {
2024-07-25 20:51:31 +03:00
Veci food_pos = utils::to_world_coord(food.pos - offset);
if (utils::is_on_screen(food_pos) and not food.eaten) {
paint::circle({{
.pos = food_pos,
.color = color::scale(config_.food_color,
1.0 - food_lasted(food)),
},
food.weight * 2});
2024-07-22 21:22:52 +03:00
}
}
}
std::optional<Veci> find_nearest_food(Veci pos) {
if (food_.empty()) {
return std::nullopt;
}
Veci target = food_.front().pos;
for (const auto &food : food_) {
if ((pos - food.pos).len_sq() < (pos - target).len_sq()) {
target = food.pos;
}
}
return target;
}
private:
void generate() {
++current_gen_;
while (food_.front().gen + config_.food_exists_gens <= current_gen_) {
food_.pop_front();
}
2024-07-22 21:22:52 +03:00
for (size_t i = 0; i < config_.gen_food_count; ++i) {
food_.push_back({
.gen = current_gen_,
.pos =
{
.x = utils::rand_to(config_.size.x),
.y = utils::rand_to(config_.size.y),
},
.weight = utils::rand_in_range(config_.min_food_weight,
config_.max_food_weight),
2024-07-22 21:22:52 +03:00
});
}
}
double food_lasted(const Food &food) const {
return ((current_gen_ - food.gen) * config_.gen_interval +
time_from_last_gen_) /
(config_.food_exists_gens * config_.gen_interval);
}
private:
2024-07-25 22:42:54 +03:00
const Config config_;
double time_from_last_gen_ = 0;
size_t current_gen_ = 0;
std::deque<Food> food_ = {};
};