Eclipse SUMO - Simulation of Urban MObility
Loading...
Searching...
No Matches
StringUtils.cpp
Go to the documentation of this file.
1/****************************************************************************/
2// Eclipse SUMO, Simulation of Urban MObility; see https://eclipse.dev/sumo
3// Copyright (C) 2001-2026 German Aerospace Center (DLR) and others.
4// This program and the accompanying materials are made available under the
5// terms of the Eclipse Public License 2.0 which is available at
6// https://www.eclipse.org/legal/epl-2.0/
7// This Source Code may also be made available under the following Secondary
8// Licenses when the conditions for such availability set forth in the Eclipse
9// Public License 2.0 are satisfied: GNU General Public License, version 2
10// or later which is available at
11// https://www.gnu.org/licenses/old-licenses/gpl-2.0-standalone.html
12// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-or-later
13/****************************************************************************/
21// Some static methods for string processing
22/****************************************************************************/
23#include <config.h>
24
25#include <string>
26#include <iostream>
27#include <cstdio>
28#include <cstring>
29#include <regex>
30#ifdef WIN32
31#define NOMINMAX
32#include <windows.h>
33#undef NOMINMAX
34#else
35#include <unistd.h>
36#endif
40#include "StringUtils.h"
41
42#define KM_PER_MILE 1.609344
43
44
45// ===========================================================================
46// static member definitions
47// ===========================================================================
48std::string StringUtils::emptyString;
49
50
51// ===========================================================================
52// method definitions
53// ===========================================================================
54std::string
55StringUtils::prune(const std::string& str) {
56 const std::string::size_type endpos = str.find_last_not_of(" \t\n\r");
57 if (std::string::npos != endpos) {
58 const int startpos = (int)str.find_first_not_of(" \t\n\r");
59 return str.substr(startpos, endpos - startpos + 1);
60 }
61 return "";
62}
63
64
65std::string
66StringUtils::pruneZeros(const std::string& str, int max) {
67 const std::string::size_type endpos = str.find_last_not_of("0");
68 if (endpos != std::string::npos && str.back() == '0') {
69 std::string res = str.substr(0, MAX2((int)str.size() - max, (int)endpos + 1));
70 return res;
71 }
72 return str;
73}
74
75std::string
76StringUtils::to_lower_case(const std::string& str) {
77 std::string s = str;
78 std::transform(s.begin(), s.end(), s.begin(), [](char c) {
79 return (char)::tolower(c);
80 });
81 return s;
82}
83
84
85std::string
86StringUtils::to_upper_case(const std::string& str) {
87 std::string s = str;
88 std::transform(s.begin(), s.end(), s.begin(), [](char c) {
89 return (char)::toupper(c);
90 });
91 return s;
92}
93
94
95std::string
97 // inspired by http://stackoverflow.com/questions/4059775/convert-iso-8859-1-strings-to-utf-8-in-c-c
98 std::string result;
99 for (const auto& c : str) {
100 const unsigned char uc = (unsigned char)c;
101 if (uc < 128) {
102 result += uc;
103 } else {
104 result += (char)(0xc2 + (uc > 0xbf));
105 result += (char)((uc & 0x3f) + 0x80);
106 }
107 }
108 return result;
109}
110
111
112std::string
114 str = replace(str, "\xE4", "ae");
115 str = replace(str, "\xC4", "Ae");
116 str = replace(str, "\xF6", "oe");
117 str = replace(str, "\xD6", "Oe");
118 str = replace(str, "\xFC", "ue");
119 str = replace(str, "\xDC", "Ue");
120 str = replace(str, "\xDF", "ss");
121 str = replace(str, "\xC9", "E");
122 str = replace(str, "\xE9", "e");
123 str = replace(str, "\xC8", "E");
124 str = replace(str, "\xE8", "e");
125 return str;
126}
127
128
129std::string
130StringUtils::replace(std::string str, const std::string& what, const std::string& by) {
131 std::string::size_type idx = str.find(what);
132 const int what_len = (int)what.length();
133 if (what_len > 0) {
134 const int by_len = (int)by.length();
135 while (idx != std::string::npos) {
136 str = str.replace(idx, what_len, by);
137 idx = str.find(what, idx + by_len);
138 }
139 }
140 return str;
141}
142
143
144std::string
145StringUtils::substituteEnvironment(const std::string& str, const std::chrono::time_point<std::chrono::system_clock>* const timeRef) {
146 std::string s = str;
147 if (timeRef != nullptr) {
148 const std::string::size_type localTimeIndex = str.find("${LOCALTIME}");
149 const std::string::size_type utcIndex = str.find("${UTC}");
150 const bool isUTC = utcIndex != std::string::npos;
151 if (localTimeIndex != std::string::npos || isUTC) {
152 const time_t rawtime = std::chrono::system_clock::to_time_t(*timeRef);
153 char buffer [80];
154 struct tm* timeinfo = isUTC ? gmtime(&rawtime) : localtime(&rawtime);
155 strftime(buffer, 80, "%Y-%m-%d-%H-%M-%S.", timeinfo);
156 auto seconds = std::chrono::time_point_cast<std::chrono::seconds>(*timeRef);
157 auto microseconds = std::chrono::duration_cast<std::chrono::microseconds>(*timeRef - seconds);
158 const std::string micro = buffer + toString(microseconds.count());
159 if (isUTC) {
160 s.replace(utcIndex, 6, micro);
161 } else {
162 s.replace(localTimeIndex, 12, micro);
163 }
164 }
165 }
166 const std::string::size_type pidIndex = str.find("${PID}");
167 if (pidIndex != std::string::npos) {
168#ifdef WIN32
169 s.replace(pidIndex, 6, toString(::GetCurrentProcessId()));
170#else
171 s.replace(pidIndex, 6, toString(::getpid()));
172#endif
173 }
174 if (std::getenv("SUMO_LOGO") == nullptr) {
175 s = replace(s, "${SUMO_LOGO}", "${SUMO_HOME}/data/logo/sumo-128x138.png");
176 }
177 const std::string::size_type tildeIndex = str.find("~");
178 if (tildeIndex == 0) {
179 s.replace(0, 1, "${HOME}");
180 }
181 s = replace(s, ",~", ",${HOME}");
182#ifdef WIN32
183 if (std::getenv("HOME") == nullptr) {
184 s = replace(s, "${HOME}", "${USERPROFILE}");
185 }
186#endif
187
188 // Expression for an environment variables, e.g. ${NAME}
189 // Note: - R"(...)" is a raw string literal syntax to simplify a regex declaration
190 // - .+? looks for the shortest match (non-greedy)
191 // - (.+?) defines a "subgroup" which is already stripped of the $ and {, }
192 std::regex envVarExpr(R"(\$\{(.+?)\})");
193
194 // Are there any variables in this string?
195 std::smatch match;
196 std::string strIter = s;
197
198 // Loop over the entire value string and look for variable names
199 while (std::regex_search(strIter, match, envVarExpr)) {
200 std::string varName = match[1];
201
202 // Find the variable in the environment and its value
203 std::string varValue;
204 if (std::getenv(varName.c_str()) != nullptr) {
205 varValue = std::getenv(varName.c_str());
206 }
207
208 // Replace the variable placeholder with its value in the original string
209 s = std::regex_replace(s, std::regex("\\$\\{" + varName + "\\}"), varValue);
210
211 // Continue the loop with the remainder of the string
212 strIter = match.suffix();
213 }
214 return s;
215}
216
217
218std::string
219StringUtils::isoTimeString(const std::chrono::time_point<std::chrono::system_clock>* const timeRef) {
220 const std::chrono::system_clock::time_point now = timeRef == nullptr ? std::chrono::system_clock::now() : *timeRef;
221 const auto now_seconds = std::chrono::time_point_cast<std::chrono::seconds>(now);
222 const std::time_t now_c = std::chrono::system_clock::to_time_t(now);
223 const auto microseconds = std::chrono::duration_cast<std::chrono::microseconds>(now - now_seconds).count();
224 std::tm local_tm = *std::localtime(&now_c);
225
226 // Get the time zone offset
227 std::time_t utc_time = std::time(nullptr);
228 std::tm utc_tm = *std::gmtime(&utc_time);
229 const double offset = std::difftime(std::mktime(&local_tm), std::mktime(&utc_tm)) / 3600.0;
230 const int hours_offset = static_cast<int>(offset);
231 const int minutes_offset = static_cast<int>((offset - hours_offset) * 60);
232
233 // Format the time
234 std::ostringstream oss;
235 char buf[32];
236 std::strftime(buf, sizeof(buf), "%Y-%m-%dT%H:%M:%S", &local_tm);
237 oss << buf << "."
238 << std::setw(6) << std::setfill('0') << std::abs(microseconds)
239 << (hours_offset >= 0 ? "+" : "-")
240 << std::setw(2) << std::setfill('0') << std::abs(hours_offset) << ":"
241 << std::setw(2) << std::setfill('0') << std::abs(minutes_offset);
242 return oss.str();
243}
244
245
246bool
247StringUtils::startsWith(const std::string& str, const std::string prefix) {
248 return str.compare(0, prefix.length(), prefix) == 0;
249}
250
251
252bool
253StringUtils::endsWith(const std::string& str, const std::string suffix) {
254 if (str.length() >= suffix.length()) {
255 return str.compare(str.length() - suffix.length(), suffix.length(), suffix) == 0;
256 } else {
257 return false;
258 }
259}
260
261
262std::string
263StringUtils::padFront(const std::string& str, int length, char padding) {
264 return std::string(MAX2(0, length - (int)str.size()), padding) + str;
265}
266
267
268std::string
269StringUtils::escapeXML(const std::string& orig, const bool maskDoubleHyphen) {
270 std::string result = replace(orig, "&", "&amp;");
271 result = replace(result, ">", "&gt;");
272 result = replace(result, "<", "&lt;");
273 result = replace(result, "\"", "&quot;");
274 if (maskDoubleHyphen) {
275 result = replace(result, "--", "&#45;&#45;");
276 }
277 for (char invalid = '\1'; invalid < ' '; invalid++) {
278 result = replace(result, std::string(1, invalid).c_str(), "");
279 }
280 return replace(result, "'", "&apos;");
281}
282
283
284std::string
285StringUtils::escapeShell(const std::string& orig) {
286 std::string result = replace(orig, "\"", "\\\"");
287 return result;
288}
289
290
291std::string
292StringUtils::urlEncode(const std::string& toEncode, const std::string encodeWhich) {
293 std::ostringstream out;
294
295 for (int i = 0; i < (int)toEncode.length(); ++i) {
296 const char t = toEncode.at(i);
297
298 if ((encodeWhich != "" && encodeWhich.find(t) == std::string::npos) ||
299 (encodeWhich == "" &&
300 ((t >= 45 && t <= 57) || // hyphen, period, slash, 0-9
301 (t >= 65 && t <= 90) || // A-Z
302 t == 95 || // underscore
303 (t >= 97 && t <= 122) || // a-z
304 t == 126)) // tilde
305 ) {
306 out << toEncode.at(i);
307 } else {
308 out << charToHex(toEncode.at(i));
309 }
310 }
311
312 return out.str();
313}
314
315
316std::string
317StringUtils::urlDecode(const std::string& toDecode) {
318 std::ostringstream out;
319
320 for (int i = 0; i < (int)toDecode.length(); ++i) {
321 if (toDecode.at(i) == '%') {
322 std::string str(toDecode.substr(i + 1, 2));
323 out << hexToChar(str);
324 i += 2;
325 } else {
326 out << toDecode.at(i);
327 }
328 }
329
330 return out.str();
331}
332
333std::string
334StringUtils::charToHex(unsigned char c) {
335 short i = c;
336
337 std::stringstream s;
338
339 s << "%" << std::setw(2) << std::setfill('0') << std::hex << i;
340
341 return s.str();
342}
343
344
345unsigned char
346StringUtils::hexToChar(const std::string& str) {
347 short c = 0;
348 if (!str.empty()) {
349 std::istringstream in(str);
350 in >> std::hex >> c;
351 if (in.fail()) {
352 throw NumberFormatException(str + " could not be interpreted as hex");
353 }
354 }
355 return static_cast<unsigned char>(c);
356}
357
358
359int
360StringUtils::toInt(const std::string& sData) {
361 long long int result = toLong(sData);
362 if (result > std::numeric_limits<int>::max() || result < std::numeric_limits<int>::min()) {
363 throw NumberFormatException(toString(result) + " int overflow");
364 }
365 return (int)result;
366}
367
368
369bool
370StringUtils::isInt(const std::string& sData) {
371 // first check if can be converted to long int
372 if (isLong(sData)) {
373 const long long int result = toLong(sData);
374 // now check if the result is in the range of an int
375 return ((result <= std::numeric_limits<int>::max()) && (result >= std::numeric_limits<int>::min()));
376 }
377 return false;
378}
379
380
381int
382StringUtils::toIntSecure(const std::string& sData, int def) {
383 if (sData.length() == 0) {
384 return def;
385 }
386 return toInt(sData);
387}
388
389
390long long int
391StringUtils::toLong(const std::string& sData) {
392 const char* const data = sData.c_str();
393 if (data == 0 || data[0] == 0) {
394 throw EmptyData();
395 }
396 char* end;
397 errno = 0;
398#ifdef _MSC_VER
399 long long int ret = _strtoi64(data, &end, 10);
400#else
401 long long int ret = strtoll(data, &end, 10);
402#endif
403 if (errno == ERANGE) {
404 errno = 0;
405 throw NumberFormatException("(long long integer range) " + sData);
406 }
407 if ((int)(end - data) != (int)strlen(data)) {
408 throw NumberFormatException("(long long integer format) " + sData);
409 }
410 return ret;
411}
412
413
414bool
415StringUtils::isLong(const std::string& sData) {
416 const char* const data = sData.c_str();
417 if (data == 0 || data[0] == 0) {
418 return false;
419 }
420 char* end;
421 // reset errno before parsing, to keep errors
422 errno = 0;
423 // continue depending of current plattform
424#ifdef _MSC_VER
425 _strtoi64(data, &end, 10);
426#else
427 strtoll(data, &end, 10);
428#endif
429 // check out of range
430 if (errno == ERANGE) {
431 return false;
432 }
433 // check length of converted data
434 if ((int)(end - data) != (int)strlen(data)) {
435 return false;
436 }
437 return true;
438}
439
440
441int
442StringUtils::hexToInt(const std::string& sData) {
443 if (sData.length() == 0) {
444 throw EmptyData();
445 }
446 size_t idx = 0;
447 int result;
448 try {
449 if (sData[0] == '#') { // for html color codes
450 result = std::stoi(sData.substr(1), &idx, 16);
451 idx++;
452 } else {
453 result = std::stoi(sData, &idx, 16);
454 }
455 } catch (...) {
456 throw NumberFormatException("(hex integer format) " + sData);
457 }
458 if (idx != sData.length()) {
459 throw NumberFormatException("(hex integer format) " + sData);
460 }
461 return result;
462}
463
464
465bool
466StringUtils::isHex(std::string sData) {
467 if (sData.length() == 0) {
468 return false;
469 }
470 // remove the first character (for HTML color codes)
471 if (sData[0] == '#') {
472 sData = sData.substr(1);
473 }
474 const char* sDataPtr = sData.c_str();
475 char* returnPtr;
476 // reset errno
477 errno = 0;
478 // call string to long (size 16) from standard library
479 strtol(sDataPtr, &returnPtr, 16);
480 // check out of range
481 if (errno == ERANGE) {
482 return false;
483 }
484 // check if there was an error converting sDataPtr to double,
485 if (sDataPtr == returnPtr) {
486 return false;
487 }
488 // compare size of start and end points
489 if (static_cast<size_t>(returnPtr - sDataPtr) != sData.size()) {
490 return false;
491 }
492 return true;
493}
494
495
496double
497StringUtils::toDouble(const std::string& sData) {
498 if (sData.size() == 0) {
499 throw EmptyData();
500 }
501 try {
502 size_t idx = 0;
503 const double result = std::stod(sData, &idx);
504 if (idx != sData.size()) {
505 throw NumberFormatException("(double format) " + sData);
506 } else {
507 return result;
508 }
509 } catch (...) {
510 // invalid_argument or out_of_range
511 throw NumberFormatException("(double) " + sData);
512 }
513}
514
515
516bool
517StringUtils::isDouble(const std::string& sData) {
518 if (sData.size() == 0) {
519 return false;
520 }
521 const char* sDataPtr = sData.c_str();
522 char* returnPtr;
523 // reset errno
524 errno = 0;
525 // call string to double from standard library
526 strtod(sDataPtr, &returnPtr);
527 // check out of range
528 if (errno == ERANGE) {
529 return false;
530 }
531 // check if there was an error converting sDataPtr to double,
532 if (sDataPtr == returnPtr) {
533 return false;
534 }
535 // compare size of start and end points
536 if (static_cast<size_t>(returnPtr - sDataPtr) != sData.size()) {
537 return false;
538 }
539 return true;
540}
541
542
543double
544StringUtils::toDoubleSecure(const std::string& sData, const double def) {
545 if (sData.length() == 0) {
546 return def;
547 }
548 return toDouble(sData);
549}
550
551
552bool
553StringUtils::toBool(const std::string& sData) {
554 if (sData.length() == 0) {
555 throw EmptyData();
556 }
557 const std::string s = to_lower_case(sData);
558 if (s == "1" || s == "yes" || s == "true" || s == "on" || s == "x" || s == "t") {
559 return true;
560 }
561 if (s == "0" || s == "no" || s == "false" || s == "off" || s == "-" || s == "f") {
562 return false;
563 }
564 throw BoolFormatException(s);
565}
566
567
568bool
569StringUtils::isBool(const std::string& sData) {
570 if (sData.length() == 0) {
571 return false;
572 }
573 const std::string s = to_lower_case(sData);
574 // check true values
575 if (s == "1" || s == "yes" || s == "true" || s == "on" || s == "x" || s == "t") {
576 return true;
577 }
578 // check false values
579 if (s == "0" || s == "no" || s == "false" || s == "off" || s == "-" || s == "f") {
580 return true;
581 }
582 // no valid true or false values
583 return false;
584}
585
586
588StringUtils::toVersion(const std::string& sData) {
589 std::vector<std::string> parts = StringTokenizer(sData, ".").getVector();
590 return MMVersion(toInt(parts.front()), toDouble(parts.back()));
591}
592
593
594double
595StringUtils::parseDist(const std::string& sData) {
596 if (sData.size() == 0) {
597 throw EmptyData();
598 }
599 try {
600 size_t idx = 0;
601 const double result = std::stod(sData, &idx);
602 if (idx != sData.size()) {
603 const std::string unit = prune(sData.substr(idx));
604 if (unit == "m" || unit == "metre" || unit == "meter" || unit == "metres" || unit == "meters") {
605 return result;
606 }
607 if (unit == "km" || unit == "kilometre" || unit == "kilometer" || unit == "kilometres" || unit == "kilometers") {
608 return result * 1000.;
609 }
610 if (unit == "mi" || unit == "mile" || unit == "miles") {
611 return result * 1000. * KM_PER_MILE;
612 }
613 if (unit == "nmi") {
614 return result * 1852.;
615 }
616 if (unit == "ft" || unit == "foot" || unit == "feet") {
617 return result * 12. * 0.0254;
618 }
619 if (unit == "\"" || unit == "in" || unit == "inch" || unit == "inches") {
620 return result * 0.0254;
621 }
622 if (unit[0] == '\'') {
623 double inches = 12 * result;
624 if (unit.length() > 1) {
625 inches += std::stod(unit.substr(1), &idx);
626 if (unit.substr(idx) == "\"") {
627 return inches * 0.0254;
628 }
629 }
630 }
631 throw NumberFormatException("(distance format) " + sData);
632 } else {
633 return result;
634 }
635 } catch (...) {
636 // invalid_argument or out_of_range
637 throw NumberFormatException("(double) " + sData);
638 }
639}
640
641
642double
643StringUtils::parseSpeed(const std::string& sData, const bool defaultKmph) {
644 if (sData.size() == 0) {
645 throw EmptyData();
646 }
647 try {
648 size_t idx = 0;
649 const double result = std::stod(sData, &idx);
650 if (idx != sData.size()) {
651 const std::string unit = prune(sData.substr(idx));
652 if (unit == "km/h" || unit == "kph" || unit == "kmh" || unit == "kmph") {
653 return result / 3.6;
654 }
655 if (unit == "m/s") {
656 return result;
657 }
658 if (unit == "mph") {
659 return result * KM_PER_MILE / 3.6;
660 }
661 if (unit == "knots") {
662 return result * 1.852 / 3.6;
663 }
664 throw NumberFormatException("(speed format) " + sData);
665 } else {
666 return defaultKmph ? result / 3.6 : result;
667 }
668 } catch (...) {
669 // invalid_argument or out_of_range
670 throw NumberFormatException("(double) " + sData);
671 }
672}
673
674
675
676std::string
677StringUtils::trim_left(const std::string s, const std::string& t) {
678 std::string result = s;
679 result.erase(0, s.find_first_not_of(t));
680 return result;
681}
682
683std::string
684StringUtils::trim_right(const std::string s, const std::string& t) {
685 std::string result = s;
686 result.erase(s.find_last_not_of(t) + 1);
687 return result;
688}
689
690std::string
691StringUtils::trim(const std::string s, const std::string& t) {
692 return trim_right(trim_left(s, t), t);
693}
694
695
696std::string
697StringUtils::wrapText(const std::string s, int width) {
698 std::vector<std::string> parts = StringTokenizer(s).getVector();
699 std::string result;
700 std::string line;
701 bool firstLine = true;
702 bool firstWord = true;
703 for (std::string p : parts) {
704 if ((int)(line.size() + p.size()) < width || firstWord) {
705 if (firstWord) {
706 firstWord = false;
707 } else {
708 line += " ";
709 }
710 line += p;
711 } else {
712 if (firstLine) {
713 firstLine = false;
714 } else {
715 result += "\n";
716 }
717 result += line;
718 line.clear();
719 line += p;
720 }
721 }
722 if (line.size() > 0) {
723 if (firstLine) {
724 firstLine = false;
725 } else {
726 result += "\n";
727 }
728 result += line;
729 }
730 return result;
731}
732
733
734std::string
735StringUtils::adjustDecimalValue(double value, int precision) {
736 // obtain value in string format with 20 decimals precision
737 auto valueStr = toString(value, precision);
738 // now clear all zeros
739 while (valueStr.size() > 1) {
740 if (valueStr.back() == '0') {
741 valueStr.pop_back();
742 } else if (valueStr.back() == '.') {
743 valueStr.pop_back();
744 return valueStr;
745 } else {
746 return valueStr;
747 }
748 }
749 return valueStr;
750}
751
752
753/****************************************************************************/
std::pair< int, double > MMVersion
(M)ajor/(M)inor version for written networks and default version for loading
Definition StdDefs.h:71
T MAX2(T a, T b)
Definition StdDefs.h:86
#define KM_PER_MILE
std::string toString(const T &t, std::streamsize accuracy=gPrecision)
Definition ToString.h:49
std::vector< std::string > getVector()
return vector of strings
static std::string pruneZeros(const std::string &str, int max)
Removes trailing zeros (at most 'max')
static std::string urlEncode(const std::string &url, const std::string encodeWhich="")
encode url (stem from http://bogomip.net/blog/cpp-url-encoding-and-decoding/)
static bool isDouble(const std::string &sData)
check if the given sData can be conveted to double
static MMVersion toVersion(const std::string &sData)
parse a (network) version string
static bool isBool(const std::string &sData)
check if the given value can be converted to bool
static std::string to_upper_case(const std::string &str)
Transfers the content to upper case.
static std::string charToHex(unsigned char c)
char to hexadecimal
static std::string urlDecode(const std::string &encoded)
decode url (stem from http://bogomip.net/blog/cpp-url-encoding-and-decoding/)
static long long int toLong(const std::string &sData)
converts a string into the long value described by it by calling the char-type converter,...
static double toDoubleSecure(const std::string &sData, const double def)
converts a string into the integer value described by it
static std::string trim(const std::string s, const std::string &t=" \t\n")
remove leading and trailing whitespace
static std::string to_lower_case(const std::string &str)
Transfers the content to lower case.
static std::string trim_right(const std::string s, const std::string &t=" \t\n")
remove trailing whitespace from string
static std::string trim_left(const std::string s, const std::string &t=" \t\n")
remove leading whitespace from string
static std::string escapeShell(const std::string &orig)
Escape special characters with backslash.
static std::string replace(std::string str, const std::string &what, const std::string &by)
Replaces all occurrences of the second string by the third string within the first string.
static int hexToInt(const std::string &sData)
converts a string with a hex value into the integer value described by it by calling the char-type co...
static double toDouble(const std::string &sData)
converts a string into the double value described by it by calling the char-type converter
static std::string escapeXML(const std::string &orig, const bool maskDoubleHyphen=false)
Replaces the standard escapes by their XML entities.
static bool isHex(std::string sData)
check if the given string can be converted to hex
static std::string latin1_to_utf8(std::string str)
Transfers from Latin 1 (ISO-8859-1) to UTF-8.
static std::string prune(const std::string &str)
Removes trailing and leading whitechars.
static std::string padFront(const std::string &str, int length, char padding)
static std::string convertUmlaute(std::string str)
Converts german "Umlaute" to their latin-version.
static double parseDist(const std::string &sData)
parse a distance, length or width value with a unit
static std::string adjustDecimalValue(double value, int precision)
write with maximum precision if needed but remove trailing zeros
static unsigned char hexToChar(const std::string &str)
hexadecimal to char
static bool startsWith(const std::string &str, const std::string prefix)
Checks whether a given string starts with the prefix.
static std::string wrapText(const std::string s, int width)
remove leading and trailing whitespace
static double parseSpeed(const std::string &sData, const bool defaultKmph=true)
parse a speed value with a unit
static std::string emptyString
An empty string.
Definition StringUtils.h:96
static bool endsWith(const std::string &str, const std::string suffix)
Checks whether a given string ends with the suffix.
static std::string substituteEnvironment(const std::string &str, const std::chrono::time_point< std::chrono::system_clock > *const timeRef=nullptr)
Replaces an environment variable with its value (similar to bash); syntax for a variable is ${NAME}.
static bool isLong(const std::string &sData)
Check if the given sData can be converted to long.
static int toIntSecure(const std::string &sData, int def)
converts a string into the integer value described by it
static std::string isoTimeString(const std::chrono::time_point< std::chrono::system_clock > *const timeRef=nullptr)
Returns an ISO8601 formatted time string with microsecond precision.
static int toInt(const std::string &sData)
converts a string into the integer value described by it by calling the char-type converter,...
static bool isInt(const std::string &sData)
check if the given sData can be converted to int
static bool toBool(const std::string &sData)
converts a string into the bool value described by it by calling the char-type converter