JSBSim Flight Dynamics Model 1.2.2 (22 Mar 2025)
An Open Source Flight Dynamics and Control Software Library in C++
Loading...
Searching...
No Matches
string_utilities.cpp
1/*%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
2
3 Module: string_utilities.cpp
4 Author: Bertrand Coconnier / Sean McLeod
5 Date started: 12/28/22
6 Purpose: Utilities to manipulate strings.
7
8 ------------ Copyright (C) 2022 Bertrand Coconnier, Sean McLeod, -------------
9
10 This program is free software; you can redistribute it and/or modify it under
11 the terms of the GNU Lesser General Public License as published by the Free
12 Software Foundation; either version 2 of the License, or (at your option) any
13 later version.
14
15 This program is distributed in the hope that it will be useful, but WITHOUT
16 ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
17 FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for more
18 details.
19
20 You should have received a copy of the GNU Lesser General Public License along
21 with this program; if not, write to the Free Software Foundation, Inc., 59
22 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
23
24 Further information about the GNU Lesser General Public License can also be
25 found on the world wide web at http://www.gnu.org.
26
27FUNCTIONAL DESCRIPTION
28--------------------------------------------------------------------------------
29String handling conveniences such as trim, is_number, split, etc.; these new
30capabilities have been incorporated into the source code where the
31string::find() functions were formerly used.
32
33%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%
34INCLUDES
35%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%*/
36
37#ifdef __CYGWIN__
38#define _GNU_SOURCE 1
39#endif
40#include <errno.h>
41#include <iostream>
42#include <sstream>
43#include <stdio.h>
44#include <regex>
45#ifdef __APPLE__
46#include <xlocale.h>
47#else
48#include <locale.h>
49#endif
50
51#include "FGJSBBase.h"
52#include "string_utilities.h"
53
54#ifdef _WIN32
55typedef _locale_t locale_t;
56#define freelocale _free_locale
57#define strtod_l _strtod_l
58#endif
59
60using namespace std;
61
62namespace JSBSim {
64{
66 {
67#ifdef _WIN32
68 Locale = _create_locale(LC_NUMERIC, "C");
69#else
70 Locale = newlocale(LC_NUMERIC_MASK, "C", 0);
71#endif
72 }
73
75 {
76 freelocale(Locale);
77 }
78
79 locale_t Locale;
80};
81
82/* A locale independent version of atof().
83 * Whatever is the current locale of the application, atof_locale_c() reads
84 * numbers assuming that the decimal point is the period (.)
85 */
86double atof_locale_c(const string& input)
87{
88 static const std::regex number_format(R"(^\s*[+-]?(\d+(\.\d*)?|\.\d+)([eE][+-]?\d+)?\s*$)");
89 const char* first = input.c_str();
90
91 // Skip leading whitespaces
92 while (isspace(*first)) ++first;
93
94 if (!*first)
95 throw InvalidNumber("Expecting a numeric attribute value, but only got spaces");
96
97 if (!std::regex_match(input, number_format))
98 throw InvalidNumber("Expecting a numeric attribute value, but got: " + input);
99
100 CNumericLocale numeric_c;
101 errno = 0; // Reset the error code
102 double value = strtod_l(first, nullptr, numeric_c.Locale);
103
104 // Error management
105 std::stringstream s;
106
107 if (fabs(value) == HUGE_VAL && errno == ERANGE)
108 s << "This number is too large: " << input;
109 else if (fabs(value) == 0 && errno == EINVAL)
110 s << "Expecting a numeric attribute value, but got: " << input;
111 else
112 return value;
113
114 throw InvalidNumber(s.str());
115}
116
117
118std::string& trim_left(std::string& str)
119{
120 while (!str.empty() && isspace((unsigned char)str[0])) {
121 str = str.erase(0,1);
122 }
123 return str;
124}
125
126std::string& trim_right(std::string& str)
127{
128 while (!str.empty() && isspace((unsigned char)str[str.size()-1])) {
129 str = str.erase(str.size()-1,1);
130 }
131 return str;
132}
133
134std::string& trim(std::string& str)
135{
136 if (str.empty()) return str;
137 std::string temp_str = trim_right(str);
138 return str = trim_left(temp_str);
139}
140
141std::string& trim_all_space(std::string& str)
142{
143 for (size_t i=0; i<str.size(); i++) {
144 if (isspace((unsigned char)str[i])) {
145 str = str.erase(i,1);
146 --i;
147 }
148 }
149 return str;
150}
151
152std::string& to_upper(std::string& str)
153{
154 for (size_t i=0; i<str.size(); i++) str[i] = toupper(str[i]);
155 return str;
156}
157
158std::string& to_lower(std::string& str)
159{
160 for (size_t i=0; i<str.size(); i++) str[i] = tolower(str[i]);
161 return str;
162}
163
164bool is_number(const std::string& str)
165{
166 try {
167 atof_locale_c(str);
168 } catch (InvalidNumber&) {
169 return false;
170 }
171
172 return true;
173}
174
175std::vector <std::string> split(std::string str, char d)
176{
177 std::vector <std::string> str_array;
178 size_t index=0;
179 std::string temp = "";
180
181 trim(str);
182 index = str.find(d);
183 while (index != std::string::npos) {
184 temp = str.substr(0,index);
185 trim(temp);
186 if (!temp.empty()) str_array.push_back(temp);
187 str = str.erase(0,index+1);
188 index = str.find(d);
189 }
190 if (!str.empty()) {
191 temp = trim(str);
192 if (!temp.empty()) str_array.push_back(temp);
193 }
194
195 return str_array;
196}
197
198std::string replace(std::string str, const std::string& oldstr, const std::string& newstr)
199{
200 std::string temp = str;
201 size_t old_idx = str.find(oldstr);
202 if (old_idx != std::string::npos) {
203 temp = str.replace(old_idx, 1, newstr);
204 }
205 return temp;
206}
207};