blob: 2945650bf3c27d2be7d91782d0a5052aad41ccec (
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
|
#pragma once
// Wrappers of some glib functions (why are we using glib then?) which we want, but glib.h is a bit C like
#ifdef __cplusplus
# include <cstdint>
extern "C" {
#else
# include <stdint.h>
#endif
const char * next_char(const char *);
uint32_t get_codepoint(const char *);
#ifdef __cplusplus
}
# include <string_view>
# include <algorithm>
struct utf8_string_view {
struct iter {
constexpr explicit iter(const char * p) : pos {p} { }
[[nodiscard]] auto
operator!=(const iter & other) const
{
return pos != other.pos;
}
auto &
operator++()
{
pos = next_char(pos);
return *this;
}
[[nodiscard]] auto
operator*() const
{
return get_codepoint(pos);
}
private:
const char * pos;
};
// cppcheck-suppress noExplicitConstructor; NOLINTNEXTLINE(hicpp-explicit-conversions)
template<typename... Args> constexpr utf8_string_view(Args &&... args) : str {std::forward<Args>(args)...} { }
[[nodiscard]] auto
begin() const
{
return iter {str.cbegin()};
}
[[nodiscard]] auto
end() const
{
return iter {str.cend()};
}
[[nodiscard]] size_t length() const;
private:
std::string_view str;
};
template<> struct std::iterator_traits<utf8_string_view::iter> {
using difference_type = size_t;
using value_type = uint32_t;
using pointer = void;
using reference = void;
using iterator_category = std::forward_iterator_tag;
};
[[nodiscard]] inline size_t
utf8_string_view::length() const
{
return std::distance(begin(), end());
}
#endif
|