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
|
#pragma once
#include "enumDetails.h"
#include <glm/glm.hpp>
#include <iostream>
#include <maths.h>
#include <span>
#include <sstream>
#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 & v)
{
s << '(';
for (const auto & i : v) {
if (&i != &*v.begin()) {
s << ", ";
}
s << i;
}
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 << ')');
}
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>
std::string
streamed_string(const T & v)
{
std::stringstream ss;
ss << v;
return std::move(ss).str();
}
#define CLOG(x) std::cerr << __LINE__ << " : " #x " : " << x << "\n";
|