summaryrefslogtreecommitdiff
path: root/lib/stream_support.h
blob: f5c5e37ec3e69a8ca2905572db35230a5bd059f2 (plain)
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)