XTL  0.1
eXtended Template Library
 All Data Structures Namespaces Files Functions Variables Typedefs Enumerations Enumerator Macros Groups Pages
string.hpp
Go to the documentation of this file.
1 
7 #pragma once
8 
9 #include <xtd/xtd.hpp>
10 
11 #include <string>
12 #include <vector>
13 #include <stdexcept>
14 
15 #if (XTD_HAS_CODECVT)
16  #include <codecvt>
17 #elif (XTD_HAS_EXP_CODECVT)
18  #include <experimental/codecvt>
19 #elif (XTD_HAS_ICONV)
20  #include <iconv.h>
21 #endif
22 #include <cwctype>
23 
24 #include <locale>
25 
26 #include <string.h>
27 
28 namespace xtd{
29 
30  template <typename _ChT> class xstring;
31 
32  using string = xstring<char>;
33  using wstring = xstring<wchar_t>;
34  using tstring = xstring<tchar>;
35 
36 #if (!DOXY_INVOKED)
37  namespace _{
38  template <typename, typename> class xstring_format;
39 
40  }
41 #endif
42 
46  template <typename _ChT> class xstring : public std::basic_string<_ChT>{
47  public:
48  using _super_t = std::basic_string<_ChT>;
49  using size_type = typename _super_t::size_type;
50 
52  template <typename ... _ArgsT>
53  xstring(_ArgsT&&...oArgs)
54  : _super_t(std::forward<_ArgsT>(oArgs)...){}
55 
62  static xstring format(){
63  xstring sRet = ((_ChT*)"\0\0\0");
64  return sRet;
65  }
66 
67 
68  template <typename _Ty, typename ... _ArgsT>
69  static xstring format(const _Ty & val, _ArgsT&&...oArgs){
70  xstring sRet = _::xstring_format<_ChT, const _Ty &>::format(static_cast<const _Ty &>(val));
71  sRet += format(std::forward<_ArgsT>(oArgs)...);
72  return sRet;
73  }
74 
75 
80  xstring& to_lower(const std::locale& loc){
81  for (auto & ch : *this){
82  ch = std::tolower(ch, loc);
83  }
84  return *this;
85  }
86 
91  xstring& to_upper(const std::locale& loc){
92  for (auto & ch : *this){
93  ch = std::toupper(ch, loc);
94  }
95  return *this;
96  }
97 
100  auto oBegin = _super_t::begin();
101  if (oBegin >= _super_t::end()){
102  return *this;
103  }
104  for(;oBegin != _super_t::end() ; ++oBegin){
105  if (!std::iswspace(*oBegin)){
106  break;
107  }
108  }
109  _super_t::erase(_super_t::begin(), oBegin);
110  return *this;
111  }
112 
115  auto oBegin = _super_t::end();
116  if (oBegin <= _super_t::begin()){
117  return *this;
118  }
119  --oBegin;
120  for(;oBegin != _super_t::begin() ; --oBegin){
121  if (!std::iswspace(*oBegin) && *oBegin){
122  break;
123  }
124  }
125  _super_t::erase(++oBegin, _super_t::end());
126  return *this;
127  }
128 
129  //Trim leading and trailing whitespace
130  xstring& trim(){
131  ltrim();
132  return rtrim();
133  }
134 
135  //replaces all occurances of the characters in the oItems list with a specified character
136  xstring& replace(std::initializer_list<_ChT> oItems, _ChT chReplace){
137  for (auto & oCh : *this){
138  bool bFound = false;
139  for (const auto & oFind : oItems){
140  if (oFind == oCh){
141  bFound = true;
142  break;
143  }
144  }
145  if (bFound) {
146  oCh = chReplace;
147  }
148  }
149  return *this;
150  }
151 
153  size_type find_first_of(const std::initializer_list<_ChT>& delimiters, size_type pos = 0) const{
154  size_type sRet = _super_t::npos;
155  for (const _ChT ch : delimiters){
156  auto x = _super_t::find_first_of(ch, pos);
157  if ((_super_t::npos != x) && (_super_t::npos == sRet || x < sRet)){
158  sRet = x;
159  }
160  }
161  return sRet;
162  }
163 
165  std::vector<xstring<_ChT>> split(const std::initializer_list<_ChT>& delimiters, bool trimEmpty = false) const{
166  using container_t = std::vector<xstring<_ChT>>;
167  container_t oRet;
168  using _my_t = xstring<_ChT>;
169  size_type pos;
170  size_type lastPos = 0;
171 
172  using value_type = typename container_t::value_type;
173 
174  forever{
175  pos = find_first_of(delimiters, lastPos);
176  if (pos == _my_t::npos){
177  pos = _super_t::length();
178  if (pos != lastPos || !trimEmpty){
179  oRet.push_back(value_type(_super_t::data() + lastPos, (pos - lastPos)));
180  }
181  break;
182  } else if (pos != lastPos || !trimEmpty){
183  oRet.push_back(value_type(_super_t::data() + lastPos , (pos - lastPos)));
184  }
185 
186  lastPos = pos + 1;
187  }
188  return oRet;
189  }
190 
191 #if (!(XTD_HAS_CODECVT | XTD_HAS_EXP_CODECVT)) && (XTD_HAS_ICONV)
192 
194  class iconv_helper{
195  public:
196  iconv_helper(const iconv_helper&) = delete;
197  iconv_helper& operator=(const iconv_helper&) = delete;
198  iconv_helper(const char * to, const char * from) : _iconv(iconv_open(to, from)){
199  if ((iconv_t)-1 == _iconv){
200  throw std::runtime_error("Failure creating iconv handle");
201  }
202  }
203  ~iconv_helper(){
204  iconv_close(_iconv);
205  }
206  operator iconv_t(){ return _iconv; }
207  iconv_t _iconv;
208  };
209 #endif
210  };
211 
212 #if (!DOXY_INVOKED)
213 
214  namespace _{
215 
216  #if (XTD_HAS_CODECVT || XTD_HAS_EXP_CODECVT)
217  template <> class xstring_format<char, const wchar_t * const &>{
218  public:
219  static inline string format(const wchar_t * const & src){
220  std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>, wchar_t> oConv;
221  return oConv.to_bytes(src);
222  }
223  };
224 
225  template <> class xstring_format<wchar_t, const char * const &>{
226  public:
227  static inline wstring format(const char * const & src){
228  std::wstring_convert<std::codecvt_utf8_utf16<wchar_t>, wchar_t> oConv;
229  return oConv.from_bytes(src);
230  }
231  };
232 
233  #elif (XTD_HAS_ICONV)
234 
235  template <> class xstring_format<char, const wchar_t * const &>{
236  public:
237  static string format(const wchar_t * const & src){
238  static iconv_helper oIconv("UTF-8", "WCHAR_T");
239  size_t srclen = wcslen(src);
240  string sRet(srclen, 0);
241  size_t retlen = srclen;
242  iconv(oIconv, nullptr, nullptr, nullptr, &retlen);
243  forever{
244  retlen = sRet.size();
245  auto pSrc = reinterpret_cast<const char *>(src);
246  auto iSrcLen = srclen * sizeof(wchar_t);
247  auto sRetPtr = &sRet[0];
248  auto iRet = iconv(oIconv, const_cast<char**>(&pSrc), &iSrcLen, &sRetPtr, &retlen);
249  if (static_cast<size_t>(-1) != iRet){
250  return sRet;
251  }
252  if (errno != E2BIG){
253  throw std::runtime_error("A string conversion error occurred");
254  }
255  sRet.resize(sRet.size() * 2);
256  }
257  }
258  };
259 
260  template <> class xstring_format<wchar_t, const char * const &>{
261  public:
262  static inline wstring format(const char * const & src){
263  static iconv_helper oIconv("WCHAR_T", "UTF-8");
264  size_t srclen = strlen(src);
265  wstring sRet(srclen, 0);
266  size_t retlen = srclen;
267  iconv(oIconv, nullptr, nullptr, nullptr, &retlen);
268  forever{
269  retlen = sRet.size() * sizeof(wchar_t);
270  auto pSrc = const_cast<char*>(src);
271  auto iSrcLen = srclen;
272  auto sRetPtr = reinterpret_cast<char*>(&sRet[0]);
273  auto iRet = iconv(oIconv, &pSrc, &iSrcLen, &sRetPtr, &retlen);
274  if (static_cast<size_t>(-1) != iRet){
275  return sRet;
276  }
277  if (errno != E2BIG){
278  throw std::runtime_error("A string conversion error occurred");
279  }
280  sRet.resize(sRet.size() * 2);
281  }
282  }
283  };
284 
285  #else
286  template <> class xstring_format<char, const wchar_t * const &>{
287  public:
288  static string format(const wchar_t * const & src){
289  size_t srclen = wcslen(src);
290  string sRet(srclen, 0);
291  forever{
292  srclen = wcstombs(&sRet[0], src, sRet.size());
293  if (static_cast<size_t>(-1) == srclen){
294  throw std::runtime_error("A string conversion error occurred");
295  }
296  if (srclen < sRet.size()){
297  break;
298  }
299  sRet.resize(srclen * 2);
300  }
301  sRet.resize(srclen);
302  return sRet;
303  }
304  };
305 
306  template <> class xstring_format<wchar_t, const char * const &>{
307  public:
308 
309  static inline wstring format(const char * const & src){
310  size_t srclen = strlen(src);
311  wstring sRet(1+srclen, '\0');
312  forever{
313  srclen = mbstowcs(&sRet[0], src, sRet.size());
314  if (static_cast<size_t>(-1) == srclen){
315  throw std::runtime_error("A string conversion error occurred");
316  }
317  if (srclen <= sRet.size()){
318  break;
319  }
320  sRet.resize(srclen * 2);
321  }
322  sRet.resize(srclen);
323  return sRet;
324  }
325  };
326 
327  #endif
328 
329  template <typename _ChT> class xstring_format<_ChT, const _ChT * const &>{
330  public:
331  inline static xstring<_ChT> format(const _ChT * const & src){
332  return xstring<_ChT>(src);
333  }
334  };
335 
336  template <typename _ChT, typename _Ch2, size_t _Len> class xstring_format<_ChT, const _Ch2(&)[_Len]>{
337  public:
338  inline static xstring<_ChT> format(const _Ch2(&src)[_Len]){
339  return xstring_format<_ChT, const _Ch2 * const &>::format(src);
340  }
341  };
342 
343  template <typename _ChT, typename _Ch2> class xstring_format<_ChT, const xtd::xstring<_Ch2> &>{
344  public:
345  inline static xstring<_ChT> format(const xtd::xstring<_Ch2> & src){
346  return xstring<_ChT>::format(src.c_str());
347  }
348  };
349  template <typename _ChT, typename _Ch2> class xstring_format<_ChT, const std::basic_string<_Ch2> &>{
350  public:
351  inline static xstring<_ChT> format(const std::basic_string<_Ch2> & src){
352  return xstring<_ChT>::format(src.c_str());
353  }
354  };
355 
356 
357  template <> class xstring_format<char, const void * const &>{
358  public:
359  inline static string format(const void * const & value){
360  return std::to_string(reinterpret_cast<size_t>(value));
361  }
362  };
363 
364 
365  template <> class xstring_format<char, const int32_t &>{
366  public:
367  inline static string format(const int32_t & value){
368  return std::to_string(value);
369  }
370  };
371 
372  template <> class xstring_format<char, const uint32_t &>{
373  public:
374  inline static string format(const uint32_t & value){
375  return std::to_string(value);
376  }
377  };
378 
379  template <> class xstring_format<char, const int64_t &>{
380  public:
381  inline static string format(const int64_t & value){
382  return std::to_string(value);
383  }
384  };
385 
386  template <> class xstring_format<char, const uint64_t &>{
387  public:
388  inline static string format(const uint64_t & value){
389  return std::to_string(value);
390  }
391  };
392 
393 
394  template <> class xstring_format<char, void * const &>{
395  public:
396  inline static string format(const void * const & value){
397  return std::to_string(reinterpret_cast<size_t>(value));
398  }
399  };
400 
401 #if ((XTD_OS_MINGW | XTD_OS_WINDOWS) & XTD_OS)
402  template <> class xstring_format<char, const DWORD &>{
403  public:
404  inline static string format(const DWORD & value){
405  return std::to_string(value);
406  }
407  };
408 #endif
409  }
410 #endif
411 }
xstring & to_upper(const std::locale &loc)
Converts the string to upper case.
Definition: string.hpp:91
xstring & rtrim()
Trim trailing whitespace.
Definition: string.hpp:114
xstring & ltrim()
Trim leading whitespace.
Definition: string.hpp:99
xstring(_ArgsT &&...oArgs)
Generic constructor forwards everything to the base class.
Definition: string.hpp:53
std::vector< xstring< _ChT > > split(const std::initializer_list< _ChT > &delimiters, bool trimEmpty=false) const
splits the string by the specified delmiters into constituent elements
Definition: string.hpp:165
host, target and build configurations and settings Various components are purpose built for specific ...
xstring & to_lower(const std::locale &loc)
Converts the string to lower case.
Definition: string.hpp:80
Extends std::string with some added functionality.
Definition: string.hpp:30
size_type find_first_of(const std::initializer_list< _ChT > &delimiters, size_type pos=0) const
finds the first occurance of any item
Definition: string.hpp:153
static xstring format()
Type safe formatting Appends each item in the parameter list together performing type-safe verificati...
Definition: string.hpp:62