#include "tile-atlas.hpp" #include "compat/assert.hpp" #include "tile-image.hpp" #include "compat/exception.hpp" #include #include #include #include namespace floormat { tile_atlas::tile_atlas(StringView path, StringView name, const ImageView2D& image, Vector2ub tile_count, Optional p) : texcoords_{make_texcoords_array(Vector2ui(image.size()), tile_count)}, path_{path}, name_{name}, size_{image.size()}, dims_{tile_count}, passability{std::move(p)} { constexpr auto variant_max = std::numeric_limits::max(); fm_soft_assert(num_tiles() <= variant_max); fm_soft_assert(dims_[0] > 0 && dims_[1] > 0); fm_soft_assert(size_ % Vector2ui{tile_count} == Vector2ui()); tex_.setLabel(path_) .setWrapping(GL::SamplerWrapping::ClampToEdge) .setMagnificationFilter(GL::SamplerFilter::Nearest) .setMinificationFilter(GL::SamplerFilter::Linear) .setMaxAnisotropy(1) .setBorderColor(Color4{1, 0, 0, 1}) .setStorage(1, GL::textureFormat(image.format()), image.size()) .setSubImage(0, {}, image); } std::array tile_atlas::texcoords_for_id(size_t i) const { fm_assert(i < num_tiles()); return texcoords_[i]; } auto tile_atlas::texcoords_at(Vector2ui pos_, Vector2ui size_, Vector2ui image_size_) -> texcoords { auto pos = Vector2(pos_), size = Vector2(size_), image_size = Vector2(image_size_); auto offset = pos + Vector2(.5f), end = offset + size - Vector2(1); auto x0 = offset / image_size, x1 = end / image_size; return {{ { x1.x(), 1.f - x1.y() }, // bottom right { x1.x(), 1.f - x0.y() }, // top right { x0.x(), 1.f - x1.y() }, // bottom left { x0.x(), 1.f - x0.y() }, // top left }}; } auto tile_atlas::make_texcoords(Vector2ui pixel_size, Vector2ub tile_count, size_t i) -> texcoords { const auto sz = pixel_size/Vector2ui{tile_count}; const auto id = Vector2ui{ uint32_t(i % tile_count[0]), uint32_t(i / tile_count[0]) }; const auto p0 = id * sz; return texcoords_at(p0, sz, pixel_size); } auto tile_atlas::make_texcoords_array(Vector2ui pixel_size, Vector2ub tile_count) -> std::unique_ptr { const size_t N = Vector2ui{tile_count}.product(); auto ptr = std::make_unique[]>(N); for (auto i = 0uz; i < N; i++) ptr[i] = make_texcoords(pixel_size, tile_count, i); return ptr; } size_t tile_atlas::num_tiles() const { return Vector2ui{dims_}.product(); } Optional tile_atlas::pass_mode() const { return passability; } enum pass_mode tile_atlas::pass_mode(enum pass_mode p) const { return passability ? *passability : p; } void tile_atlas::set_pass_mode(enum pass_mode p) { fm_assert(!passability || passability == p); passability = { InPlaceInit, p }; } } // namespace floormat