19#include <CoreFoundation/CoreFoundation.h>
43 template<std::
integral T> [[nodiscard]] std::optional<T>
stringTo(std::string_view s);
48 template<
int BASE, std::
integral T> [[nodiscard]] std::optional<T>
stringToBase(std::string_view s);
54 void trimRight(std::string& str,
const char* chars);
55 void trimRight(std::string& str,
char chars);
56 void trimRight(std::string_view& str, std::string_view chars);
57 void trimRight(std::string_view& str,
char chars);
58 void trimLeft (std::string& str,
const char* chars);
59 void trimLeft (std::string& str,
char chars);
60 void trimLeft (std::string_view& str, std::string_view chars);
61 void trimLeft (std::string_view& str,
char chars);
62 void trim (std::string_view& str, std::string_view chars);
63 void trim (std::string_view& str,
char chars);
65 [[nodiscard]] std::pair<std::string_view, std::string_view>
splitOnFirst(
66 std::string_view str, std::string_view chars);
67 [[nodiscard]] std::pair<std::string_view, std::string_view>
splitOnFirst(
68 std::string_view str,
char chars);
69 [[nodiscard]] std::pair<std::string_view, std::string_view>
splitOnLast(
70 std::string_view str, std::string_view chars);
71 [[nodiscard]] std::pair<std::string_view, std::string_view>
splitOnLast(
72 std::string_view str,
char chars);
74 unsigned min,
unsigned max);
78 [[nodiscard]]
inline auto split_view(std::string_view str,
char c) {
82 using value_type = std::string_view;
83 using difference_type = ptrdiff_t;
86 Iterator(
const Iterator&) =
default;
87 Iterator& operator=(
const Iterator&) =
default;
89 Iterator(std::string_view str_,
char c_)
94 [[nodiscard]] std::string_view
operator*()
const {
95 return {str.data(), p};
98 Iterator& operator++() {
100 str.remove_prefix(p + 1);
107 Iterator operator++(
int) {
auto copy = *
this; ++(*this);
return copy; }
109 [[nodiscard]]
bool operator==(
const Iterator&)
const =
default;
110 [[nodiscard]]
bool operator==(Sentinel)
const {
return str.empty(); }
114 while ((p < str.size()) && (str[p] != c)) ++p;
118 std::string_view str;
119 std::string_view::size_type p;
122 static_assert(std::forward_iterator<Iterator>);
123 static_assert(std::sentinel_for<Sentinel, Iterator>);
126 std::string_view str;
129 [[nodiscard]]
auto begin()
const {
return Iterator{str, c}; }
130 [[nodiscard]]
auto end()
const {
return Sentinel{}; }
133 return Splitter{str, c};
138 [[nodiscard]]
bool operator()(std::string_view s1, std::string_view s2)
const {
139 auto m =
std::min(s1.size(), s2.size());
140 int r = strncasecmp(s1.data(), s2.data(), m);
141 return (r != 0) ? (r < 0) : (s1.size() < s2.size());
145 [[nodiscard]]
bool operator()(std::string_view s1, std::string_view s2)
const {
146 if (s1.size() != s2.size())
return false;
147 return strncasecmp(s1.data(), s2.data(), s1.size()) == 0;
151#if defined(__APPLE__)
152 [[nodiscard]] std::string fromCFString(CFStringRef str);
155 template<
int BASE, std::
integral T>
160 auto e = s.data() + s.size();
161 if (
auto [p, ec] = std::from_chars(b,
e, result, BASE);
162 (ec == std::errc()) && (p ==
e)) {
168 template<std::
integral T>
169 [[nodiscard]] std::optional<T>
stringTo(std::string_view s)
171 if (s.empty()) [[unlikely]]
return {};
172 if constexpr (std::is_signed_v<T>) {
174 if (s[0] ==
'-') [[unlikely]] {
179 using U = std::make_unsigned_t<T>;
180 auto tmp = stringTo<U>(s);
181 if (!tmp) [[unlikely]]
return {};
184 if (negate) [[unlikely]] {
185 if (*tmp >
max) [[unlikely]]
return {};
188 if (*tmp >=
max) [[unlikely]]
return {};
192 if (s[0] !=
'0') [[likely]] {
193 return stringToBase<10, T>(s);
195 if (s.size() == 1)
return T(0);
196 if ((s[1] ==
'x') || (s[1] ==
'X')) [[likely]] {
198 return stringToBase<16, T>(s);
199 }
else if ((s[1] ==
'b') || (s[1] ==
'B')) {
201 return stringToBase<2, T>(s);
203 return stringToBase<10, T>(s);
void trim(string_view &str, string_view chars)
IterableBitSet< 64 > parseRange(string_view str, unsigned min, unsigned max)
bool stringToBool(string_view str)
std::optional< T > stringToBase(std::string_view s)
As above, but without dynamic base detection.
std::pair< string_view, string_view > splitOnLast(string_view str, string_view chars)
void trimRight(string &str, const char *chars)
std::optional< T > stringTo(std::string_view s)
Convert a string to an integral type 'T' (int, uint64_t, ...).
std::pair< string_view, string_view > splitOnFirst(string_view str, string_view chars)
void trimLeft(string &str, const char *chars)
auto split_view(std::string_view str, char c)
constexpr vecN< N, T > min(const vecN< N, T > &x, const vecN< N, T > &y)
constexpr vecN< N, T > max(const vecN< N, T > &x, const vecN< N, T > &y)
bool operator==(const Event &x, const Event &y)
auto copy(InputRange &&range, OutputIter out)
bool operator()(std::string_view s1, std::string_view s2) const
bool operator()(std::string_view s1, std::string_view s2) const
constexpr uint128 operator*(const uint128 &a, const uint128 &b)
constexpr auto begin(const zstring_view &x)
constexpr auto end(const zstring_view &x)