Terminal++ 3.1.0.4
A C++ library for interacting with ANSI terminal windows
Loading...
Searching...
No Matches
glyph.hpp
1#pragma once
2
3#include "terminalpp/character_set.hpp"
4#include "terminalpp/core.hpp"
5#include "terminalpp/detail/ascii.hpp"
6
7#include <boost/container_hash/hash.hpp>
8
9#include <compare>
10#include <iosfwd>
11
12namespace terminalpp {
13
14//* =========================================================================
17//* =========================================================================
19{
20public:
21 //* =====================================================================
26 //* =====================================================================
27 constexpr glyph( // NOLINT
28 byte const character = detail::ascii::space,
29 character_set const charset = character_set()) noexcept
30 : character_(character), charset_(charset)
31 {
32 }
33
34 //* =====================================================================
36 //* =====================================================================
37 explicit constexpr glyph(byte const (&text)[2]) noexcept
38 : ucharacter_{text[0]}, charset_(terminalpp::charset::utf8)
39 {
40 }
41
42 //* =====================================================================
44 //* =====================================================================
45 explicit constexpr glyph(byte const (&text)[3]) noexcept
46 : ucharacter_{text[0], text[1]}, charset_(terminalpp::charset::utf8)
47 {
48 }
49
50 //* =====================================================================
52 //* =====================================================================
53 explicit constexpr glyph(byte const (&text)[4]) noexcept
54 : ucharacter_{text[0], text[1], text[2]},
55 charset_(terminalpp::charset::utf8)
56 {
57 }
58
59 //* =====================================================================
61 //* =====================================================================
62 explicit constexpr glyph(char8_t const (&text)[2]) noexcept
63 : ucharacter_{text[0]}, charset_(terminalpp::charset::utf8)
64 {
65 }
66
67 //* =====================================================================
69 //* =====================================================================
70 explicit constexpr glyph(char8_t const (&text)[3]) noexcept
71 : ucharacter_{text[0], text[1]}, charset_(terminalpp::charset::utf8)
72 {
73 }
74
75 //* =====================================================================
77 //* =====================================================================
78 explicit constexpr glyph(char8_t const (&text)[4]) noexcept
79 : ucharacter_{text[0], text[1], text[2]},
80 charset_(terminalpp::charset::utf8)
81 {
82 }
83
84 //* =====================================================================
94 //* =====================================================================
95 template <class T = void> // This makes matching these parameters "worse"
96 // than any of the array matches above, and so
97 // avoids ambiguity.
98 explicit constexpr glyph(char const *ustr) noexcept
99 : ucharacter_{0}, charset_(terminalpp::charset::utf8)
100 {
101 for (size_t index = 0; index < sizeof(ucharacter_); ++index)
102 {
103 ucharacter_[index] = static_cast<byte>(ustr[index]);
104
105 if (!(ucharacter_[index] & 0x80))
106 {
107 break;
108 }
109 }
110 }
111
112 //* =====================================================================
114 //* =====================================================================
116 [[nodiscard]] constexpr friend auto operator<=>(
117 glyph const &lhs, glyph const &rhs) noexcept
118 {
119 if (lhs < rhs) return std::strong_ordering::less; // NOLINT
120 if (rhs < lhs) return std::strong_ordering::greater; // NOLINT
121 return std::strong_ordering::equal;
122 }
123
124 //* =====================================================================
126 //* =====================================================================
128 [[nodiscard]] constexpr friend bool operator==(
129 glyph const &lhs, glyph const &rhs) noexcept
130 {
131 if (lhs.charset_ == rhs.charset_)
132 {
133 if (lhs.charset_ == terminalpp::charset::utf8)
134 {
135 using std::begin;
136 using std::end;
137
138 // Re-implementing std::equal here for constexprness.
139 for (auto lch = begin(lhs.ucharacter_),
140 rch = begin(rhs.ucharacter_);
141 lch != end(lhs.ucharacter_);
142 ++lch, ++rch)
143 {
144 if (*lch != *rch)
145 {
146 return false;
147 }
148 }
149
150 return true;
151 }
152 else
153 {
154 return lhs.character_ == rhs.character_;
155 }
156 }
157
158 return false;
159 }
160
161 //* =====================================================================
163 //* =====================================================================
164 [[nodiscard]] constexpr friend bool operator<(
165 glyph const &lhs, glyph const &rhs) noexcept
166 {
167 if (lhs.charset_ < rhs.charset_)
168 {
169 return true;
170 }
171
172 if (lhs.charset_ == rhs.charset_)
173 {
174 if (lhs.charset_ == terminalpp::charset::utf8)
175 {
176 using std::begin;
177 using std::end;
178
179 // Reimplementing lexicographical_compare here for
180 // constexprness.
181 for (auto begin1 = begin(lhs.ucharacter_),
182 end1 = end(lhs.ucharacter_),
183 begin2 = begin(rhs.ucharacter_);
184 begin1 != end1;
185 ++begin1, ++begin2)
186 {
187 // We are expecting unicode characters to be ordered so that
188 // outside of the ASCII range is greater, so therefore we
189 // must remove the sign from these comparisons.
190 if (byte(*begin1) < byte(*begin2)) return true; // NOLINT
191 if (byte(*begin2) < byte(*begin1)) return false; // NOLINT
192 }
193
194 return false;
195 }
196
197 return lhs.character_ < rhs.character_;
198 }
199
200 return false;
201 }
202
203 //* =====================================================================
205 //* =====================================================================
206 [[nodiscard]] friend std::size_t hash_value(glyph const &gly) noexcept
207 {
208 std::size_t seed = 0;
209 boost::hash_combine(seed, gly.charset_);
210
211 if (gly.charset_ == terminalpp::charset::utf8)
212 {
213 for (auto ch : gly.ucharacter_)
214 {
215 boost::hash_combine(seed, ch);
216 }
217 }
218 else
219 {
220 boost::hash_combine(seed, gly.character_);
221 }
222
223 return seed;
224 }
225
226 union
227 {
228 byte character_;
229 byte ucharacter_[3];
230 };
231
232 character_set charset_;
233};
234
235//* =========================================================================
237//* =========================================================================
238TERMINALPP_EXPORT
239bool is_printable(glyph const &gly) noexcept;
240
241//* =========================================================================
244//* =========================================================================
245TERMINALPP_EXPORT
246std::ostream &operator<<(std::ostream &out, glyph const &gly);
247
248} // namespace terminalpp
249
250namespace std {
251
252template <>
253struct hash<terminalpp::glyph>
254{
256 using result_type = std::size_t;
257
258 [[nodiscard]] result_type operator()(
259 argument_type const &elem) const noexcept
260 {
261 return hash_value(elem);
262 }
263};
264
265} // namespace std
A structure that represents a character set.
Definition character_set.hpp:44
A structure representing an ANSI graphics effect (e.g. intensity, underlining)
Definition effect.hpp:27
constexpr effect() noexcept
Initialises the intensity to the default (normal) value.
Definition effect.hpp:31
A structure that carries around the character attributes of an ANSI element.
Definition glyph.hpp:19
friend std::size_t hash_value(glyph const &gly) noexcept
Hash function.
Definition glyph.hpp:206
constexpr glyph(byte const (&text)[4]) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:53
constexpr glyph(char8_t const (&text)[4]) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:78
constexpr glyph(char8_t const (&text)[2]) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:62
constexpr friend bool operator<(glyph const &lhs, glyph const &rhs) noexcept
Less-than operator.
Definition glyph.hpp:164
constexpr glyph(byte const (&text)[2]) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:37
constexpr glyph(char8_t const (&text)[3]) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:70
constexpr glyph(char const *ustr) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:98
constexpr glyph(byte const character=detail::ascii::space, character_set const charset=character_set()) noexcept
Default Constructor.
Definition glyph.hpp:27
constexpr glyph(byte const (&text)[3]) noexcept
Constructs a UTF-8 glyph from a char sequence.
Definition glyph.hpp:45
TERMINALPP_EXPORT constexpr friend bool operator==(glyph const &lhs, glyph const &rhs) noexcept
Equality operator.
Definition glyph.hpp:128
TERMINALPP_EXPORT constexpr friend auto operator<=>(glyph const &lhs, glyph const &rhs) noexcept
Relational operators for glyphs.
Definition glyph.hpp:116