0.3.0
Loading...
Searching...
No Matches
StringUtil.hpp
Go to the documentation of this file.
1// This file is part of INSTINCT, the INS Toolkit for Integrated
2// Navigation Concepts and Training by the Institute of Navigation of
3// the University of Stuttgart, Germany.
4//
5// This Source Code Form is subject to the terms of the Mozilla Public
6// License, v. 2.0. If a copy of the MPL was not distributed with this
7// file, You can obtain one at https://mozilla.org/MPL/2.0/.
8
13
14#pragma once
15
16#include <algorithm>
17#include <cctype>
18#include <cstdint>
19#include <locale>
20#include <vector>
21#include <string>
22#include <string_view>
23
24namespace NAV::str
25{
28static inline void ltrim(std::string& s)
29{
30 if (!s.empty() && s[0] == '\n')
31 {
32 s.erase(0, 1);
33 }
34 s.erase(s.begin(), std::ranges::find_if(s, [](int ch) { return !std::isspace(ch); }));
35}
36
39static inline void rtrim(std::string& s)
40{
41 if (!s.empty() && s[s.length() - 1] == '\n')
42 {
43 s.erase(s.length() - 1);
44 }
45 s.erase(std::find_if(s.rbegin(), s.rend(), [](int ch) { // NOLINT(boost-use-ranges,modernize-use-ranges) // ranges::find_last_if is C++23 and not supported yet
46 return !std::isspace(ch);
47 })
48 .base(),
49 s.end());
50}
51
54static inline void trim(std::string& s)
55{
56 ltrim(s);
57 rtrim(s);
58}
59
62static inline void ltrim(std::string_view& sv)
63{
64 sv.remove_prefix(std::min(sv.find_first_not_of(' '), sv.size()));
65}
66
69static inline void rtrim(std::string_view& sv)
70{
71 sv.remove_suffix(std::min(sv.size() - sv.find_last_not_of(' ') - 1, sv.size()));
72}
73
76static inline void trim(std::string_view& sv)
77{
78 ltrim(sv);
79 rtrim(sv);
80}
81
85static inline std::string ltrim_copy(std::string s)
86{
87 ltrim(s);
88 return s;
89}
90
94static inline std::string rtrim_copy(std::string s)
95{
96 rtrim(s);
97 return s;
98}
99
103static inline std::string trim_copy(std::string s)
104{
105 trim(s);
106 return s;
107}
108
112static inline std::string_view ltrim_copy(std::string_view sv)
113{
114 ltrim(sv);
115 return sv;
116}
117
121static inline std::string_view rtrim_copy(std::string_view sv)
122{
123 rtrim(sv);
124 return sv;
125}
126
130static inline std::string_view trim_copy(std::string_view sv)
131{
132 trim(sv);
133 return sv;
134}
135
142
149static inline bool replace(std::string& str, const std::string& from, const std::string& to, CaseSensitivity cs = RespectCase)
150{
151 if (from.empty())
152 {
153 return false;
154 }
155 auto it = std::search(str.begin(), str.end(), // NOLINT(boost-use-ranges,modernize-use-ranges) // ranges::search is C++23 and not supported yet
156 from.begin(), from.end(),
157 [cs](char ch1, char ch2) { return cs == RespectCase
158 ? ch1 == ch2
159 : std::toupper(ch1) == std::toupper(ch2); });
160
161 if (it == str.end())
162 {
163 return false;
164 }
165 auto start_pos = static_cast<size_t>(it - str.begin());
166 str.replace(start_pos, from.length(), to);
167 return true;
168}
169
175static inline void replaceAll(std::string& str, const std::string& from, const std::string& to, CaseSensitivity cs)
176{
177 while (replace(str, from, to, cs)) {}
178}
179
184static inline void replaceAll(std::string& str, const std::string& from, const std::string& to)
185{
186 std::string::size_type n = 0;
187 while ((n = str.find(from, n)) != std::string::npos)
188 {
189 str.replace(n, from.size(), to);
190 n += to.size();
191 }
192}
193
200static inline std::string replaceAll_copy(std::string str, const std::string& from, const std::string& to, CaseSensitivity cs)
201{
202 replaceAll(str, from, to, cs);
203 return str;
204}
205
211static inline std::string replaceAll_copy(std::string str, const std::string& from, const std::string& to)
212{
213 replaceAll(str, from, to);
214 return str;
215}
216
221static inline std::vector<std::string> split(const std::string& str, const std::string& delimiter)
222{
223 size_t pos_start = 0;
224 size_t pos_end = 0;
225 size_t delim_len = delimiter.length();
226 std::vector<std::string> res;
227
228 while ((pos_end = str.find(delimiter, pos_start)) != std::string::npos)
229 {
230 res.push_back(str.substr(pos_start, pos_end - pos_start));
231 pos_start = pos_end + delim_len;
232 }
233 res.push_back(str.substr(pos_start));
234 return res;
235}
236
241static inline std::vector<std::string> split(const std::string& str, char delimiter)
242{
243 return split(str, std::string(1, delimiter));
244}
245
250static inline std::vector<std::string> split_wo_empty(const std::string& str, const std::string& delimiter)
251{
252 size_t pos_start = 0;
253 size_t pos_end = 0;
254 size_t delim_len = delimiter.length();
255 std::vector<std::string> res;
256
257 while ((pos_end = str.find(delimiter, pos_start)) != std::string::npos)
258 {
259 if (pos_start != pos_end)
260 {
261 res.push_back(str.substr(pos_start, pos_end - pos_start));
262 }
263 pos_start = pos_end + delim_len;
264 while (pos_start < str.size() && str.find(delimiter, pos_start) == pos_start)
265 {
266 pos_start += delim_len;
267 }
268 }
269 if (pos_start != str.size())
270 {
271 res.push_back(str.substr(pos_start));
272 }
273 return res;
274}
275
280static inline std::vector<std::string> split_wo_empty(const std::string& str, char delimiter)
281{
282 return split_wo_empty(str, std::string(1, delimiter));
283}
284
286template<typename T>
287concept StdString = std::convertible_to<T, std::string> || std::convertible_to<T, std::wstring>;
288
296template<StdString String>
297int stoi(const String& str, int default_value, std::size_t* pos = nullptr, int base = 10) noexcept
298{
299 try
300 {
301 return std::stoi(str, pos, base);
302 }
303 catch (...) // NOLINT(bugprone-empty-catch)
304 {}
305
306 return default_value;
307}
308
316template<StdString String>
317int64_t stol(const String& str, int64_t default_value, std::size_t* pos = nullptr, int base = 10) noexcept
318{
319 try
320 {
321 return std::stol(str, pos, base);
322 }
323 catch (...) // NOLINT(bugprone-empty-catch)
324 {}
325
326 return default_value;
327}
328
336template<StdString String>
337int64_t stoll(const String& str, int64_t default_value, std::size_t* pos = nullptr, int base = 10) noexcept
338{
339 try
340 {
341 return std::stoll(str, pos, base);
342 }
343 catch (...) // NOLINT(bugprone-empty-catch)
344 {}
345
346 return default_value;
347}
348
355template<StdString String>
356float stof(const String& str, float default_value, std::size_t* pos = nullptr) noexcept
357{
358 try
359 {
360 return std::stof(str, pos);
361 }
362 catch (...) // NOLINT(bugprone-empty-catch)
363 {}
364
365 return default_value;
366}
367
374template<StdString String>
375double stod(const String& str, double default_value, std::size_t* pos = nullptr) noexcept
376{
377 try
378 {
379 return std::stod(str, pos);
380 }
381 catch (...) // NOLINT(bugprone-empty-catch)
382 {}
383
384 return default_value;
385}
386
393template<StdString String>
394long double stold(const String& str, long double default_value, std::size_t* pos = nullptr) noexcept
395{
396 try
397 {
398 return std::stold(str, pos);
399 }
400 catch (...) // NOLINT(bugprone-empty-catch)
401 {}
402
403 return default_value;
404}
405
406} // namespace NAV::str
int stoi(const String &str, int default_value, std::size_t *pos=nullptr, int base=10) noexcept
Interprets a value in the string str.
Definition StringUtil.hpp:297
double stod(const String &str, double default_value, std::size_t *pos=nullptr) noexcept
Interprets a value in the string str.
Definition StringUtil.hpp:375
long double stold(const String &str, long double default_value, std::size_t *pos=nullptr) noexcept
Interprets a value in the string str.
Definition StringUtil.hpp:394
static std::string trim_copy(std::string s)
Trim from both ends (copying)
Definition StringUtil.hpp:103
static std::string ltrim_copy(std::string s)
Trim from start (copying)
Definition StringUtil.hpp:85
static void rtrim(std::string &s)
Trim from end (in place)
Definition StringUtil.hpp:39
static void ltrim(std::string &s)
Trim from start (in place)
Definition StringUtil.hpp:28
static void trim(std::string &s)
Trim from both ends (in place)
Definition StringUtil.hpp:54
int64_t stoll(const String &str, int64_t default_value, std::size_t *pos=nullptr, int base=10) noexcept
Interprets a value in the string str.
Definition StringUtil.hpp:337
static std::string replaceAll_copy(std::string str, const std::string &from, const std::string &to, CaseSensitivity cs)
Replaces all occurrence of a search pattern with another sequence.
Definition StringUtil.hpp:200
float stof(const String &str, float default_value, std::size_t *pos=nullptr) noexcept
Interprets a value in the string str.
Definition StringUtil.hpp:356
static std::string rtrim_copy(std::string s)
Trim from end (copying)
Definition StringUtil.hpp:94
static void replaceAll(std::string &str, const std::string &from, const std::string &to, CaseSensitivity cs)
Replaces all occurrence of a search pattern with another sequence.
Definition StringUtil.hpp:175
static std::vector< std::string > split_wo_empty(const std::string &str, const std::string &delimiter)
Splits a string into parts at a delimiter and removes empty entries.
Definition StringUtil.hpp:250
CaseSensitivity
Enum for case sensitive tasks.
Definition StringUtil.hpp:138
@ RespectCase
Respect the case.
Definition StringUtil.hpp:139
@ IgnoreCase
Ignore case.
Definition StringUtil.hpp:140
static std::vector< std::string > split(const std::string &str, const std::string &delimiter)
Splits a string into parts at a delimiter.
Definition StringUtil.hpp:221
int64_t stol(const String &str, int64_t default_value, std::size_t *pos=nullptr, int base=10) noexcept
Interprets a value in the string str.
Definition StringUtil.hpp:317
static bool replace(std::string &str, const std::string &from, const std::string &to, CaseSensitivity cs=RespectCase)
Replaces the first occurrence of a search pattern with another sequence.
Definition StringUtil.hpp:149
Concept limiting the type to std::string and std::wstring, but also allowing convertible types like c...
Definition StringUtil.hpp:287