#include "chunk.hpp" #include "compat/LooseQuadtree-impl.h" #include "src/tile-atlas.hpp" #include "src/collision.hpp" #include #include namespace floormat { struct collision_bbox final { using BB = loose_quadtree::BoundingBox; operator BB() const noexcept; std::int16_t left = 0, top = 0; std::uint16_t width = 0, height = 0; enum pass_mode pass_mode = pass_mode::pass; }; collision_bbox::operator BB() const noexcept { return { left, top, (std::int16_t)width, (std::int16_t)height }; } static constexpr Vector2s tile_start(std::size_t k) { const auto i = std::uint8_t(k); const local_coords coord{i}; return sTILE_SIZE2 * Vector2s(coord.x, coord.y) - sTILE_SIZE2/2; } void chunk::ensure_passability() noexcept { if (!_pass_modified) return; _pass_modified = false; if (!_lqt_move) _lqt_move = make_lqt(); if (!_lqt_shoot) _lqt_shoot = make_lqt(); if (!_lqt_view) _lqt_view = make_lqt(); _lqt_move->Clear(); _lqt_shoot->Clear(); _lqt_view->Clear(); std::vector bboxes; #ifndef FLOORMAT_64 _bboxes.clear(); bboxes.reserve(TILE_COUNT*4); #endif constexpr auto whole_tile = [](std::size_t k, pass_mode p) constexpr -> collision_bbox { auto start = tile_start(k); return { start[0], start[1], usTILE_SIZE2[0], usTILE_SIZE2[1], p }; }; constexpr auto wall_north = [](std::size_t k, pass_mode p) constexpr -> collision_bbox { auto start = tile_start(k) - Vector2s(0, 1); return { start[0], start[1], usTILE_SIZE2[0], 2, p }; }; constexpr auto wall_west = [](std::size_t k, pass_mode p) constexpr -> collision_bbox { auto start = tile_start(k) - Vector2s(1, 0); return { start[0], start[1], 2, usTILE_SIZE2[1], p }; }; for (std::size_t i = 0; i < TILE_COUNT; i++) { const auto tile = const_cast(*this)[i]; if (auto s = tile.scenery()) if (auto p = s.frame.passability; p != pass_mode::pass) bboxes.push_back(whole_tile(i, p)); if (auto atlas = tile.ground_atlas()) if (auto p = atlas->pass_mode(pass_mode::pass); p != pass_mode::pass) bboxes.push_back(whole_tile(i, p)); if (auto atlas = tile.wall_north_atlas()) if (auto p = atlas->pass_mode(pass_mode::blocked); p != pass_mode::pass) bboxes.push_back(wall_north(i, p)); if (auto atlas = tile.wall_west_atlas()) if (auto p = atlas->pass_mode(pass_mode::blocked); p != pass_mode::pass) bboxes.push_back(wall_west(i, p)); } #ifndef FLOORMAT_64 _bboxes.reserve(bboxes.size()); #endif for (const collision_bbox& bbox : bboxes) { #ifdef FLOORMAT_64 auto* ptr = std::bit_cast(collision_bbox::BB(bbox)); #else _bboxes.push_back(bbox); auto* ptr = &_bboxes.back(); #endif switch (bbox.pass_mode) { case pass_mode::blocked: _lqt_view->Insert(ptr); [[fallthrough]]; case pass_mode::see_through: _lqt_shoot->Insert(ptr); [[fallthrough]]; case pass_mode::shoot_through: _lqt_move->Insert(ptr); break; case pass_mode::pass: break; } } } auto chunk::query_collisions(Vector4s vec, collision type) const -> Query { const_cast(*this).ensure_passability(); loose_quadtree::BoundingBox bbox { vec[0], vec[1], vec[2], vec[3] }; return { lqt_from_collision_type(type)->QueryIntersectsRegion(bbox) }; } auto chunk::query_collisions(Vector2s position, Vector2us size, collision type) const -> Query { const_cast(*this).ensure_passability(); constexpr auto half = sTILE_SIZE2/2; const auto start = position - half; loose_quadtree::BoundingBox bbox {start[0], start[1], (Short)size[0], (Short)size[1] }; return { lqt_from_collision_type(type)->QueryIntersectsRegion(bbox) }; } auto chunk::query_collisions(local_coords p, Vector2us size, Vector2s offset, collision type) const -> Query { const_cast(*this).ensure_passability(); const auto pos = Vector2s(p.x, p.y) * sTILE_SIZE2 + offset; const auto start = pos - Vector2s(size/2); loose_quadtree::BoundingBox bbox { start[0], start[1], (Short)size[0], (Short)size[1] }; return { lqt_from_collision_type(type)->QueryIntersectsRegion(bbox) }; } auto chunk::lqt_from_collision_type(collision type) const noexcept -> lqt* { switch (type) { case collision::move: return _lqt_move.get(); case collision::shoot: return _lqt_shoot.get(); case collision::view: return _lqt_view.get(); } fm_abort("wrong collision type '%hhu'", std::uint8_t(type)); } auto chunk::make_lqt() -> std::unique_ptr { return std::make_unique(); } void chunk::cleanup_lqt() { if (_lqt_move) _lqt_move->ForceCleanup(); if (_lqt_shoot) _lqt_shoot->ForceCleanup(); if (_lqt_view) _lqt_view->ForceCleanup(); } } // namespace floormat