1 //
2 // String.h
3 //
4 // Library: Foundation
5 // Package: Core
6 // Module: String
7 //
8 // Copyright (c) 2004-2006, Applied Informatics Software Engineering GmbH.
9 // and Contributors.
10 //
11 // SPDX-License-Identifier: BSL-1.0
12 //
13
14
15 #include "Poco/String.h"
16
17
18 namespace Poco {
19
20
21 #if defined(POCO_NO_TEMPLATE_ICOMPARE)
22
23
icompare(const std::string & str,std::string::size_type pos,std::string::size_type n,std::string::const_iterator it2,std::string::const_iterator end2)24 int icompare(const std::string& str, std::string::size_type pos, std::string::size_type n, std::string::const_iterator it2, std::string::const_iterator end2)
25 {
26 std::string::size_type sz = str.size();
27 if (pos > sz) pos = sz;
28 if (pos + n > sz) n = sz - pos;
29 std::string::const_iterator it1 = str.begin() + pos;
30 std::string::const_iterator end1 = str.begin() + pos + n;
31 while (it1 != end1 && it2 != end2)
32 {
33 std::string::value_type c1 = Ascii::toLower(*it1);
34 std::string::value_type c2 = Ascii::toLower(*it2);
35 if (c1 < c2)
36 return -1;
37 else if (c1 > c2)
38 return 1;
39 ++it1; ++it2;
40 }
41
42 if (it1 == end1)
43 return it2 == end2 ? 0 : -1;
44 else
45 return 1;
46 }
47
48
icompare(const std::string & str1,const std::string & str2)49 int icompare(const std::string& str1, const std::string& str2)
50 {
51 return icompare(str1, 0, str1.size(), str2.begin(), str2.end());
52 }
53
54
icompare(const std::string & str1,std::string::size_type n1,const std::string & str2,std::string::size_type n2)55 int icompare(const std::string& str1, std::string::size_type n1, const std::string& str2, std::string::size_type n2)
56 {
57 if (n2 > str2.size()) n2 = str2.size();
58 return icompare(str1, 0, n1, str2.begin(), str2.begin() + n2);
59 }
60
61
icompare(const std::string & str1,std::string::size_type n,const std::string & str2)62 int icompare(const std::string& str1, std::string::size_type n, const std::string& str2)
63 {
64 if (n > str2.size()) n = str2.size();
65 return icompare(str1, 0, n, str2.begin(), str2.begin() + n);
66 }
67
68
icompare(const std::string & str1,std::string::size_type pos,std::string::size_type n,const std::string & str2)69 int icompare(const std::string& str1, std::string::size_type pos, std::string::size_type n, const std::string& str2)
70 {
71 return icompare(str1, pos, n, str2.begin(), str2.end());
72 }
73
74
icompare(const std::string & str1,std::string::size_type pos1,std::string::size_type n1,const std::string & str2,std::string::size_type pos2,std::string::size_type n2)75 int icompare(const std::string& str1, std::string::size_type pos1, std::string::size_type n1, const std::string& str2, std::string::size_type pos2, std::string::size_type n2)
76 {
77 std::string::size_type sz2 = str2.size();
78 if (pos2 > sz2) pos2 = sz2;
79 if (pos2 + n2 > sz2) n2 = sz2 - pos2;
80 return icompare(str1, pos1, n1, str2.begin() + pos2, str2.begin() + pos2 + n2);
81 }
82
83
icompare(const std::string & str1,std::string::size_type pos1,std::string::size_type n,const std::string & str2,std::string::size_type pos2)84 int icompare(const std::string& str1, std::string::size_type pos1, std::string::size_type n, const std::string& str2, std::string::size_type pos2)
85 {
86 std::string::size_type sz2 = str2.size();
87 if (pos2 > sz2) pos2 = sz2;
88 if (pos2 + n > sz2) n = sz2 - pos2;
89 return icompare(str1, pos1, n, str2.begin() + pos2, str2.begin() + pos2 + n);
90 }
91
92
icompare(const std::string & str,std::string::size_type pos,std::string::size_type n,const std::string::value_type * ptr)93 int icompare(const std::string& str, std::string::size_type pos, std::string::size_type n, const std::string::value_type* ptr)
94 {
95 poco_check_ptr (ptr);
96 std::string::size_type sz = str.size();
97 if (pos > sz) pos = sz;
98 if (pos + n > sz) n = sz - pos;
99 std::string::const_iterator it = str.begin() + pos;
100 std::string::const_iterator end = str.begin() + pos + n;
101 while (it != end && *ptr)
102 {
103 std::string::value_type c1 = Ascii::toLower(*it);
104 std::string::value_type c2 = Ascii::toLower(*ptr);
105 if (c1 < c2)
106 return -1;
107 else if (c1 > c2)
108 return 1;
109 ++it; ++ptr;
110 }
111
112 if (it == end)
113 return *ptr == 0 ? 0 : -1;
114 else
115 return 1;
116 }
117
118
icompare(const std::string & str,std::string::size_type pos,const std::string::value_type * ptr)119 int icompare(const std::string& str, std::string::size_type pos, const std::string::value_type* ptr)
120 {
121 return icompare(str, pos, str.size() - pos, ptr);
122 }
123
124
icompare(const std::string & str,const std::string::value_type * ptr)125 int icompare(const std::string& str, const std::string::value_type* ptr)
126 {
127 return icompare(str, 0, str.size(), ptr);
128 }
129
130
replace(const std::string & str,const std::string & from,const std::string & to,std::string::size_type start)131 std::string replace(const std::string& str, const std::string& from, const std::string& to, std::string::size_type start)
132 {
133 std::string result(str);
134 replaceInPlace(result, from, to, start);
135 return result;
136 }
137
138
replace(const std::string & str,const std::string::value_type * from,const std::string::value_type * to,std::string::size_type start)139 std::string replace(const std::string& str, const std::string::value_type* from, const std::string::value_type* to, std::string::size_type start)
140 {
141 std::string result(str);
142 replaceInPlace(result, from, to, start);
143 return result;
144 }
145
146
replace(const std::string & str,const std::string::value_type from,const std::string::value_type to,std::string::size_type start)147 std::string replace(const std::string& str, const std::string::value_type from, const std::string::value_type to, std::string::size_type start)
148 {
149 std::string result(str);
150 replaceInPlace(result, from, to, start);
151 return result;
152 }
153
154
remove(const std::string & str,const std::string::value_type ch,std::string::size_type start)155 std::string remove(const std::string& str, const std::string::value_type ch, std::string::size_type start)
156 {
157 std::string result(str);
158 replaceInPlace(result, ch, 0, start);
159 return result;
160 }
161
162
replaceInPlace(std::string & str,const std::string & from,const std::string & to,std::string::size_type start)163 std::string& replaceInPlace(std::string& str, const std::string& from, const std::string& to, std::string::size_type start)
164 {
165 poco_assert (from.size() > 0);
166
167 std::string result;
168 std::string::size_type pos = 0;
169 result.append(str, 0, start);
170 do
171 {
172 pos = str.find(from, start);
173 if (pos != std::string::npos)
174 {
175 result.append(str, start, pos - start);
176 result.append(to);
177 start = pos + from.length();
178 }
179 else result.append(str, start, str.size() - start);
180 }
181 while (pos != std::string::npos);
182 str.swap(result);
183 return str;
184 }
185
186
replaceInPlace(std::string & str,const std::string::value_type * from,const std::string::value_type * to,std::string::size_type start)187 std::string& replaceInPlace(std::string& str, const std::string::value_type* from, const std::string::value_type* to, std::string::size_type start)
188 {
189 poco_assert (*from);
190
191 std::string result;
192 std::string::size_type pos = 0;
193 std::string::size_type fromLen = std::strlen(from);
194 result.append(str, 0, start);
195 do
196 {
197 pos = str.find(from, start);
198 if (pos != std::string::npos)
199 {
200 result.append(str, start, pos - start);
201 result.append(to);
202 start = pos + fromLen;
203 }
204 else result.append(str, start, str.size() - start);
205 }
206 while (pos != std::string::npos);
207 str.swap(result);
208 return str;
209 }
210
211
replaceInPlace(std::string & str,const std::string::value_type from,const std::string::value_type to,std::string::size_type start)212 std::string& replaceInPlace(std::string& str, const std::string::value_type from, const std::string::value_type to, std::string::size_type start)
213 {
214 if (from == to) return str;
215
216 std::string::size_type pos = 0;
217 do
218 {
219 pos = str.find(from, start);
220 if (pos != std::string::npos)
221 {
222 if (to) str[pos] = to;
223 else str.erase(pos, 1);
224 }
225 } while (pos != std::string::npos);
226
227 return str;
228 }
229
230
removeInPlace(std::string & str,const std::string::value_type ch,std::string::size_type start)231 std::string& removeInPlace(std::string& str, const std::string::value_type ch, std::string::size_type start)
232 {
233 return replaceInPlace(str, ch, 0, start);
234 }
235
236
237 #endif
238
239
240 } // namespace Poco
241