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
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
|
#pragma once
#include "enumDetails.h"
#include <glm/glm.hpp>
#include <iostream>
#include <maths.h>
#include <optional>
#include <source_location>
#include <span>
#include <sstream>
#include <tuple>
#include <type_traits>
template<typename S>
concept stringlike = requires(const S & s) { s.substr(0); };
template<typename T>
concept NonStringIterableCollection
= std::is_same_v<decltype(std::declval<T>().begin()), decltype(std::declval<T>().end())> && !stringlike<T>;
namespace std {
std::ostream &
operator<<(std::ostream & s, const NonStringIterableCollection auto & collection)
{
s << '(';
for (size_t nth {}; const auto & element : collection) {
if (nth++) {
s << ", ";
}
s << element;
}
return s << ')';
}
template<glm::length_t L, glm::length_t R, typename T, glm::qualifier Q>
std::ostream &
operator<<(std::ostream & s, const glm::mat<L, R, T, Q> & m)
{
return (s << std::span {&m[0], L});
}
template<glm::length_t L, typename T, glm::qualifier Q>
std::ostream &
operator<<(std::ostream & s, const glm::vec<L, T, Q> & v)
{
return (s << std::span {&v[0], L});
}
template<typename First, typename Second>
std::ostream &
operator<<(std::ostream & s, const std::pair<First, Second> & v)
{
return (s << '(' << v.first << ", " << v.second << ')');
}
namespace {
template<typename... T, size_t... Idx>
std::ostream &
printTuple(std::ostream & s, const std::tuple<T...> & v, std::integer_sequence<size_t, Idx...>)
{
return ((s << (Idx ? ", " : "") << std::get<Idx>(v)), ...);
}
}
template<typename... T>
std::ostream &
operator<<(std::ostream & s, const std::tuple<T...> & v)
{
return printTuple(s << '{', v, std::make_index_sequence<sizeof...(T)>()) << '}';
}
inline std::ostream &
operator<<(std::ostream & s, const Arc & arc)
{
return s << arc.first << " ↺ " << arc.second;
}
template<typename E>
concept IsEnum = std::is_enum_v<E>;
template<IsEnum E>
inline std::ostream &
operator<<(std::ostream & s, const E & e)
{
return s << EnumTypeDetails<E>::typeName << "::" << EnumDetails<E>::to_string(e).value();
}
template<typename T>
inline std::ostream &
operator<<(std::ostream & s, const std::optional<T> & v)
{
if (v) {
return s << *v;
}
return s << "nullopt";
}
}
template<typename T>
std::string
streamed_string(const T & v)
{
std::stringstream ss;
ss << v;
return std::move(ss).str();
}
namespace {
template<typename T>
void
clogImpl(const T & value, const std::string_view name,
const std::source_location loc = std::source_location::current())
{
std::cerr << loc.line() << " : " << name << " : " << value << "\n";
}
}
#define CLOG(x) clogImpl(x, #x)
|