1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
|
#pragma once
#include "global-coords.hpp"
#include <compare>
#include <type_traits>
#include <Corrade/Utility/StlForwardTuple.h>
namespace floormat { struct point; }
template<> struct std::tuple_size<floormat::point> : std::integral_constant<floormat::size_t, 2> {};
template<floormat::size_t N> struct std::tuple_element<N, floormat::point>;
template<> struct std::tuple_element<0, floormat::point> { using type = floormat::global_coords; };
template<> struct std::tuple_element<1, floormat::point> { using type = Magnum::Vector2b; };
namespace floormat {
struct point
{
constexpr point();
constexpr point(global_coords coord, Vector2b offset);
constexpr point(chunk_coords_ coord, local_coords tile, Vector2b offset);
constexpr bool operator==(const point&) const noexcept;
friend constexpr std::strong_ordering operator<=>(const point& a, const point& b);
constexpr global_coords coord() const;
constexpr chunk_coords chunk() const;
constexpr chunk_coords_ chunk3() const;
constexpr local_coords local() const;
constexpr Vector2b offset() const;
template<size_t N> std::tuple_element_t<N, point> constexpr get() const;
size_t hash() const;
friend Debug& operator<<(Debug& dbg, const point& pt);
private:
int16_t cx = 0, cy = 0;
int8_t cz = 0;
local_coords tile;
Vector2b _offset;
};
constexpr point::point() = default;
constexpr point::point(global_coords coord, Vector2b offset) : point{coord.chunk3(), coord.local(), offset} {}
constexpr point::point(chunk_coords_ coord, local_coords tile, Vector2b offset) :
cx{coord.x}, cy{coord.y}, cz{coord.z}, tile{tile}, _offset{offset}
{}
constexpr bool point::operator==(const point&) const noexcept = default;
constexpr std::strong_ordering operator<=>(const point& p1, const point& p2)
{
if (auto val = p1.cz <=> p2.cz; val != std::strong_ordering::equal) return val;
if (auto val = p1.cy <=> p2.cy; val != std::strong_ordering::equal) return val;
if (auto val = p1.cx <=> p2.cx; val != std::strong_ordering::equal) return val;
if (auto val = p1.tile.y <=> p2.tile.y; val != std::strong_ordering::equal) return val;
if (auto val = p1.tile.x <=> p2.tile.x; val != std::strong_ordering::equal) return val;
if (auto val = p1._offset.y() <=> p2._offset.y(); val != std::strong_ordering::equal) return val;
if (auto val = p1._offset.x() <=> p2._offset.x(); val != std::strong_ordering::equal) return val;
return std::strong_ordering::equal;
}
constexpr global_coords point::coord() const { return {{cx, cy}, tile, cz}; }
constexpr chunk_coords_ point::chunk3() const { return {cx, cy, cz}; }
constexpr chunk_coords point::chunk() const { return {cx, cy}; }
constexpr local_coords point::local() const { return tile; }
constexpr Vector2b point::offset() const { return _offset; }
template<size_t N> std::tuple_element_t<N, point> constexpr point::get() const
{
static_assert(N < 2);
if constexpr(N == 0)
return global_coords{{cx, cy}, tile, cz};
if constexpr(N == 1)
return _offset;
return {};
}
} // namespace floormat
|