PCL
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Friends Macros Modules Pages
String.h
Go to the documentation of this file.
1 // ____ ______ __
2 // / __ \ / ____// /
3 // / /_/ // / / /
4 // / ____// /___ / /___ PixInsight Class Library
5 // /_/ \____//_____/ PCL 2.9.3
6 // ----------------------------------------------------------------------------
7 // pcl/String.h - Released 2025-02-21T12:13:32Z
8 // ----------------------------------------------------------------------------
9 // This file is part of the PixInsight Class Library (PCL).
10 // PCL is a multiplatform C++ framework for development of PixInsight modules.
11 //
12 // Copyright (c) 2003-2025 Pleiades Astrophoto S.L. All Rights Reserved.
13 //
14 // Redistribution and use in both source and binary forms, with or without
15 // modification, is permitted provided that the following conditions are met:
16 //
17 // 1. All redistributions of source code must retain the above copyright
18 // notice, this list of conditions and the following disclaimer.
19 //
20 // 2. All redistributions in binary form must reproduce the above copyright
21 // notice, this list of conditions and the following disclaimer in the
22 // documentation and/or other materials provided with the distribution.
23 //
24 // 3. Neither the names "PixInsight" and "Pleiades Astrophoto", nor the names
25 // of their contributors, may be used to endorse or promote products derived
26 // from this software without specific prior written permission. For written
27 // permission, please contact info@pixinsight.com.
28 //
29 // 4. All products derived from this software, in any form whatsoever, must
30 // reproduce the following acknowledgment in the end-user documentation
31 // and/or other materials provided with the product:
32 //
33 // "This product is based on software from the PixInsight project, developed
34 // by Pleiades Astrophoto and its contributors (https://pixinsight.com/)."
35 //
36 // Alternatively, if that is where third-party acknowledgments normally
37 // appear, this acknowledgment must be reproduced in the product itself.
38 //
39 // THIS SOFTWARE IS PROVIDED BY PLEIADES ASTROPHOTO AND ITS CONTRIBUTORS
40 // "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
41 // TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
42 // PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL PLEIADES ASTROPHOTO OR ITS
43 // CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
44 // EXEMPLARY OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, BUSINESS
45 // INTERRUPTION; PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; AND LOSS OF USE,
46 // DATA OR PROFITS) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
47 // CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
48 // ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
49 // POSSIBILITY OF SUCH DAMAGE.
50 // ----------------------------------------------------------------------------
51 
52 #ifndef __PCL_String_h
53 #define __PCL_String_h
54 
56 
57 #include <pcl/Defs.h>
58 #include <pcl/Diagnostics.h>
59 
60 #include <pcl/Allocator.h>
61 #include <pcl/Atomic.h>
62 #include <pcl/ByteArray.h>
63 #include <pcl/CharTraits.h>
64 #include <pcl/Container.h>
65 #include <pcl/Flags.h>
66 #include <pcl/Iterator.h>
67 #include <pcl/Math.h>
68 #include <pcl/ReferenceCounter.h>
69 #include <pcl/Sort.h>
70 #include <pcl/Utility.h>
71 
72 #include <stdarg.h>
73 
74 #ifdef __PCL_ALIGNED_STRING_ALLOCATION
75 # include <pcl/AlignedAllocator.h>
76 # define PCL_STRING_ALLOCATOR AlignedAllocator
77 #else
78 # include <pcl/StandardAllocator.h>
79 # define PCL_STRING_ALLOCATOR StandardAllocator
80 #endif
81 
82 #ifndef __PCL_NO_STRING_COMPLEX
83 # include <pcl/Complex.h>
84 #endif
85 
86 #ifndef __PCL_NO_STRING_OSTREAM
87 # ifndef _MSC_VER
88 # define _GLIBCXX_USE_WCHAR_T 1
89 # endif
90 # include <ostream>
91 #endif
92 
93 #ifdef __PCL_QT_INTERFACE
94 # include <QtCore/QByteArray>
95 # include <QtCore/QDateTime>
96 # include <QtCore/QString>
97 # define PCL_GET_CHARPTR_FROM_QSTRING( qs ) (qs.toLatin1().data())
98 # define PCL_GET_CHAR16PTR_FROM_QSTRING( qs ) ((char16_type*)( qs.utf16() ))
99 # define PCL_GET_CHARPTR_FROM_QSTRINGVIEW( qv ) ((const char*)qv.data())
100 # define PCL_GET_CHAR16PTR_FROM_QSTRINGVIEW( qv ) ((const char16_type*)qv.data())
101 # define PCL_GET_QSTRING_FROM_CHAR16PTR( s ) (QString::fromUtf16( (const char16_t*)(s) ))
102 # define PCL_GET_CHARPTR_FROM_QBYTEARRAY( qb ) (qb.data())
103 # define PCL_QDATE_FMT_STR "yyyy/MM/dd"
104 # define PCL_QDATETIME_FMT_STR "yyyy/MM/dd hh:mm:ss"
105 #endif
106 
107 namespace pcl
108 {
109 
110 #ifndef __PCL_NO_STRING_VECTOR
111  template <typename T> class PCL_CLASS GenericVector;
112 #endif
113 
114 // ----------------------------------------------------------------------------
115 
132 namespace RandomizationOption
133 {
134  enum mask_type
135  {
136  Lowercase = 0x00000001, // Generate random lowercase alphabetic characters: a..z
137  Uppercase = 0x00000002, // Generate random uppercase alphabetic characters: A..Z
138  Alpha = Lowercase|Uppercase, // Generate random alphabetic characters: a..zA..Z
139  Digits = 0x00000004, // Generate random decimal digits: 0..9
140  Symbols = 0x00000008, // Generate random symbol characters: .#/[]() etc
141  HexDigits = 0x00000010, // Generate hexadecimal digits: 0..9a..f or 0..9A..F
142  BinDigits = 0x00000020, // Generate binary digits: 0 and 1
143  FullRange = 0x80000000, // Generate random characters in the whole range of code points except zero
144  Default = Alpha|Digits
145  };
146 };
147 
152 using RandomizationOptions = Flags<RandomizationOption::mask_type>;
153 
154 // ----------------------------------------------------------------------------
155 
170 {
177  unsigned items : 2;
178 
183  unsigned precision : 4;
184 
189  bool sign : 1;
190 
198  unsigned width : 4;
199 
204  char separator;
205 
211  char padding;
212 
216  constexpr SexagesimalConversionOptions( unsigned items_ = 3,
217  unsigned precision_ = 2,
218  bool sign_ = false,
219  unsigned width_ = 0,
220  char separator_ = ':',
221  char padding_ = ' ' )
222  : items( items_ )
223  , precision( precision_ )
224  , sign( sign_ )
225  , width( width_ )
226  , separator( separator_ )
227  , padding( padding_ )
228  {
229  }
230 
235 
240 };
241 
242 // ----------------------------------------------------------------------------
243 
257 {
265  constexpr AngleConversionOptions( unsigned precision = 2, unsigned width = 3, char padding = ' ' )
266  : SexagesimalConversionOptions( 3/*items*/, precision, false/*sign*/, width, ' '/*separator*/, padding )
267  {}
268 };
269 
270 // ----------------------------------------------------------------------------
271 
286 {
295  constexpr LongitudeConversionOptions( unsigned precision = 2, unsigned width = 4, char padding = ' ' )
296  : SexagesimalConversionOptions( 3/*items*/, precision, true/*sign*/, width, ' '/*separator*/, padding )
297  {}
298 };
299 
300 // ----------------------------------------------------------------------------
301 
317 {
325  constexpr RAConversionOptions( unsigned precision = 3, unsigned width = 2, char padding = ' ' )
326  : SexagesimalConversionOptions( 3/*items*/, precision, false/*sign*/, width, ' '/*separator*/, padding )
327  {}
328 };
329 
330 // ----------------------------------------------------------------------------
331 
346 {
355  constexpr LatitudeConversionOptions( unsigned precision = 2, unsigned width = 3, char padding = ' ' )
356  : SexagesimalConversionOptions( 3/*items*/, precision, true/*sign*/, width, ' '/*separator*/, padding )
357  {}
358 };
359 
360 // ----------------------------------------------------------------------------
361 
375 {
384  constexpr DecConversionOptions( unsigned precision = 2, unsigned width = 3, char padding = ' ' )
385  : LatitudeConversionOptions( precision, width, padding )
386  {}
387 };
388 
389 // ----------------------------------------------------------------------------
390 
398 struct PCL_CLASS ISO8601ConversionOptions
399 {
406  unsigned timeItems : 2;
407 
412  unsigned precision : 4;
413 
418  bool timeZone : 1;
419 
424  bool zuluTime : 1;
425 
429  constexpr ISO8601ConversionOptions( unsigned timeItems_ = 3,
430  unsigned precision_ = 3,
431  bool timeZone_ = true,
432  bool zuluTime_ = true )
433  : timeItems( timeItems_ )
434  , precision( precision_ )
435  , timeZone( timeZone_ )
436  , zuluTime( zuluTime_ )
437  {
438  }
439 
444 
448  ISO8601ConversionOptions& operator =( const ISO8601ConversionOptions& ) = default;
449 };
450 
459 {
463  constexpr ISO8601ConversionOptionsNoTimeZone( unsigned timeItems_ = 3,
464  unsigned precision_ = 3 )
465  : ISO8601ConversionOptions( timeItems_, precision_, false/*timeZone*/, false/*zuluTime*/ )
466  {
467  }
468 };
469 
470 // ----------------------------------------------------------------------------
471 
491 template <class T, class R, class A = PCL_STRING_ALLOCATOR>
492 class PCL_CLASS GenericString : public DirectContainer<T>
493 {
494 public:
495 
499 
503 
507 
511  using char_type = T;
512 
516  using char_traits = R;
517 
521  using block_allocator = A;
522 
527 
531  using c_string = T*;
532 
536  using const_c_string = const T*;
537 
541  using iterator = T*;
542 
546  using const_iterator = const T*;
547 
552 
557 
558  // -------------------------------------------------------------------------
559 
563  static const size_type notFound = ~size_type( 0 );
564 
568  static const size_type maxPos = ~size_type( 0 );
569 
570  // -------------------------------------------------------------------------
571 
576  {
577  m_data = Data::New();
578  }
579 
584  : m_data( s.m_data )
585  {
586  m_data->Attach();
587  }
588 
593  : m_data( s.m_data )
594  {
595  s.m_data = nullptr;
596  }
597 
603  {
604  size_type len = R::Length( t );
605  if ( len > 0 )
606  {
607  m_data = Data::New( len );
608  R::Copy( m_data->string, t, len );
609  }
610  else
611  m_data = Data::New();
612  }
613 
632  {
633  if ( i < j )
634  {
635  size_type len = j - i;
636  m_data = Data::New( len );
637  R::Copy( m_data->string, i, len );
638  }
639  else
640  m_data = Data::New();
641  }
642 
651  GenericString( std::initializer_list<char_type> l )
652  : GenericString( l.begin(), l.end() )
653  {
654  }
655 
661  {
662  size_type len = R::Length( t );
663  if ( i < len && (n = pcl::Min( n, len-i )) > 0 )
664  {
665  m_data = Data::New( n );
666  R::Copy( m_data->string, t+i, n );
667  }
668  else
669  m_data = Data::New();
670  }
671 
676  {
677  if ( n > 0 )
678  {
679  m_data = Data::New( n );
680  R::Fill( m_data->string, c, n );
681  }
682  else
683  m_data = Data::New();
684  }
685 
691  {
692  if ( m_data != nullptr )
693  {
694  DetachFromData();
695  m_data = nullptr;
696  }
697  }
698 
704  bool IsUnique() const noexcept
705  {
706  return m_data->IsUnique();
707  }
708 
716  bool IsAliasOf( const GenericString& s ) const noexcept
717  {
718  return m_data == s.m_data;
719  }
720 
731  {
732  if ( !IsUnique() )
733  {
734  size_type len = Length();
735  Data* newData = Data::New( len );
736  if ( len > 0 )
737  R::Copy( newData->string, m_data->string, len );
738  DetachFromData();
739  m_data = newData;
740  }
741  }
742 
750  static size_type BytesPerChar() noexcept
751  {
752  return R::BytesPerChar();
753  }
754 
761  size_type Size() const noexcept
762  {
763  return Length()*BytesPerChar();
764  }
765 
772  size_type Length() const noexcept
773  {
774  return m_data->end - m_data->string;
775  }
776 
785  size_type Capacity() const noexcept
786  {
787  return m_data->capacity - m_data->string;
788  }
789 
797  size_type Available() const noexcept
798  {
799  return m_data->capacity - m_data->end;
800  }
801 
818  bool IsValid() const noexcept
819  {
820  return m_data != nullptr;
821  }
822 
830  bool IsEmpty() const noexcept
831  {
832  return m_data->string == m_data->end;
833  }
834 
843  size_type LowerBound() const noexcept
844  {
845  return 0;
846  }
847 
856  size_type UpperBound() const noexcept
857  {
858  PCL_PRECONDITION( !IsEmpty() )
859  return Length()-1;
860  }
861 
867  const allocator& Allocator() const noexcept
868  {
869  return m_data->alloc;
870  }
871 
881  void SetAllocator( const allocator& a )
882  {
883  EnsureUnique();
884  m_data->alloc = a;
885  }
886 
900  {
901  PCL_PRECONDITION( i < Length() )
902  EnsureUnique();
903  return m_data->string + i;
904  }
905 
915  const_iterator At( size_type i ) const noexcept
916  {
917  PCL_PRECONDITION( i < Length() )
918  return m_data->string + i;
919  }
920 
933  char_type& operator []( size_type i )
934  {
935  return *At( i );
936  }
937 
947  char_type operator []( size_type i ) const noexcept
948  {
949  return *At( i );
950  }
951 
960  {
961  EnsureUnique();
962  return *m_data->string;
963  }
964 
972  char_type operator *() const noexcept
973  {
974  return *m_data->string;
975  }
976 
989  {
990  EnsureUnique();
991  return m_data->string;
992  }
993 
1002  const_iterator Begin() const noexcept
1003  {
1004  return m_data->string;
1005  }
1006 
1019  {
1020  EnsureUnique();
1021  return m_data->end;
1022  }
1023 
1032  const_iterator End() const noexcept
1033  {
1034  return m_data->end;
1035  }
1036 
1051  {
1052  EnsureUnique();
1053  return (m_data->string < m_data->end) ? m_data->end-1 : nullptr;
1054  }
1055 
1067  {
1068  return (m_data->string < m_data->end) ? m_data->end-1 : nullptr;
1069  }
1070 
1085  {
1086  EnsureUnique();
1087  return (m_data->string < m_data->end) ? m_data->string-1 : nullptr;
1088  }
1089 
1101  {
1102  return (m_data->string < m_data->end) ? m_data->string-1 : nullptr;
1103  }
1104 
1115  size_type IndexAt( const_iterator i ) const noexcept
1116  {
1117  PCL_PRECONDITION( i >= m_data->string )
1118  return i - m_data->string;
1119  }
1120 
1121 #ifndef __PCL_NO_STL_COMPATIBLE_ITERATORS
1126  {
1127  return Begin();
1128  }
1129 
1133  const_iterator begin() const noexcept
1134  {
1135  return Begin();
1136  }
1137 
1142  {
1143  return End();
1144  }
1145 
1149  const_iterator end() const noexcept
1150  {
1151  return End();
1152  }
1153 #endif // !__PCL_NO_STL_COMPATIBLE_ITERATORS
1154 
1162  const_c_string c_str() const noexcept
1163  {
1164  static const char_type theNullChar = char_traits::Null();
1165  return IsEmpty() ? &theNullChar : Begin();
1166  }
1167 
1173  GenericString& operator =( const GenericString& s )
1174  {
1175  Assign( s );
1176  return *this;
1177  }
1178 
1193  void Assign( const GenericString& s )
1194  {
1195  s.m_data->Attach();
1196  DetachFromData();
1197  m_data = s.m_data;
1198  }
1199 
1205  GenericString& operator =( GenericString&& s )
1206  {
1207  Transfer( s );
1208  return *this;
1209  }
1210 
1225  {
1226  if ( &s != this )
1227  {
1228  DetachFromData();
1229  m_data = s.m_data;
1230  s.m_data = nullptr;
1231  }
1232  }
1233 
1240  {
1241  if ( &s != this )
1242  {
1243  DetachFromData();
1244  m_data = s.m_data;
1245  s.m_data = nullptr;
1246  }
1247  }
1248 
1254  {
1255  Assign( t );
1256  return *this;
1257  }
1258 
1263  GenericString& operator =( char_type c )
1264  {
1265  Assign( c, 1 );
1266  return *this;
1267  }
1268 
1273  void Assign( const GenericString& s, size_type i, size_type n )
1274  {
1275  Transfer( s.Substring( i, n ) );
1276  }
1277 
1286  {
1287  size_type len = R::Length( t );
1288  if ( len > 0 )
1289  {
1290  MaybeReallocate( len );
1291  R::Copy( m_data->string, t, len );
1292  }
1293  else
1294  Clear();
1295  }
1296 
1318  {
1319  if ( i < j )
1320  {
1321  size_type len = j - i;
1322  MaybeReallocate( len );
1323  R::Copy( m_data->string, i, len );
1324  }
1325  else
1326  Clear();
1327  }
1328 
1335  void Assign( std::initializer_list<char_type> l )
1336  {
1337  Assign( l.begin(), l.end() );
1338  }
1339 
1354  {
1355  size_type len = R::Length( t );
1356  if ( i < len && (n = pcl::Min( n, len-i )) > 0 )
1357  {
1358  MaybeReallocate( n );
1359  R::Copy( m_data->string, t+i, n );
1360  }
1361  else
1362  Clear();
1363  }
1364 
1370  void Assign( char_type c, size_type n = 1 )
1371  {
1372  if ( !R::IsNull( c ) && n > 0 )
1373  {
1374  MaybeReallocate( n );
1375  R::Fill( m_data->string, c, n );
1376  }
1377  else
1378  Clear();
1379  }
1380 
1384  void Swap( GenericString& s ) noexcept
1385  {
1386  pcl::Swap( m_data, s.m_data );
1387  }
1388 
1398  void Fill( char_type c )
1399  {
1400  size_type len = Length();
1401  if ( len > 0 )
1402  {
1403  if ( !IsUnique() )
1404  {
1405  Data* newData = Data::New( len );
1406  DetachFromData();
1407  m_data = newData;
1408  }
1409  R::Fill( m_data->string, c, len );
1410  }
1411  }
1412 
1427  void Fill( char_type c, size_type i, size_type n = maxPos )
1428  {
1429  size_type len = Length();
1430  if ( i < len )
1431  {
1432  n = pcl::Min( n, len-i );
1433  if ( n < len )
1434  EnsureUnique();
1435  else if ( !IsUnique() )
1436  {
1437  Data* newData = Data::New( len );
1438  DetachFromData();
1439  m_data = newData;
1440  }
1441  R::Fill( m_data->string+i, c, n );
1442  }
1443  }
1444 
1466  void SecureFill( char c = '\0' ) noexcept
1467  {
1468  volatile char* s = reinterpret_cast<volatile char*>( m_data->string );
1469  for ( size_type n = Capacity()*sizeof( char_type ); n > 0; --n )
1470  *s++ = c;
1471  }
1472 
1480  void Reserve( size_type n )
1481  {
1482  if ( n > 0 )
1483  if ( IsUnique() )
1484  {
1485  if ( Capacity() < n+1 )
1486  {
1487  iterator old = m_data->string;
1488  size_type len = Length();
1489  m_data->Reserve( len, n );
1490  if ( old != nullptr )
1491  {
1492  if ( len > 0 )
1493  R::Copy( m_data->string, old, len );
1494  m_data->alloc.Deallocate( old );
1495  }
1496  }
1497  }
1498  else
1499  {
1500  size_type len = Length();
1501  Data* newData = Data::New(); // ### FIXME: unlikely, but this is a potential leak
1502  newData->Reserve( len, pcl::Max( len, n ) );
1503  if ( len > 0 )
1504  R::Copy( newData->string, m_data->string, len );
1505  DetachFromData();
1506  m_data = newData;
1507  }
1508  }
1509 
1524  {
1525  if ( n != Length() )
1526  {
1527  if ( n > 0 )
1528  {
1529  if ( !IsUnique() || m_data->ShouldReallocate( n ) )
1530  {
1531  Data* newData = Data::New( n );
1532  size_type m = Capacity();
1533  if ( m > 0 )
1534  R::Copy( newData->string, m_data->string, pcl::Min( n, m ) );
1535  DetachFromData();
1536  m_data = newData;
1537  }
1538  else
1539  m_data->SetLength( n );
1540  }
1541  else
1542  Clear();
1543  }
1544  else
1545  EnsureUnique();
1546  }
1547 
1554  {
1555  // NB: This can be done more efficiently if this string is unique.
1556  GenericString s( *this );
1557  s.SetLength( n );
1558  return s;
1559  }
1560 
1570  {
1571  SetLength( R::Length( m_data->string ) );
1572  }
1573 
1579  {
1580  GenericString s( *this );
1582  return s;
1583  }
1584 
1597  void Squeeze()
1598  {
1599  if ( Available() > 0 )
1600  {
1601  size_type len = Length();
1602  if ( len > 0 )
1603  {
1604  if ( IsUnique() )
1605  {
1606  iterator old = m_data->string;
1607  m_data->Allocate( len, 0 );
1608  R::Copy( m_data->string, old, len );
1609  m_data->alloc.Deallocate( old );
1610  }
1611  else
1612  {
1613  Data* newData = Data::New( len, 0 );
1614  R::Copy( newData->string, m_data->string, len );
1615  DetachFromData();
1616  m_data = newData;
1617  }
1618  }
1619  else
1620  Clear();
1621  }
1622  }
1623 
1629  {
1630  // NB: This can be done more efficiently if this string is unique.
1631  GenericString s( *this );
1632  s.Squeeze();
1633  return s;
1634  }
1635 
1649  {
1650  EnsureUnique();
1651  iterator string = m_data->string;
1652  m_data->Reset();
1653  return string;
1654  }
1655 
1672  void c_copy( iterator dst, size_type maxCharsToCopy, size_type i = 0 ) const noexcept
1673  {
1674  if ( dst != nullptr )
1675  if ( maxCharsToCopy > 0 )
1676  {
1677  if ( --maxCharsToCopy > 0 )
1678  {
1679  size_type len = Length();
1680  if ( i < len )
1681  {
1682  size_type n = pcl::Min( maxCharsToCopy, len-i );
1683  R::Copy( dst, m_data->string+i, n );
1684  dst += n;
1685  }
1686  }
1687 
1688  *dst = R::Null();
1689  }
1690  }
1691 
1695  template <class R1, class A1>
1697  {
1698  size_type n = s.Length();
1699  if ( n > 0 )
1700  {
1701  UninitializedGrow( i, n ); // -> 0 <= i <= len
1702  R::Copy( m_data->string+i, s.m_data->string, n );
1703  }
1704  }
1705 
1714  {
1715  if ( p < q )
1716  {
1717  size_type n = q - p;
1718  UninitializedGrow( i, n ); // -> 0 <= i <= len
1719  R::Copy( m_data->string+i, p, n );
1720  }
1721  }
1722 
1728  {
1729  size_type n = R::Length( t );
1730  if ( n > 0 )
1731  {
1732  UninitializedGrow( i, n ); // -> 0 <= i <= len
1733  R::Copy( m_data->string+i, t, n );
1734  }
1735  }
1736 
1742  {
1743  if ( (n = pcl::Min( n, R::Length( t ) )) > 0 )
1744  {
1745  UninitializedGrow( i, n ); // -> 0 <= i <= len
1746  R::Copy( m_data->string+i, t, n );
1747  }
1748  }
1749 
1753  void Insert( size_type i, char_type c, size_type n = 1 )
1754  {
1755  if ( n > 0 )
1756  {
1757  UninitializedGrow( i, n ); // -> 0 <= i <= len
1758  R::Fill( m_data->string+i, c, n );
1759  }
1760  }
1761 
1765  template <class R1, class A1>
1767  {
1768  Insert( maxPos, s );
1769  }
1770 
1775  template <class R1, class A1>
1776  GenericString& operator +=( const GenericString<T,R1,A1>& s )
1777  {
1778  Append( s );
1779  return *this;
1780  }
1781 
1790  {
1791  Insert( maxPos, i, j );
1792  }
1793 
1799  {
1800  Insert( maxPos, t, n );
1801  }
1802 
1807  {
1808  Insert( maxPos, t );
1809  }
1810 
1815  GenericString& operator +=( const_c_string t )
1816  {
1817  Append( t );
1818  return *this;
1819  }
1820 
1824  void Append( char_type c, size_type n = 1 )
1825  {
1826  Insert( maxPos, c, n );
1827  }
1828 
1833  template <class R1, class A1>
1834  void Add( const GenericString<T,R1,A1>& s )
1835  {
1836  Append( s );
1837  }
1838 
1844  {
1845  Append( i, j );
1846  }
1847 
1853  {
1854  Append( t, n );
1855  }
1856 
1862  {
1863  Append( t );
1864  }
1865 
1870  void Add( char_type c, size_type n = 1 )
1871  {
1872  Append( c, n );
1873  }
1874 
1879  GenericString& operator +=( char_type c )
1880  {
1881  Append( c );
1882  return *this;
1883  }
1884 
1889  template <class R1, class A1>
1891  {
1892  Insert( 0, s );
1893  }
1894 
1899  template <class R1, class A1>
1900  GenericString& operator -=( const GenericString<T,R1,A1>& s )
1901  {
1902  Prepend( s );
1903  return *this;
1904  }
1905 
1914  {
1915  Insert( 0, i, j );
1916  }
1917 
1923  {
1924  Insert( 0, t, n );
1925  }
1926 
1932  {
1933  Insert( 0, t );
1934  }
1935 
1940  GenericString& operator -=( const_c_string t )
1941  {
1942  Prepend( t );
1943  return *this;
1944  }
1945 
1949  void Prepend( char_type c, size_type n = 1 )
1950  {
1951  Insert( 0, c, n );
1952  }
1953 
1958  GenericString& operator -=( char_type c )
1959  {
1960  Prepend( c );
1961  return *this;
1962  }
1963 
1968  template <class R1, class A1>
1970  {
1971  if ( n > 0 )
1972  {
1973  size_type len = Length();
1974  if ( i < len )
1975  {
1976  n = pcl::Min( n, len-i );
1977  if ( n != len )
1978  {
1979  size_type ns = s.Length();
1980  if ( ns > 0 )
1981  {
1982  if ( n < ns )
1983  UninitializedGrow( i, ns-n );
1984  else if ( ns < n )
1985  Delete( i, n-ns );
1986  else
1987  EnsureUnique();
1988 
1989  R::Copy( m_data->string+i, s.m_data->string, ns );
1990  }
1991  else
1992  Delete( i, n );
1993  }
1994  else
1995  Assign( s );
1996  }
1997  }
1998  }
1999 
2009  {
2010  if ( n > 0 )
2011  {
2012  size_type len = Length();
2013  if ( i < len )
2014  {
2015  n = pcl::Min( n, len-i );
2016  if ( n != len )
2017  {
2018  size_type nt = R::Length( t );
2019  if ( nt > 0 )
2020  {
2021  if ( n < nt )
2022  UninitializedGrow( i, nt-n );
2023  else if ( nt < n )
2024  Delete( i, n-nt );
2025  else
2026  EnsureUnique();
2027 
2028  R::Copy( m_data->string+i, t, nt );
2029  }
2030  else
2031  Delete( i, n );
2032  }
2033  else
2034  Assign( t );
2035  }
2036  }
2037  }
2038 
2044  {
2045  if ( n > 0 )
2046  {
2047  size_type len = Length();
2048  if ( i < len )
2049  {
2050  n = pcl::Min( n, len-i );
2051  if ( n != len )
2052  {
2053  if ( nc > 0 )
2054  {
2055  if ( n < nc )
2056  UninitializedGrow( i, nc-n );
2057  else if ( nc < n )
2058  Delete( i, n-nc );
2059  else
2060  EnsureUnique();
2061 
2062  R::Fill( m_data->string+i, c, nc );
2063  }
2064  else
2065  Delete( i, n );
2066  }
2067  else
2068  Assign( c, nc );
2069  }
2070  }
2071  }
2072 
2078  void ReplaceChar( char_type c1, char_type c2, size_type i = 0, size_type n = maxPos )
2079  {
2080  ReplaceChar( c1, c2, i, n, true/*case*/ );
2081  }
2082 
2090  void ReplaceCharIC( char_type c1, char_type c2, size_type i = 0, size_type n = maxPos )
2091  {
2092  ReplaceChar( c1, c2, i, n, false/*case*/ );
2093  }
2094 
2100  template <class R1, class A1, class R2, class A2>
2102  const GenericString<T,R2,A2>& s2, size_type i = 0 )
2103  {
2104  ReplaceString( s1.m_data->string, s1.Length(), s2.m_data->string, s2.Length(), i, true/*case*/ );
2105  }
2106 
2113  {
2114  ReplaceString( t1, R::Length( t1 ), t2, R::Length( t2 ), i, true/*case*/ );
2115  }
2116 
2124  template <class R1, class A1, class R2, class A2>
2126  const GenericString<T,R2,A2>& s2, size_type i = 0 )
2127  {
2128  ReplaceString( s1.m_data->string, s1.Length(), s2.m_data->string, s2.Length(), i, false/*case*/ );
2129  }
2130 
2139  {
2140  ReplaceString( t1, R::Length( t1 ), t2, R::Length( t2 ), i, false/*case*/ );
2141  }
2142 
2147  void Delete( size_type i, size_type n = 1 )
2148  {
2149  if ( n > 0 )
2150  {
2151  size_type len = Length();
2152  if ( i < len )
2153  {
2154  n = pcl::Min( n, len-i );
2155  if ( n < len )
2156  {
2157  size_type newLen = len-n;
2158  if ( !IsUnique() || m_data->ShouldReallocate( newLen ) )
2159  {
2160  Data* newData = Data::New( newLen );
2161  if ( i > 0 )
2162  R::Copy( newData->string, m_data->string, i );
2163  if ( i < newLen )
2164  R::Copy( newData->string+i, m_data->string+i+n, newLen-i );
2165  DetachFromData();
2166  m_data = newData;
2167  }
2168  else
2169  {
2170  if ( i < newLen )
2171  R::CopyOverlapped( m_data->string+i, m_data->string+i+n, newLen-i );
2172  m_data->SetLength( newLen );
2173  }
2174  }
2175  else
2176  Clear();
2177  }
2178  }
2179  }
2180 
2186  {
2187  Delete( i, maxPos );
2188  }
2189 
2195  {
2196  Delete( 0, i );
2197  }
2198 
2204  void DeleteChar( char_type c, size_type i = 0 )
2205  {
2206  ReplaceString( &c, 1, nullptr, 0, i, true/*case*/ );
2207  }
2208 
2217  {
2218  ReplaceString( &c, 1, nullptr, 0, i, false/*case*/ );
2219  }
2220 
2226  template <class R1, class A1>
2228  {
2229  ReplaceString( s.m_data->string, s.Length(), nullptr, 0, i, true/*case*/ );
2230  }
2231 
2238  {
2239  ReplaceString( t, R::Length( t ), nullptr, 0, i, true/*case*/ );
2240  }
2241 
2249  template <class R1, class A1>
2251  {
2252  ReplaceString( s.m_data->string, s.Length(), nullptr, 0, i, false/*case*/ );
2253  }
2254 
2263  {
2264  ReplaceString( t, R::Length( t ), nullptr, 0, i, false/*case*/ );
2265  }
2266 
2276  void Clear()
2277  {
2278  if ( !IsEmpty() )
2279  if ( IsUnique() )
2280  m_data->Deallocate();
2281  else
2282  {
2283  Data* newData = Data::New();
2284  DetachFromData();
2285  m_data = newData;
2286  }
2287  }
2288 
2294  {
2295  size_type len = Length();
2296  if ( i < len )
2297  {
2298  n = pcl::Min( n, len-i );
2299  if ( n > 0 )
2300  {
2301  if ( n < len )
2302  {
2303  GenericString t;
2304  t.m_data->Allocate( n );
2305  R::Copy( t.m_data->string, m_data->string+i, n );
2306  return t;
2307  }
2308  return *this;
2309  }
2310  }
2311  return GenericString();
2312  }
2313 
2319  {
2320  size_type len = Length();
2321  n = pcl::Min( n, len );
2322  if ( n > 0 )
2323  {
2324  if ( n < len )
2325  {
2326  GenericString t;
2327  t.m_data->Allocate( n );
2328  R::Copy( t.m_data->string, m_data->string, n );
2329  return t;
2330  }
2331  return *this;
2332  }
2333  return GenericString();
2334  }
2335 
2341  {
2342  size_type len = Length();
2343  n = pcl::Min( n, len );
2344  if ( n > 0 )
2345  {
2346  if ( n < len )
2347  {
2348  GenericString t;
2349  t.m_data->Allocate( n );
2350  R::Copy( t.m_data->string, m_data->string+len-n, n );
2351  return t;
2352  }
2353  return *this;
2354  }
2355  return GenericString();
2356  }
2357 
2365  {
2366  return Substring( i );
2367  }
2368 
2376  {
2377  return Left( (i > 0) ? i-1 : 0 );
2378  }
2379 
2397  template <class C, class R1, class A1>
2398  size_type Break( C& list, const GenericString<T,R1,A1>& s, bool trim = false, size_type i = 0 ) const
2399  {
2400  size_type count = 0;
2401  size_type len = Length();
2402  if ( i < len )
2403  {
2404  size_type n = s.Length();
2405  if ( n > 0 )
2406  for ( SearchEngine S( s.m_data->string, n, true/*case*/ ); ; )
2407  {
2408  size_type j = S( m_data->string, i, len );
2409 
2410  GenericString t;
2411  if ( i < j )
2412  {
2413  const_iterator l = m_data->string + i;
2414  size_type m = j - i;
2415  if ( trim )
2416  {
2417  const_iterator r = l + m;
2418  l = R::SearchTrimLeft( l, r );
2419  m = R::SearchTrimRight( l, r ) - l;
2420  }
2421  if ( m > 0 )
2422  {
2423  t.m_data->Allocate( m );
2424  R::Copy( t.m_data->string, l, m );
2425  }
2426  }
2427  list.Add( t );
2428  ++count;
2429 
2430  if ( j == len )
2431  break;
2432 
2433  i = j + n;
2434  }
2435  }
2436  return count;
2437  }
2438 
2457  template <class C>
2458  size_type Break( C& list, const_c_string s, bool trim = false, size_type i = 0 ) const
2459  {
2460  size_type count = 0;
2461  size_type len = Length();
2462  if ( i < len )
2463  {
2464  size_type n = R::Length( s );
2465  if ( n > 0 )
2466  for ( SearchEngine S( s, n, true/*case*/ ); ; )
2467  {
2468  size_type j = S( m_data->string, i, len );
2469 
2470  GenericString t;
2471  if ( i < j )
2472  {
2473  const_iterator l = m_data->string + i;
2474  size_type m = j - i;
2475  if ( trim )
2476  {
2477  const_iterator r = l + m;
2478  l = R::SearchTrimLeft( l, r );
2479  m = R::SearchTrimRight( l, r ) - l;
2480  }
2481  if ( m > 0 )
2482  {
2483  t.m_data->Allocate( m );
2484  R::Copy( t.m_data->string, l, m );
2485  }
2486  }
2487  list.Add( t );
2488  ++count;
2489 
2490  if ( j == len )
2491  break;
2492 
2493  i = j + n;
2494  }
2495  }
2496  return count;
2497  }
2498 
2517  template <class C>
2518  size_type Break( C& list, char_type c, bool trim = false, size_type i = 0 ) const
2519  {
2520  size_type count = 0;
2521  size_type len = Length();
2522  if ( i < len )
2523  {
2524  for ( ;; )
2525  {
2526  size_type j = i;
2527  for ( const_iterator p = m_data->string + i; j < len; ++j, ++p )
2528  if ( *p == c )
2529  break;
2530 
2531  GenericString t;
2532  if ( i < j )
2533  {
2534  const_iterator l = m_data->string + i;
2535  size_type m = j - i;
2536  if ( trim )
2537  {
2538  const_iterator r = l + m;
2539  l = R::SearchTrimLeft( l, r );
2540  m = R::SearchTrimRight( l, r ) - l;
2541  }
2542  if ( m > 0 )
2543  {
2544  t.m_data->Allocate( m );
2545  R::Copy( t.m_data->string, l, m );
2546  }
2547  }
2548  list.Add( typename C::item_type( t ) );
2549  ++count;
2550 
2551  if ( j == len )
2552  break;
2553 
2554  i = j + 1;
2555  }
2556  }
2557  return count;
2558  }
2559 
2582  template <class C, typename S>
2583  size_type Break( C& list, const Array<S>& ca, bool trim = false, size_type i = 0 ) const
2584  {
2585  size_type count = 0;
2586  if ( !ca.IsEmpty() )
2587  {
2588  size_type len = Length();
2589  if ( i < len )
2590  {
2591  for ( ;; )
2592  {
2593  size_type j = i;
2594  for ( const_iterator p = m_data->string + i; j < len; ++j, ++p )
2595  for ( auto c : ca )
2596  if ( *p == char_type( c ) )
2597  goto __Break_1;
2598 __Break_1:
2599  GenericString t;
2600  if ( i < j )
2601  {
2602  const_iterator l = m_data->string + i;
2603  size_type m = j - i;
2604  if ( trim )
2605  {
2606  const_iterator r = l + m;
2607  l = R::SearchTrimLeft( l, r );
2608  m = R::SearchTrimRight( l, r ) - l;
2609  }
2610  if ( m > 0 )
2611  {
2612  t.m_data->Allocate( m );
2613  R::Copy( t.m_data->string, l, m );
2614  }
2615  }
2616  list.Add( typename C::item_type( t ) );
2617  ++count;
2618 
2619  if ( j == len )
2620  break;
2621 
2622  i = j + 1;
2623  }
2624  }
2625  }
2626  return count;
2627  }
2628 
2649  template <class C, class R1, class A1>
2650  size_type BreakIC( C& list, const GenericString<T,R1,A1>& s, bool trim = false, size_type i = 0 ) const
2651  {
2652  size_type count = 0;
2653  size_type len = Length();
2654  if ( i < len )
2655  {
2656  size_type n = s.Length();
2657  if ( n > 0 )
2658  {
2659  for ( SearchEngine S( s.m_data->string, n, false/*case*/ ); ; )
2660  {
2661  size_type j = S( m_data->string, i, len );
2662 
2663  GenericString t;
2664  if ( i < j )
2665  {
2666  const_iterator l = m_data->string + i;
2667  size_type m = j - i;
2668  if ( trim )
2669  {
2670  const_iterator r = l + m;
2671  l = R::SearchTrimLeft( l, r );
2672  m = R::SearchTrimRight( l, r ) - l;
2673  }
2674  if ( m > 0 )
2675  {
2676  t.m_data->Allocate( m );
2677  R::Copy( t.m_data->string, l, m );
2678  }
2679  }
2680  list.Add( typename C::item_type( t ) );
2681  ++count;
2682 
2683  if ( j == len )
2684  break;
2685 
2686  i = j + n;
2687  }
2688  }
2689  }
2690  return count;
2691  }
2692 
2715  template <class C>
2716  size_type BreakIC( C& list, const_c_string s, bool trim = false, size_type i = 0 ) const
2717  {
2718  size_type count = 0;
2719  size_type len = Length();
2720  if ( i < len )
2721  {
2722  size_type n = R::Length( s );
2723  if ( n > 0 )
2724  {
2725  for ( SearchEngine S( s, n, false/*case*/ ); ; )
2726  {
2727  size_type j = S( m_data->string, i, len );
2728 
2729  GenericString t;
2730  if ( i < j )
2731  {
2732  const_iterator l = m_data->string + i;
2733  size_type m = j - i;
2734  if ( trim )
2735  {
2736  const_iterator r = l + m;
2737  l = R::SearchTrimLeft( l, r );
2738  m = R::SearchTrimRight( l, r ) - l;
2739  }
2740  if ( m > 0 )
2741  {
2742  t.m_data->Allocate( m );
2743  R::Copy( t.m_data->string, l, m );
2744  }
2745  }
2746  list.Add( typename C::item_type( t ) );
2747  ++count;
2748 
2749  if ( j == len )
2750  break;
2751 
2752  i = j + n;
2753  }
2754  }
2755  }
2756  return count;
2757  }
2758 
2780  template <class C>
2781  size_type BreakIC( C& list, char_type c, bool trim = false, size_type i = 0 ) const
2782  {
2783  size_type count = 0;
2784  size_type len = Length();
2785  if ( i < len )
2786  {
2787  c = R::ToCaseFolded( c );
2788  for ( ;; )
2789  {
2790  size_type j = i;
2791  for ( const_iterator p = m_data->string + i; j < len; ++j, ++p )
2792  if ( R::ToCaseFolded( *p ) == c )
2793  break;
2794 
2795  GenericString t;
2796  if ( i < j )
2797  {
2798  const_iterator l = m_data->string + i;
2799  size_type m = j - i;
2800  if ( trim )
2801  {
2802  const_iterator r = l + m;
2803  l = R::SearchTrimLeft( l, r );
2804  m = R::SearchTrimRight( l, r ) - l;
2805  }
2806  if ( m > 0 )
2807  {
2808  t.m_data->Allocate( m );
2809  R::Copy( t.m_data->string, l, m );
2810  }
2811  }
2812  list.Add( typename C::item_type( t ) );
2813  ++count;
2814 
2815  if ( j == len )
2816  break;
2817 
2818  i = j + 1;
2819  }
2820  }
2821  return count;
2822  }
2823 
2828  char_type FirstChar() const noexcept
2829  {
2830  return (m_data->string != nullptr) ? *m_data->string : R::Null();
2831  }
2832 
2837  char_type LastChar() const noexcept
2838  {
2839  return (m_data->string < m_data->end) ? *(m_data->end-1) : R::Null();
2840  }
2841 
2845  template <class R1, class A1>
2846  bool StartsWith( const GenericString<T,R1,A1>& s ) const noexcept
2847  {
2848  if ( s.IsEmpty() || Length() < s.Length() )
2849  return false;
2850  for ( const_iterator p = m_data->string, q = s.m_data->string; q < s.m_data->end; ++p, ++q )
2851  if ( *p != *q )
2852  return false;
2853  return true;
2854  }
2855 
2860  bool StartsWith( const_c_string t ) const noexcept
2861  {
2862  size_type n = R::Length( t );
2863  if ( n == 0 || Length() < n )
2864  return false;
2865  for ( const_iterator p = m_data->string, e = p+n; p < e; ++p, ++t )
2866  if ( *p != *t )
2867  return false;
2868  return true;
2869  }
2870 
2874  bool StartsWith( char_type c ) const noexcept
2875  {
2876  return m_data->string < m_data->end && *m_data->string == c;
2877  }
2878 
2883  template <class R1, class A1>
2884  bool StartsWithIC( const GenericString<T,R1,A1>& s ) const noexcept
2885  {
2886  if ( s.IsEmpty() || Length() < s.Length() )
2887  return false;
2888  for ( const_iterator p = m_data->string, q = s.m_data->string; q < s.m_data->end; ++p, ++q )
2889  if ( R::ToCaseFolded( *p ) != R::ToCaseFolded( *q ) )
2890  return false;
2891  return true;
2892  }
2893 
2898  bool StartsWithIC( const_c_string t ) const noexcept
2899  {
2900  size_type n = R::Length( t );
2901  if ( n == 0 || Length() < n )
2902  return false;
2903  for ( const_iterator p = m_data->string, e = p+n; p < e; ++p, ++t )
2904  if ( R::ToCaseFolded( *p ) != R::ToCaseFolded( *t ) )
2905  return false;
2906  return true;
2907  }
2908 
2913  bool StartsWithIC( char_type c ) const noexcept
2914  {
2915  return m_data->string < m_data->end && R::ToCaseFolded( *m_data->string ) == R::ToCaseFolded( c );
2916  }
2917 
2921  template <class R1, class A1>
2922  bool EndsWith( const GenericString<T,R1,A1>& s ) const noexcept
2923  {
2924  size_type n = s.Length();
2925  if ( n == 0 || Length() < n )
2926  return false;
2927  for ( const_iterator p = m_data->end-n, q = s.m_data->string; p < m_data->end; ++p, ++q )
2928  if ( *p != *q )
2929  return false;
2930  return true;
2931  }
2932 
2937  bool EndsWith( const_c_string t ) const noexcept
2938  {
2939  size_type n = R::Length( t );
2940  if ( n == 0 || Length() < n )
2941  return false;
2942  for ( const_iterator p = m_data->end-n; p < m_data->end; ++p, ++t )
2943  if ( *p != *t )
2944  return false;
2945  return true;
2946  }
2947 
2951  bool EndsWith( char_type c ) const noexcept
2952  {
2953  return m_data->string < m_data->end && *(m_data->end-1) == c;
2954  }
2955 
2960  template <class R1, class A1>
2961  bool EndsWithIC( const GenericString<T,R1,A1>& s ) const noexcept
2962  {
2963  size_type n = s.Length();
2964  if ( n == 0 || Length() < n )
2965  return false;
2966  for ( const_iterator p = m_data->end-n, q = s.m_data->string; p < m_data->end; ++p, ++q )
2967  if ( R::ToCaseFolded( *p ) != R::ToCaseFolded( *q ) )
2968  return false;
2969  return true;
2970  }
2971 
2976  bool EndsWithIC( const_c_string t ) const noexcept
2977  {
2978  size_type n = R::Length( t );
2979  if ( n == 0 || Length() < n )
2980  return false;
2981  for ( const_iterator p = m_data->end-n; p < m_data->end; ++p, ++t )
2982  if ( R::ToCaseFolded( *p ) != R::ToCaseFolded( *t ) )
2983  return false;
2984  return true;
2985  }
2986 
2991  bool EndsWithIC( char_type c ) const noexcept
2992  {
2993  return m_data->string < m_data->end && R::ToCaseFolded( *(m_data->end-1) ) == R::ToCaseFolded( c );
2994  }
2995 
3001  template <class R1, class A1>
3002  size_type FindFirst( const GenericString<T,R1,A1>& s, size_type i = 0 ) const noexcept
3003  {
3004  size_type len = Length();
3005  i = SearchEngine( s.m_data->string, s.Length(), true/*case*/ )( m_data->string, i, len );
3006  return (i < len) ? i : notFound;
3007  }
3008 
3014  size_type FindFirst( const_c_string t, size_type i = 0 ) const noexcept
3015  {
3016  size_type len = Length();
3017  i = SearchEngine( t, R::Length( t ), true/*case*/ )( m_data->string, i, len );
3018  return (i < len) ? i : notFound;
3019  }
3020 
3026  size_type FindFirst( char_type c, size_type i = 0 ) const noexcept
3027  {
3028  for ( const_iterator p = m_data->string+i; p < m_data->end; ++p )
3029  if ( *p == c )
3030  return p - m_data->string;
3031  return notFound;
3032  }
3033 
3042  template <class R1, class A1>
3043  size_type FindFirstIC( const GenericString<T,R1,A1>& s, size_type i = 0 ) const noexcept
3044  {
3045  size_type len = Length();
3046  i = SearchEngine( s.m_data->string, s.Length(), false/*case*/ )( m_data->string, i, len );
3047  return (i < len) ? i : notFound;
3048  }
3049 
3058  size_type FindFirstIC( const_c_string t, size_type i = 0 ) const noexcept
3059  {
3060  size_type len = Length();
3061  i = SearchEngine( t, R::Length( t ), false/*case*/ )( m_data->string, i, len );
3062  return (i < len) ? i : notFound;
3063  }
3064 
3073  size_type FindFirstIC( char_type c, size_type i = 0 ) const noexcept
3074  {
3075  c = R::ToCaseFolded( c );
3076  for ( const_iterator p = m_data->string+i; p < m_data->end; ++p )
3077  if ( R::ToCaseFolded( *p ) == c )
3078  return p - m_data->string;
3079  return notFound;
3080  }
3081 
3085  template <class R1, class A1>
3086  size_type Find( const GenericString<T,R1,A1>& s, size_type i = 0 ) const noexcept
3087  {
3088  return FindFirst( s, i );
3089  }
3090 
3094  size_type Find( const_c_string t, size_type i = 0 ) const noexcept
3095  {
3096  return FindFirst( t, i );
3097  }
3098 
3102  size_type Find( char_type c, size_type i = 0 ) const noexcept
3103  {
3104  return FindFirst( c, i );
3105  }
3106 
3110  template <class R1, class A1>
3111  size_type FindIC( const GenericString<T,R1,A1>& s, size_type i = 0 ) const noexcept
3112  {
3113  return FindFirstIC( s, i );
3114  }
3115 
3119  size_type FindIC( const_c_string t, size_type i = 0 ) const noexcept
3120  {
3121  return FindFirstIC( t, i );
3122  }
3123 
3127  size_type FindIC( char_type c, size_type i = 0 ) const noexcept
3128  {
3129  return FindFirstIC( c, i );
3130  }
3131 
3141  template <class R1, class A1>
3142  size_type FindLast( const GenericString<T,R1,A1>& s, size_type r = maxPos ) const noexcept
3143  {
3144  r = pcl::Min( r, Length() );
3145  size_type i = SearchEngine( s.m_data->string, s.Length(), true/*case*/, true/*last*/ )( m_data->string, 0, r );
3146  return (i < r) ? i : notFound;
3147  }
3148 
3158  size_type FindLast( const_c_string t, size_type r = maxPos ) const noexcept
3159  {
3160  r = pcl::Min( r, Length() );
3161  size_type i = SearchEngine( t, R::Length( t ), true/*case*/, true/*last*/ )( m_data->string, 0, r );
3162  return (i < r) ? i : notFound;
3163  }
3164 
3170  size_type FindLast( char_type c, size_type r = maxPos ) const noexcept
3171  {
3172  r = pcl::Min( r, Length() );
3173  for ( const_iterator p = m_data->string+r; r > 0; --r )
3174  if ( *--p == c )
3175  return r - 1;
3176  return notFound;
3177  }
3178 
3191  template <class R1, class A1>
3192  size_type FindLastIC( const GenericString<T,R1,A1>& s, size_type r = maxPos ) const noexcept
3193  {
3194  r = pcl::Min( r, Length() );
3195  size_type i = SearchEngine( s.m_data->string, s.Length(), false/*case*/, true/*last*/ )( m_data->string, 0, r );
3196  return (i < r) ? i : notFound;
3197  }
3198 
3211  size_type FindLastIC( const_c_string t, size_type r = maxPos ) const noexcept
3212  {
3213  r = pcl::Min( r, Length() );
3214  size_type i = SearchEngine( t, R::Length( t ), false/*case*/, true/*last*/ )( m_data->string, 0, r );
3215  return (i < r) ? i : notFound;
3216  }
3217 
3226  size_type FindLastIC( char_type c, size_type r = maxPos ) const noexcept
3227  {
3228  c = R::ToCaseFolded( c );
3229  r = pcl::Min( r, Length() );
3230  for ( const_iterator p = m_data->string+r; r > 0; --r )
3231  if ( R::ToCaseFolded( *--p ) == c )
3232  return r - 1;
3233  return notFound;
3234  }
3235 
3239  template <class R1, class A1>
3240  bool Contains( const GenericString<T,R1,A1>& s ) const noexcept
3241  {
3242  return Find( s ) != notFound;
3243  }
3244 
3248  bool Contains( const_c_string t ) const noexcept
3249  {
3250  return Find( t ) != notFound;
3251  }
3252 
3256  bool Contains( char_type c ) const noexcept
3257  {
3258  return Find( c ) != notFound;
3259  }
3260 
3267  template <class R1, class A1>
3268  bool ContainsIC( const GenericString<T,R1,A1>& s ) const noexcept
3269  {
3270  return FindIC( s ) != notFound;
3271  }
3272 
3279  bool ContainsIC( const_c_string t ) const noexcept
3280  {
3281  return FindIC( t ) != notFound;
3282  }
3283 
3290  bool ContainsIC( char_type c ) const noexcept
3291  {
3292  return FindIC( c ) != notFound;
3293  }
3294 
3304  void Trim()
3305  {
3306  const_iterator l = R::SearchTrimLeft( m_data->string, m_data->end );
3307  const_iterator r = R::SearchTrimRight( l, m_data->end );
3308  if ( m_data->string < l || r < m_data->end )
3309  Trim( l, r - l );
3310  }
3311 
3321  void TrimLeft()
3322  {
3323  const_iterator l = R::SearchTrimLeft( m_data->string, m_data->end );
3324  if ( m_data->string < l )
3325  Trim( l, m_data->end - l );
3326  }
3327 
3337  void TrimRight()
3338  {
3339  const_iterator r = R::SearchTrimRight( m_data->string, m_data->end );
3340  if ( r < m_data->end )
3341  Trim( m_data->string, r - m_data->string );
3342  }
3343 
3350  {
3351  GenericString s( *this );
3352  s.Trim();
3353  return s;
3354  }
3355 
3362  {
3363  GenericString s( *this );
3364  s.TrimLeft();
3365  return s;
3366  }
3367 
3374  {
3375  GenericString s( *this );
3376  s.TrimRight();
3377  return s;
3378  }
3379 
3387  {
3388  int encloseLeft = 1;
3389  int encloseRight = 1;
3390  size_type len = Length();
3391  if ( len > 0 )
3392  {
3393  if ( *m_data->string == c )
3394  encloseLeft = 0;
3395  if ( *(m_data->end-1) == c )
3396  if ( len > 1 )
3397  encloseRight = 0;
3398  else
3399  encloseLeft = 0;
3400  }
3401  size_type n = len + encloseLeft + encloseRight;
3402  if ( n > len )
3403  {
3404  if ( !IsUnique() || m_data->ShouldReallocate( n ) )
3405  {
3406  Data* newData = Data::New( n );
3407  R::Copy( newData->string + encloseLeft, m_data->string, len );
3408  DetachFromData();
3409  m_data = newData;
3410  }
3411  else
3412  {
3413  m_data->SetLength( n );
3414  R::CopyOverlapped( m_data->string + encloseLeft, m_data->string, len );
3415  }
3416 
3417  if ( encloseLeft )
3418  *m_data->string = c;
3419  if ( encloseRight )
3420  *(m_data->end-1) = c;
3421  }
3422  }
3423 
3431  {
3432  GenericString s( *this );
3433  s.EnsureEnclosed( c );
3434  return s;
3435  }
3436 
3444  {
3445  EnsureEnclosed( R::SingleQuote() );
3446  }
3447 
3455  {
3456  GenericString s( *this );
3457  s.EnsureSingleQuoted();
3458  return s;
3459  }
3460 
3468  {
3469  EnsureEnclosed( R::DoubleQuote() );
3470  }
3471 
3479  {
3480  GenericString s( *this );
3481  s.EnsureDoubleQuoted();
3482  return s;
3483  }
3484 
3498  void Unquote()
3499  {
3500  size_type len = Length();
3501  if ( len > 1 )
3502  if ( *m_data->string == R::SingleQuote() && *(m_data->end-1) == R::SingleQuote() ||
3503  *m_data->string == R::DoubleQuote() && *(m_data->end-1) == R::DoubleQuote() )
3504  if ( IsUnique() )
3505  {
3506  R::CopyOverlapped( m_data->string, m_data->string+1, len-2 );
3507  m_data->SetLength( len-2 );
3508  }
3509  else
3510  {
3511  Data* newData = Data::New( len-2 );
3512  R::Copy( newData->string, m_data->string+1, len-2 );
3513  DetachFromData();
3514  m_data = newData;
3515  }
3516  }
3517 
3523  {
3524  GenericString s( *this );
3525  s.Unquote();
3526  return s;
3527  }
3528 
3540  void JustifyLeft( size_type width, char_type fill = R::Blank() )
3541  {
3542  size_type len = Length();
3543  if ( len < width )
3544  Append( fill, width-len );
3545  }
3546 
3558  void JustifyRight( size_type width, char_type fill = R::Blank() )
3559  {
3560  size_type len = Length();
3561  if ( len < width )
3562  Prepend( fill, width-len );
3563  }
3564 
3577  void JustifyCenter( size_type width, char_type fill = R::Blank() )
3578  {
3579  size_type len = Length();
3580  if ( len < width )
3581  {
3582  size_type n = width-len;
3583  size_type n2 = n >> 1;
3584  Prepend( fill, n2 );
3585  Append( fill, n-n2 );
3586  }
3587  }
3588 
3595  GenericString LeftJustified( size_type width, char_type fill = R::Blank() ) const
3596  {
3597  GenericString s( *this );
3598  s.JustifyLeft( width, fill );
3599  return s;
3600  }
3601 
3608  GenericString RightJustified( size_type width, char_type fill = R::Blank() ) const
3609  {
3610  GenericString s( *this );
3611  s.JustifyRight( width, fill );
3612  return s;
3613  }
3614 
3621  GenericString CenterJustified( size_type width, char_type fill = R::Blank() ) const
3622  {
3623  GenericString s( *this );
3624  s.JustifyCenter( width, fill );
3625  return s;
3626  }
3627 
3651  template <class R1, class A1>
3652  int CompareCodePoints( const GenericString<T,R1,A1>& s, bool caseSensitive = true ) const noexcept
3653  {
3654  return R::CompareCodePoints( m_data->string, Length(), s.m_data->string, s.Length(), caseSensitive );
3655  }
3656 
3682  int CompareCodePoints( const_c_string t, bool caseSensitive = true ) const noexcept
3683  {
3684  return R::CompareCodePoints( m_data->string, Length(), t, R::Length( t ), caseSensitive );
3685  }
3686 
3715  int CompareCodePoints( char_type c, bool caseSensitive = true ) const noexcept
3716  {
3717  return R::CompareCodePoints( m_data->string, Length(), &c, 1, caseSensitive );
3718  }
3719 
3746  template <class R1, class A1>
3748  bool caseSensitive = true, bool localeAware = true ) const noexcept
3749  {
3750  return R::Compare( m_data->string, Length(), s.m_data->string, s.Length(), caseSensitive, localeAware );
3751  }
3752 
3780  int Compare( const_c_string t, bool caseSensitive = true, bool localeAware = true ) const noexcept
3781  {
3782  return R::Compare( m_data->string, Length(), t, R::Length( t ), caseSensitive, localeAware );
3783  }
3784 
3814  int Compare( char_type c, bool caseSensitive = true, bool localeAware = true ) const noexcept
3815  {
3816  return R::Compare( m_data->string, Length(), &c, 1, caseSensitive, localeAware );
3817  }
3818 
3839  template <class R1, class A1>
3840  int CompareIC( const GenericString<T,R1,A1>& s, bool localeAware = true ) const noexcept
3841  {
3842  return R::Compare( m_data->string, Length(), s.m_data->string, s.Length(), false/*caseSensitive*/, localeAware );
3843  }
3844 
3866  int CompareIC( const_c_string t, bool localeAware = true ) const noexcept
3867  {
3868  return R::Compare( m_data->string, Length(), t, R::Length( t ), false/*caseSensitive*/, localeAware );
3869  }
3870 
3895  int CompareIC( char_type c, bool localeAware = true ) const noexcept
3896  {
3897  return R::Compare( m_data->string, Length(), &c, 1, false/*caseSensitive*/, localeAware );
3898  }
3899 
3915  template <class R1, class A1>
3916  bool WildMatch( const GenericString<T,R1,A1>& pattern, bool caseSensitive = true ) const noexcept
3917  {
3918  return R::WildMatch( m_data->string, Length(), pattern.m_data->string, pattern.Length(), caseSensitive );
3919  }
3920 
3934  template <class R1, class A1>
3935  bool WildMatchIC( const GenericString<T,R1,A1>& pattern ) const noexcept
3936  {
3937  return R::WildMatch( m_data->string, Length(), pattern.m_data->string, pattern.Length(), false/*caseSensitive*/ );
3938  }
3939 
3955  bool WildMatch( const_c_string pattern, bool caseSensitive = true ) const noexcept
3956  {
3957  return R::WildMatch( m_data->string, Length(), pattern, R::Length( pattern ), caseSensitive );
3958  }
3959 
3974  bool WildMatchIC( const_c_string pattern ) const noexcept
3975  {
3976  return R::WildMatch( m_data->string, Length(), pattern, R::Length( pattern ), false/*caseSensitive*/ );
3977  }
3978 
3983  bool HasWildcards() const noexcept
3984  {
3985  for ( iterator i = m_data->string; i < m_data->end; ++i )
3986  if ( R::IsWildcard( *i ) )
3987  return true;
3988  return false;
3989  }
3990 
3996  {
3997  size_type len = Length();
3998  if ( len > 0 )
3999  {
4000  EnsureUnique();
4001  R::ToCaseFolded( m_data->string, len );
4002  }
4003  }
4004 
4010  {
4011  size_type len = Length();
4012  if ( len > 0 )
4013  {
4014  EnsureUnique();
4015  R::ToLowercase( m_data->string, len );
4016  }
4017  }
4018 
4024  {
4025  size_type len = Length();
4026  if ( len > 0 )
4027  {
4028  EnsureUnique();
4029  R::ToUppercase( m_data->string, len );
4030  }
4031  }
4032 
4038  {
4039  GenericString s( *this );
4040  s.ToCaseFolded();
4041  return s;
4042  }
4043 
4049  {
4050  GenericString s( *this );
4051  s.ToLowercase();
4052  return s;
4053  }
4054 
4060  {
4061  GenericString s( *this );
4062  s.ToUppercase();
4063  return s;
4064  }
4065 
4071  void Reverse()
4072  {
4073  if ( !IsEmpty() )
4074  {
4075  EnsureUnique();
4076  for ( iterator i = m_data->string, j = m_data->end; i < --j; ++i )
4077  pcl::Swap( *i, *j );
4078  }
4079  }
4080 
4085  {
4086  GenericString s( *this );
4087  s.Reverse();
4088  return s;
4089  }
4090 
4094  void Sort()
4095  {
4096  if ( !IsEmpty() )
4097  {
4098  EnsureUnique();
4099  pcl::Sort( m_data->string, m_data->end );
4100  }
4101  }
4102 
4108  {
4109  GenericString s( *this );
4110  s.Sort();
4111  return s;
4112  }
4113 
4119  template <class BP>
4120  void Sort( BP p )
4121  {
4122  if ( !IsEmpty() )
4123  {
4124  EnsureUnique();
4125  pcl::Sort( m_data->string, m_data->end, p );
4126  }
4127  }
4128 
4133  template <class BP>
4134  GenericString Sorted( BP p ) const
4135  {
4136  GenericString s( *this );
4137  s.Sort( p );
4138  return s;
4139  }
4140 
4155  bool IsNumeral() const noexcept
4156  {
4157  if ( IsEmpty() )
4158  return false;
4159  char_type c = *R::SearchTrimLeft( m_data->string, m_data->end );
4160  return R::IsDigit( c ) || R::IsSign( c ) || R::IsDecimalSeparator( c );
4161  }
4162 
4177  bool IsSymbol() const noexcept
4178  {
4179  if ( IsEmpty() )
4180  return false;
4181  char_type c = *R::SearchTrimLeft( m_data->string, m_data->end );
4182  return R::IsSymbolDigit( c );
4183  }
4184 
4200  bool IsValidIdentifier( distance_type& pos ) const noexcept
4201  {
4202  if ( IsEmpty() )
4203  {
4204  pos = 0;
4205  return false;
4206  }
4207  const_iterator i = m_data->string;
4208  if ( R::IsStartingSymbolDigit( *i ) )
4209  for ( ;; )
4210  {
4211  if ( ++i == m_data->end )
4212  return true;
4213  if ( !R::IsSymbolDigit( *i ) )
4214  break;
4215  }
4216  pos = i - m_data->string;
4217  return false;
4218  }
4219 
4231  bool IsValidIdentifier() const noexcept
4232  {
4233  if ( !IsEmpty() )
4234  if ( R::IsStartingSymbolDigit( *m_data->string ) )
4235  {
4236  for ( const_iterator i = m_data->string; ++i < m_data->end; )
4237  if ( !R::IsSymbolDigit( *i ) )
4238  return false;
4239  return true;
4240  }
4241  return false;
4242  }
4243 
4252  uint64 Hash64( uint64 seed = 0 ) const noexcept
4253  {
4254  return pcl::Hash64( m_data->string, Size(), seed );
4255  }
4256 
4265  uint32 Hash32( uint32 seed = 0 ) const noexcept
4266  {
4267  return pcl::Hash32( m_data->string, Size(), seed );
4268  }
4269 
4274  uint64 Hash( uint64 seed = 0 ) const noexcept
4275  {
4276  return Hash64( seed );
4277  }
4278 
4287  {
4288  return Data::DeleteFreeList();
4289  }
4290 
4291  // -------------------------------------------------------------------------
4292 
4293 protected:
4294 
4299  void DetachFromData()
4300  {
4301  if ( !m_data->Detach() )
4302  Data::Dispose( m_data );
4303  }
4304 
4316  void MaybeReallocate( size_type len )
4317  {
4318  if ( IsUnique() )
4319  {
4320  if ( m_data->ShouldReallocate( len ) )
4321  {
4322  m_data->Deallocate();
4323  m_data->Allocate( len );
4324  }
4325  else
4326  m_data->SetLength( len );
4327  }
4328  else
4329  {
4330  Data* newData = Data::New( len );
4331  DetachFromData();
4332  m_data = newData;
4333  }
4334  }
4335 
4344  void UninitializedGrow( size_type& i, size_type n )
4345  {
4346  size_type len = Length();
4347  size_type newLen = len+n;
4348  if ( newLen > len )
4349  {
4350  if ( i > len )
4351  i = len;
4352  if ( IsUnique() )
4353  {
4354  if ( size_type( m_data->capacity - m_data->string ) < newLen+1 )
4355  {
4356  iterator old = m_data->string;
4357  m_data->Allocate( newLen );
4358  if ( old != nullptr )
4359  {
4360  if ( i > 0 )
4361  R::Copy( m_data->string, old, i );
4362  if ( i < len )
4363  R::Copy( m_data->string+i+n, old+i, len-i );
4364  m_data->alloc.Deallocate( old );
4365  }
4366  }
4367  else
4368  {
4369  if ( i < len )
4370  R::CopyOverlapped( m_data->string+i+n, m_data->string+i, len-i );
4371  m_data->SetLength( newLen );
4372  }
4373  }
4374  else
4375  {
4376  Data* newData = Data::New( newLen );
4377  if ( i > 0 )
4378  R::Copy( newData->string, m_data->string, i );
4379  if ( i < len )
4380  R::Copy( newData->string+i+n, m_data->string+i, len-i );
4381  DetachFromData();
4382  m_data = newData;
4383  }
4384  }
4385  }
4386 
4391  void Trim( const_iterator left, size_type len )
4392  {
4393  if ( len > 0 )
4394  {
4395  if ( IsUnique() )
4396  {
4397  if ( m_data->ShouldReallocate( len ) )
4398  {
4399  iterator old = m_data->string;
4400  m_data->Allocate( len );
4401  R::Copy( m_data->string, left, len );
4402  if ( old != nullptr )
4403  m_data->alloc.Deallocate( old );
4404  }
4405  else
4406  {
4407  if ( left != m_data->string ) // trim left
4408  R::CopyOverlapped( m_data->string, left, len );
4409  m_data->SetLength( len ); // trim right
4410  }
4411  }
4412  else
4413  {
4414  Data* newData = Data::New( len );
4415  R::Copy( newData->string, left, len );
4416  DetachFromData();
4417  m_data = newData;
4418  }
4419  }
4420  else
4421  Clear();
4422  }
4423 
4428  void ReplaceChar( char_type c1, char_type c2, size_type i, size_type n, bool caseSensitive )
4429  {
4430  if ( n > 0 )
4431  {
4432  size_type len = Length();
4433  if ( i < len )
4434  {
4435  n = pcl::Min( n, len-i );
4436  if ( caseSensitive )
4437  {
4438  for ( iterator p = m_data->string + i, p1 = p + n; p < p1; ++p )
4439  if ( *p == c1 )
4440  {
4441  distance_type d = p - m_data->string;
4442  EnsureUnique();
4443  p = m_data->string + d;
4444  *p = c2;
4445  for ( iterator p1 = m_data->string + i + n; ++p < p1; )
4446  if ( *p == c1 )
4447  *p = c2;
4448  break;
4449  }
4450  }
4451  else
4452  {
4453  c1 = R::ToCaseFolded( c1 );
4454  for ( iterator p = m_data->string + i, p1 = p + n; p < p1; ++p )
4455  if ( R::ToCaseFolded( *p ) == c1 )
4456  {
4457  distance_type d = p - m_data->string;
4458  EnsureUnique();
4459  p = m_data->string + d;
4460  *p = c2;
4461  for ( iterator p1 = m_data->string + i + n; ++p < p1; )
4462  if ( R::ToCaseFolded( *p ) == c1 )
4463  *p = c2;
4464  break;
4465  }
4466  }
4467  }
4468  }
4469  }
4470 
4475  void ReplaceString( const_iterator t1, size_type n1, const_iterator t2, size_type n2, size_type i, bool caseSensitive )
4476  {
4477  if ( n1 > 0 )
4478  {
4479  size_type len = Length();
4480  if ( i < len )
4481  {
4482  SearchEngine S( t1, n1, caseSensitive );
4483  if ( n1 == n2 )
4484  {
4485  EnsureUnique();
4486  for ( size_type p = i; (p = S( m_data->string, p, len )) < len; p += n1 )
4487  R::Copy( m_data->string + p, t2, n2 );
4488  }
4489  else
4490  {
4491  Array<size_type> P;
4492  for ( size_type p = i; (p = S( m_data->string, p, len )) < len; p += n1 )
4493  P.Add( p );
4494  if ( !P.IsEmpty() )
4495  {
4496  size_type newLen = len;
4497  if ( n1 < n2 )
4498  newLen += P.Length()*(n2 - n1);
4499  else
4500  newLen -= P.Length()*(n1 - n2);
4501 
4502  if ( newLen > 0 )
4503  {
4504  Data* newData = Data::New( newLen );
4505  size_type targetIndex = 0;
4506  size_type sourceIndex = 0;
4507  for ( size_type p : P )
4508  {
4509  size_type n = p - sourceIndex;
4510  if ( n > 0 )
4511  R::Copy( newData->string+targetIndex, m_data->string+sourceIndex, n );
4512  R::Copy( newData->string+targetIndex+n, t2, n2 );
4513  targetIndex += n + n2;
4514  sourceIndex = p + n1;
4515  }
4516  if ( sourceIndex < len )
4517  R::Copy( newData->string+targetIndex, m_data->string+sourceIndex, len-sourceIndex );
4518  DetachFromData();
4519  m_data = newData;
4520  }
4521  else
4522  Clear();
4523  }
4524  }
4525  }
4526  }
4527  }
4528 
4546  class PCL_CLASS SearchEngine
4547  {
4548  public:
4549 
4560  SearchEngine( const_iterator pattern, size_type patternLength,
4561  bool caseSensitive = true, bool searchLast = false, bool useBoyerMoore = true )
4562  : m_pattern( pattern )
4563  , m_patternLength( int( patternLength ) )
4564  , m_caseSensitive( caseSensitive )
4565  , m_searchLast( searchLast )
4566  , m_useBoyerMoore( useBoyerMoore && m_patternLength > 3 )
4567  {
4568  if ( m_useBoyerMoore )
4569  InitSkipList();
4570  }
4571 
4584  size_type operator()( const_iterator text, size_type startIndex, size_type endIndex ) const noexcept
4585  {
4586  if ( endIndex <= startIndex
4587  || m_patternLength <= 0
4588  || endIndex-startIndex < size_type( m_patternLength )
4589  || text == nullptr
4590  || m_pattern == nullptr )
4591  return endIndex;
4592 
4593  if ( m_caseSensitive )
4594  {
4595  if ( m_useBoyerMoore )
4596  {
4597  if ( m_searchLast )
4598  {
4599  for ( size_type i = startIndex, r = endIndex-m_patternLength; i <= r; )
4600  {
4601  int skip = 0;
4602  const_iterator t = text + r - i;
4603  const_iterator p = m_pattern;
4604  for ( int j = m_patternLength; --j >= 0; )
4605  {
4606  char_type c = *t++;
4607  if ( c != *p++ )
4608  {
4609  skip = j - m_skipList[uint8( c )];
4610  if ( skip < 1 )
4611  skip = 1;
4612  break;
4613  }
4614  }
4615  if ( skip == 0 )
4616  return r - i;
4617  i += skip;
4618  }
4619  }
4620  else
4621  {
4622  for ( size_type i = startIndex, r = endIndex-m_patternLength; i <= r; )
4623  {
4624  int skip = 0;
4625  const_iterator t = text + i + m_patternLength;
4626  const_iterator p = m_pattern + m_patternLength;
4627  for ( int j = m_patternLength; --j >= 0; )
4628  {
4629  char_type c = *--t;
4630  if ( c != *--p )
4631  {
4632  // ### N.B.: Could do better with a precomputed pattern mismatch table.
4633  skip = j - m_skipList[uint8( c )];
4634  if ( skip < 1 )
4635  skip = 1;
4636  break;
4637  }
4638  }
4639  if ( skip == 0 )
4640  return i;
4641  i += skip;
4642  }
4643  }
4644  }
4645  else
4646  {
4647  // Use a brute force search for very small patterns.
4648  if ( m_searchLast )
4649  {
4650  for ( size_type i = endIndex-m_patternLength; ; --i )
4651  {
4652  const_iterator t = text + i;
4653  const_iterator p = m_pattern;
4654  for ( int j = m_patternLength; ; ++t, ++p )
4655  {
4656  if ( *t != *p )
4657  break;
4658  if ( --j == 0 )
4659  return i;
4660  }
4661  if ( i == startIndex )
4662  break;
4663  }
4664  }
4665  else
4666  {
4667  for ( size_type i = startIndex, r = endIndex-m_patternLength; ; ++i )
4668  {
4669  const_iterator t = text + i;
4670  const_iterator p = m_pattern;
4671  for ( int j = m_patternLength; ; ++t, ++p )
4672  {
4673  if ( *t != *p )
4674  break;
4675  if ( --j == 0 )
4676  return i;
4677  }
4678  if ( i == r )
4679  break;
4680  }
4681  }
4682  }
4683  }
4684  else
4685  {
4686  if ( m_useBoyerMoore )
4687  {
4688  if ( m_searchLast )
4689  {
4690  for ( size_type i = startIndex, r = endIndex-m_patternLength; i <= r; )
4691  {
4692  int skip = 0;
4693  const_iterator t = text + r - i;
4694  const_iterator p = m_pattern;
4695  for ( int j = m_patternLength; --j >= 0; )
4696  {
4697  char_type c = R::ToCaseFolded( *t++ );
4698  if ( c != R::ToCaseFolded( *p++ ) )
4699  {
4700  skip = j - m_skipList[uint8( c )];
4701  if ( skip < 1 )
4702  skip = 1;
4703  break;
4704  }
4705  }
4706  if ( skip == 0 )
4707  return r - i;
4708  i += skip;
4709  }
4710  }
4711  else
4712  {
4713  for ( size_type i = startIndex, r = endIndex-m_patternLength; i <= r; )
4714  {
4715  int skip = 0;
4716  const_iterator t = text + i + m_patternLength;
4717  const_iterator p = m_pattern + m_patternLength;
4718  for ( int j = m_patternLength; --j >= 0; )
4719  {
4720  char_type c = R::ToCaseFolded( *--t );
4721  if ( c != R::ToCaseFolded( *--p ) )
4722  {
4723  // ### N.B.: Could do better with a precomputed pattern mismatch table.
4724  skip = j - m_skipList[uint8( c )];
4725  if ( skip < 1 )
4726  skip = 1;
4727  break;
4728  }
4729  }
4730  if ( skip == 0 )
4731  return i;
4732  i += skip;
4733  }
4734  }
4735  }
4736  else
4737  {
4738  // Use a brute force search for very small patterns.
4739  if ( m_searchLast )
4740  {
4741  for ( size_type i = endIndex-m_patternLength; ; --i )
4742  {
4743  const_iterator t = text + i;
4744  const_iterator p = m_pattern;
4745  for ( int j = m_patternLength; ; ++t, ++p )
4746  {
4747  if ( R::ToCaseFolded( *t ) != R::ToCaseFolded( *p ) )
4748  break;
4749  if ( --j == 0 )
4750  return i;
4751  }
4752  if ( i == startIndex )
4753  break;
4754  }
4755  }
4756  else
4757  {
4758  for ( size_type i = startIndex, r = endIndex-m_patternLength; ; ++i )
4759  {
4760  const_iterator t = text + i;
4761  const_iterator p = m_pattern;
4762  for ( int j = m_patternLength; ; ++t, ++p )
4763  {
4764  if ( R::ToCaseFolded( *t ) != R::ToCaseFolded( *p ) )
4765  break;
4766  if ( --j == 0 )
4767  return i;
4768  }
4769  if ( i == r )
4770  break;
4771  }
4772  }
4773  }
4774  }
4775  return endIndex;
4776  }
4777 
4778  private:
4779 
4780  int m_skipList[ 256 ];
4781  const_iterator m_pattern;
4782  int m_patternLength;
4783  bool m_caseSensitive : 1;
4784  bool m_searchLast : 1;
4785  bool m_useBoyerMoore : 1;
4786 
4787  void InitSkipList() noexcept
4788  {
4789  ::memset( m_skipList, 0xff, sizeof( m_skipList ) ); // fill with -1
4790  if ( m_searchLast )
4791  {
4792  const_iterator p = m_pattern + m_patternLength;
4793  if ( m_caseSensitive )
4794  for ( int i = 0; i < m_patternLength; ++i )
4795  m_skipList[uint8( *--p )] = i;
4796  else
4797  for ( int i = 0; i < m_patternLength; ++i )
4798  m_skipList[uint8( R::ToCaseFolded( *--p ) )] = i;
4799  }
4800  else
4801  {
4802  const_iterator p = m_pattern;
4803  if ( m_caseSensitive )
4804  for ( int i = 0; i < m_patternLength; ++i )
4805  m_skipList[uint8( *p++ )] = i;
4806  else
4807  for ( int i = 0; i < m_patternLength; ++i )
4808  m_skipList[uint8( R::ToCaseFolded( *p++ ) )] = i;
4809  }
4810  }
4811  };
4812 
4813  // -------------------------------------------------------------------------
4814 
4820  class PCL_CLASS Data : public ReferenceCounter
4821  {
4822  private:
4823 
4824 #ifndef __PCL_NO_STRING_FREE_LIST
4825  static Data* freeList;
4826  static AtomicInt freeLock;
4827 #endif
4828 
4829  public:
4830 
4831  iterator string = nullptr;
4832  iterator end = nullptr;
4833  iterator capacity = nullptr;
4834  allocator alloc;
4835 
4839  Data() = default;
4840 
4844  Data( size_type len )
4845  {
4846  Allocate( len );
4847  }
4848 
4853  Data( size_type len, size_type total )
4854  {
4855  Allocate( len, total );
4856  }
4857 
4861  ~Data()
4862  {
4863  Deallocate();
4864  }
4865 
4876  void Allocate( size_type len, size_type total )
4877  {
4878  total = (len <= total) ? alloc.PagedLength( total+1 ) : len+1; // +1 is room for a null terminating character
4879  string = alloc.Allocate( total );
4880  capacity = string + total;
4881  SetLength( len );
4882  }
4883 
4888  void Allocate( size_type len )
4889  {
4890  Allocate( len, len );
4891  }
4892 
4901  void Reserve( size_type len, size_type total )
4902  {
4903  PCL_PRECONDITION( len <= total )
4904  string = alloc.Allocate( total+1 );
4905  capacity = string + total+1;
4906  SetLength( pcl::Min( len, total ) );
4907  }
4908 
4912  void Deallocate()
4913  {
4914  PCL_CHECK( (string == nullptr) ? end == nullptr : string < end )
4915  if ( string != nullptr )
4916  {
4917  alloc.Deallocate( string );
4918  Reset();
4919  }
4920  }
4921 
4926  bool ShouldReallocate( size_type len ) const noexcept
4927  {
4928  size_type m = capacity - string;
4929  return m <= len || alloc.ReallocatedLength( m, len+1 ) < (m >> 1);
4930  }
4931 
4936  void SetLength( size_type len ) noexcept
4937  {
4938  *(end = (string + len)) = R::Null();
4939  }
4940 
4944  void Reset() noexcept
4945  {
4946  string = end = capacity = nullptr;
4947  }
4948 
4955 #ifndef __PCL_NO_STRING_FREE_LIST
4956  static Data* NextFree() noexcept
4957  {
4958  if ( freeLock.TestAndSet( 0, 1 ) )
4959  {
4960  Data* data = freeList;
4961  if ( data != nullptr )
4962  freeList = reinterpret_cast<Data*>( data->string );
4963  freeLock.Store( 0 );
4964  return data;
4965  }
4966  return nullptr;
4967  }
4968 #endif // !__PCL_NO_STRING_FREE_LIST
4969 
4977  static Data* New()
4978  {
4979 #ifndef __PCL_NO_STRING_FREE_LIST
4980  Data* data = NextFree();
4981  if ( data != nullptr )
4982  {
4983  data->string = nullptr;
4984  return data;
4985  }
4986 #endif // !__PCL_NO_STRING_FREE_LIST
4987  return new Data;
4988  }
4989 
4997  static Data* New( size_type len )
4998  {
4999 #ifndef __PCL_NO_STRING_FREE_LIST
5000  Data* data = NextFree();
5001  if ( data != nullptr )
5002  {
5003  data->Allocate( len ); // ### FIXME: If allocation fails, data is a leak.
5004  return data;
5005  }
5006 #endif // !__PCL_NO_STRING_FREE_LIST
5007  return new Data( len );
5008  }
5009 
5017  static Data* New( size_type len, size_type total )
5018  {
5019 #ifndef __PCL_NO_STRING_FREE_LIST
5020  Data* data = NextFree();
5021  if ( data != nullptr )
5022  {
5023  data->Allocate( len, total ); // ### FIXME: If allocation fails, data is a leak.
5024  return data;
5025  }
5026 #endif // !__PCL_NO_STRING_FREE_LIST
5027  return new Data( len, total );
5028  }
5029 
5037  static void Dispose( Data* data )
5038  {
5039  PCL_PRECONDITION( data != nullptr )
5040  PCL_CHECK( data->RefCount() == 0 )
5041 #ifndef __PCL_NO_STRING_FREE_LIST
5042  if ( freeLock.TestAndSet( 0, 1 ) )
5043  {
5044  data->Attach();
5045  data->Deallocate();
5046  data->string = reinterpret_cast<iterator>( freeList );
5047  freeList = data;
5048  freeLock.Store( 0 );
5049  }
5050  else
5051 #endif // !__PCL_NO_STRING_FREE_LIST
5052  delete data;
5053  }
5054 
5062  static size_type DeleteFreeList()
5063  {
5064  size_type count = 0;
5065 #ifndef __PCL_NO_STRING_FREE_LIST
5066  while ( freeList != nullptr )
5067  {
5068  Data* data = freeList;
5069  freeList = reinterpret_cast<Data*>( data->string );
5070  data->string = nullptr;
5071  delete data;
5072  ++count;
5073  }
5074 #endif // !__PCL_NO_STRING_FREE_LIST
5075  return count;
5076  }
5077  };
5078 
5079  /*
5080  * The reference-counted string data.
5081  */
5082  Data* m_data = nullptr;
5083 };
5084 
5085 #ifndef __PCL_NO_STRING_FREE_LIST
5086 
5087 template <class T, class R, class A>
5088 typename GenericString<T,R,A>::Data* GenericString<T,R,A>::Data::freeList = nullptr;
5089 
5090 template <class T, class R, class A>
5091 AtomicInt GenericString<T,R,A>::Data::freeLock;
5092 
5093 #endif // !__PCL_NO_STRING_FREE_LIST
5094 
5098 template <class T, class R, class A> inline
5100 {
5101  s1.Swap( s2 );
5102 }
5103 
5104 // ----------------------------------------------------------------------------
5105 
5110 // ----------------------------------------------------------------------------
5111 
5116 template <class T, class R1, class A1, class R2, class A2> inline
5117 bool operator ==( const GenericString<T,R1,A1>& s1, const GenericString<T,R2,A2>& s2 ) noexcept
5118 {
5119  return s1.CompareCodePoints( s2 ) == 0;
5120 }
5121 
5129 template <class T, class R1, class A1, class R2, class A2> inline
5130 bool operator <( const GenericString<T,R1,A1>& s1, const GenericString<T,R2,A2>& s2 ) noexcept
5131 {
5132  return s1.CompareCodePoints( s2 ) < 0;
5133 }
5134 
5142 template <class T, class R1, class A1, class R2, class A2> inline
5143 bool operator <=( const GenericString<T,R1,A1>& s1, const GenericString<T,R2,A2>& s2 ) noexcept
5144 {
5145  return s1.CompareCodePoints( s2 ) <= 0;
5146 }
5147 
5155 template <class T, class R1, class A1, class R2, class A2> inline
5156 bool operator >( const GenericString<T,R1,A1>& s1, const GenericString<T,R2,A2>& s2 ) noexcept
5157 {
5158  return s1.CompareCodePoints( s2 ) > 0;
5159 }
5160 
5168 template <class T, class R1, class A1, class R2, class A2> inline
5169 bool operator >=( const GenericString<T,R1,A1>& s1, const GenericString<T,R2,A2>& s2 ) noexcept
5170 {
5171  return s1.CompareCodePoints( s2 ) >= 0;
5172 }
5173 
5174 // ----------------------------------------------------------------------------
5175 
5180 template <class T, class R, class A> inline
5182 {
5183  return s1.CompareCodePoints( t2 ) == 0;
5184 }
5185 
5193 template <class T, class R, class A> inline
5195 {
5196  return s1.CompareCodePoints( t2 ) < 0;
5197 }
5198 
5206 template <class T, class R, class A> inline
5208 {
5209  return s1.CompareCodePoints( t2 ) <= 0;
5210 }
5211 
5219 template <class T, class R, class A> inline
5221 {
5222  return s1.CompareCodePoints( t2 ) > 0;
5223 }
5224 
5232 template <class T, class R, class A> inline
5234 {
5235  return s1.CompareCodePoints( t2 ) >= 0;
5236 }
5237 
5238 // ----------------------------------------------------------------------------
5239 
5244 template <class T, class R, class A> inline
5246 {
5247  return s2.CompareCodePoints( t1 ) == 0;
5248 }
5249 
5257 template <class T, class R, class A> inline
5259 {
5260  return s2.CompareCodePoints( t1 ) > 0;
5261 }
5262 
5270 template <class T, class R, class A> inline
5272 {
5273  return s2.CompareCodePoints( t1 ) >= 0;
5274 }
5275 
5283 template <class T, class R, class A> inline
5285 {
5286  return s2.CompareCodePoints( t1 ) < 0;
5287 }
5288 
5296 template <class T, class R, class A> inline
5298 {
5299  return s2.CompareCodePoints( t1 ) <= 0;
5300 }
5301 
5302 // ----------------------------------------------------------------------------
5303 
5308 template <class T, class R, class A> inline
5310 {
5311  return s1.CompareCodePoints( c2 ) == 0;
5312 }
5313 
5321 template <class T, class R, class A> inline
5322 bool operator <( const GenericString<T,R,A>& s1, typename GenericString<T,R,A>::char_type c2 ) noexcept
5323 {
5324  return s1.CompareCodePoints( c2 ) < 0;
5325 }
5326 
5334 template <class T, class R, class A> inline
5336 {
5337  return s1.CompareCodePoints( c2 ) <= 0;
5338 }
5339 
5347 template <class T, class R, class A> inline
5348 bool operator >( const GenericString<T,R,A>& s1, typename GenericString<T,R,A>::char_type c2 ) noexcept
5349 {
5350  return s1.CompareCodePoints( c2 ) > 0;
5351 }
5352 
5360 template <class T, class R, class A> inline
5362 {
5363  return s1.CompareCodePoints( c2 ) >= 0;
5364 }
5365 
5366 // ----------------------------------------------------------------------------
5367 
5372 template <class T, class R, class A> inline
5374 {
5375  return s2.CompareCodePoints( c1 ) == 0;
5376 }
5377 
5385 template <class T, class R, class A> inline
5386 bool operator <( typename GenericString<T,R,A>::char_type c1, const GenericString<T,R,A>& s2 ) noexcept
5387 {
5388  return s2.CompareCodePoints( c1 ) > 0;
5389 }
5390 
5398 template <class T, class R, class A> inline
5400 {
5401  return s2.CompareCodePoints( c1 ) >= 0;
5402 }
5403 
5411 template <class T, class R, class A> inline
5412 bool operator >( typename GenericString<T,R,A>::char_type c1, const GenericString<T,R,A>& s2 ) noexcept
5413 {
5414  return s2.CompareCodePoints( c1 ) < 0;
5415 }
5416 
5424 template <class T, class R, class A> inline
5426 {
5427  return s2.CompareCodePoints( c1 ) <= 0;
5428 }
5429 
5430 // ----------------------------------------------------------------------------
5431 
5444 class PCL_CLASS IsoString : public GenericString<char, IsoCharTraits, PCL_STRING_ALLOCATOR>
5445 {
5446 public:
5447 
5452 
5457 
5462 
5467 
5472 
5477 
5482 
5487 
5492 
5497 
5502 
5507 
5512 
5517 
5524 
5529 
5534 
5539 
5543  using const_c_ustring = ustring_base::const_c_string;
5544 
5549 
5554 
5555  // -------------------------------------------------------------------------
5556 
5560  IsoString() = default;
5561 
5567  : string_base( s )
5568  {
5569  }
5570 
5574  IsoString( const IsoString& ) = default;
5575 
5581  : string_base( std::move( s ) )
5582  {
5583  }
5584 
5588  IsoString( IsoString&& ) = default;
5589 
5597  explicit
5599  {
5600  (void)operator =( s );
5601  }
5602 
5607  : string_base( t )
5608  {
5609  }
5610 
5616  : string_base( t, i, n )
5617  {
5618  }
5619 
5624  : string_base( c, n )
5625  {
5626  }
5627 
5633  : string_base( i, j )
5634  {
5635  }
5636 
5645  IsoString( std::initializer_list<char_type> l )
5646  : IsoString( l.begin(), l.end() )
5647  {
5648  }
5649 
5659  explicit
5661  {
5662  (void)operator =( t );
5663  }
5664 
5675  IsoString( const_c_ustring t, size_type i, size_type n );
5676 
5684  explicit
5685  IsoString( const ByteArray& B )
5686  : IsoString( const_iterator( B.Begin() ), const_iterator( B.End() ) )
5687  {
5688  }
5689 
5693  explicit
5694  IsoString( bool x )
5695  : string_base( x ? "true" : "false" )
5696  {
5697  }
5698 
5703  explicit
5704  IsoString( short x )
5705  {
5706  (void)Format( "%hd", x );
5707  }
5708 
5713  explicit
5714  IsoString( unsigned short x )
5715  {
5716  (void)Format( "%hu", x );
5717  }
5718 
5723  explicit
5724  IsoString( int x )
5725  {
5726  (void)Format( "%i", x );
5727  }
5728 
5733  explicit
5734  IsoString( unsigned int x )
5735  {
5736  (void)Format( "%u", x );
5737  }
5738 
5743  explicit
5744  IsoString( long x )
5745  {
5746  (void)Format( "%ld", x );
5747  }
5748 
5753  explicit
5754  IsoString( unsigned long x )
5755  {
5756  (void)Format( "%lu", x );
5757  }
5758 
5763  explicit
5764  IsoString( long long x )
5765  {
5766  (void)Format( "%lli", x );
5767  }
5768 
5773  explicit
5774  IsoString( unsigned long long x )
5775  {
5776  (void)Format( "%llu", x );
5777  }
5778 
5783  explicit
5784  IsoString( float x )
5785  {
5786  (void)Format( "%.7g", x );
5787  }
5788 
5793  explicit
5794  IsoString( double x )
5795  {
5796  (void)Format( "%.16g", x );
5797  }
5798 
5803  explicit
5804  IsoString( long double x )
5805  {
5806 #ifdef _MSC_VER
5807  (void)Format( "%.16Lg", x );
5808 #else
5809  (void)Format( "%.18Lg", x );
5810 #endif
5811  }
5812 
5813 #ifndef __PCL_NO_STRING_COMPLEX
5814 
5819  explicit
5821  {
5822  (void)Format( "{%.7g,%.7g}", x.Real(), x.Imag() );
5823  }
5824 
5829  explicit
5831  {
5832  (void)Format( "{%.16g,%.16g}", x.Real(), x.Imag() );
5833  }
5834 
5839  explicit
5841  {
5842 #ifdef _MSC_VER
5843  (void)Format( "{%.16Lg,%.16Lg}", x.Real(), x.Imag() );
5844 #else
5845  (void)Format( "{%.18Lg,%.18Lg}", x.Real(), x.Imag() );
5846 #endif
5847  }
5848 
5849 #endif // !__PCL_NO_STRING_COMPLEX
5850 
5851 #ifdef __PCL_QT_INTERFACE
5852 
5853  explicit
5854  IsoString( const QString& qs )
5855  : string_base( qs.isEmpty() ? iterator( nullptr ) : iterator( PCL_GET_CHARPTR_FROM_QSTRING( qs ) ) )
5856  {
5857  }
5858 
5859  explicit
5860  IsoString( const QAnyStringView& qv )
5861  : string_base( qv.isEmpty() ? iterator( nullptr ) : iterator( PCL_GET_CHARPTR_FROM_QSTRINGVIEW( qv ) ) )
5862  {
5863  }
5864 
5865  explicit
5866  IsoString( const QByteArray& qb )
5867  : string_base( qb.isEmpty() ? iterator( nullptr ) : iterator( PCL_GET_CHARPTR_FROM_QBYTEARRAY( qb ) ) )
5868  {
5869  }
5870 
5871  explicit
5872  IsoString( const QDate& qd )
5873  : string_base( iterator( PCL_GET_CHARPTR_FROM_QSTRING( qd.toString( PCL_QDATE_FMT_STR ) ) ) )
5874  {
5875  }
5876 
5877  explicit
5878  IsoString( const QDateTime& qdt )
5879  : string_base( iterator( PCL_GET_CHARPTR_FROM_QSTRING( qdt.toString( PCL_QDATETIME_FMT_STR ) ) ) )
5880  {
5881  }
5882 
5883 #endif
5884 
5885  // -------------------------------------------------------------------------
5886 
5890  IsoString& operator =( const IsoString& s )
5891  {
5892  Assign( s );
5893  return *this;
5894  }
5895 
5899  IsoString& operator =( IsoString&& s )
5900  {
5901  Transfer( s );
5902  return *this;
5903  }
5904 
5909  IsoString& operator =( const string_base& s )
5910  {
5911  Assign( s );
5912  return *this;
5913  }
5914 
5919  IsoString& operator =( string_base&& s )
5920  {
5921  Transfer( s );
5922  return *this;
5923  }
5924 
5935  IsoString& operator =( const ustring_base& s )
5936  {
5937  return operator =( s.Begin() );
5938  }
5939 
5944  IsoString& operator =( const_c_string t )
5945  {
5946  Assign( t );
5947  return *this;
5948  }
5949 
5954  IsoString& operator =( char_type c )
5955  {
5956  Assign( c, 1 );
5957  return *this;
5958  }
5959 
5970  IsoString& operator =( const_c_ustring t );
5971 
5972 #ifdef __PCL_QT_INTERFACE
5973 
5974  IsoString& operator =( const QString& qs )
5975  {
5976  if ( qs.isEmpty() )
5977  Clear();
5978  else
5979  Assign( PCL_GET_CHARPTR_FROM_QSTRING( qs ) );
5980  return *this;
5981  }
5982 
5983  IsoString& operator =( const QByteArray& qb )
5984  {
5985  if ( qb.isEmpty() )
5986  Clear();
5987  else
5988  Assign( PCL_GET_CHARPTR_FROM_QBYTEARRAY( qb ) );
5989  return *this;
5990  }
5991 
5992  IsoString& operator =( const QDate& qd )
5993  {
5994  Assign( PCL_GET_CHARPTR_FROM_QSTRING( qd.toString( PCL_QDATE_FMT_STR ) ) );
5995  return *this;
5996  }
5997 
5998  IsoString& operator =( const QDateTime& qdt )
5999  {
6000  Assign( PCL_GET_CHARPTR_FROM_QSTRING( qdt.toString( PCL_QDATETIME_FMT_STR ) ) );
6001  return *this;
6002  }
6003 
6004 #endif
6005 
6006  // -------------------------------------------------------------------------
6007 
6008  IsoString SetToLength( size_type n ) const
6009  {
6010  return string_base::SetToLength( n );
6011  }
6012 
6013  IsoString ResizedToNullTerminated() const
6014  {
6015  return string_base::ResizedToNullTerminated();
6016  }
6017 
6018  IsoString Squeezed() const
6019  {
6020  return string_base::Squeezed();
6021  }
6022 
6023  // -------------------------------------------------------------------------
6024 
6025  IsoString Substring( size_type i, size_type n = maxPos ) const
6026  {
6027  return string_base::Substring( i, n );
6028  }
6029 
6030  IsoString Left( size_type n ) const
6031  {
6032  return string_base::Left( n );
6033  }
6034 
6035  IsoString Right( size_type n ) const
6036  {
6037  return string_base::Right( n );
6038  }
6039 
6040  IsoString Suffix( size_type i ) const
6041  {
6042  return string_base::Suffix( i );
6043  }
6044 
6045  IsoString Prefix( size_type i ) const
6046  {
6047  return string_base::Prefix( i );
6048  }
6049 
6050  // -------------------------------------------------------------------------
6051 
6052  IsoString Trimmed() const
6053  {
6054  return string_base::Trimmed();
6055  }
6056 
6057  IsoString TrimmedLeft() const
6058  {
6059  return string_base::TrimmedLeft();
6060  }
6061 
6062  IsoString TrimmedRight() const
6063  {
6064  return string_base::TrimmedRight();
6065  }
6066 
6067  // -------------------------------------------------------------------------
6068 
6069  IsoString LeftJustified( size_type width, char_type fill = IsoCharTraits::Blank() ) const
6070  {
6071  return string_base::LeftJustified( width, fill );
6072  }
6073 
6074  IsoString RightJustified( size_type width, char_type fill = IsoCharTraits::Blank() ) const
6075  {
6076  return string_base::RightJustified( width, fill );
6077  }
6078 
6079  IsoString CenterJustified( size_type width, char_type fill = IsoCharTraits::Blank() ) const
6080  {
6081  return string_base::CenterJustified( width, fill );
6082  }
6083 
6084  // -------------------------------------------------------------------------
6085 
6086  IsoString Enclosed( char_type c ) const
6087  {
6088  return string_base::Enclosed( c );
6089  }
6090 
6091  IsoString SingleQuoted() const
6092  {
6093  return string_base::SingleQuoted();
6094  }
6095 
6096  IsoString DoubleQuoted() const
6097  {
6098  return string_base::DoubleQuoted();
6099  }
6100 
6101  IsoString Unquoted() const
6102  {
6103  return string_base::Unquoted();
6104  }
6105 
6106  // -------------------------------------------------------------------------
6107 
6108  IsoString CaseFolded() const
6109  {
6110  return string_base::CaseFolded();
6111  }
6112 
6113  IsoString Lowercase() const
6114  {
6115  return string_base::Lowercase();
6116  }
6117 
6118  IsoString Uppercase() const
6119  {
6120  return string_base::Uppercase();
6121  }
6122 
6123  // -------------------------------------------------------------------------
6124 
6125  IsoString Reversed() const
6126  {
6127  return string_base::Reversed();
6128  }
6129 
6130  IsoString Sorted() const
6131  {
6132  return string_base::Sorted();
6133  }
6134 
6135  template <class BP>
6136  IsoString Sorted( BP p ) const
6137  {
6138  return string_base::Sorted( p );
6139  }
6140 
6141  // -------------------------------------------------------------------------
6142 
6152  template <class C>
6153  IsoString& ToSeparated( const C& c, char_type separator )
6154  {
6155  Clear();
6156  return c.ToSeparated( *this, separator );
6157  }
6158 
6175  template <class C, class AF>
6176  IsoString& ToSeparated( const C& c, char_type separator, AF append )
6177  {
6178  Clear();
6179  return c.ToSeparated( *this, separator, append );
6180  }
6181 
6191  template <class C>
6192  IsoString& ToSeparated( const C& c, const IsoString& separator )
6193  {
6194  Clear();
6195  return c.ToSeparated( *this, separator );
6196  }
6197 
6214  template <class C, class AF>
6215  IsoString& ToSeparated( const C& c, const IsoString& separator, AF append )
6216  {
6217  Clear();
6218  return c.ToSeparated( *this, separator, append );
6219  }
6220 
6230  template <class C>
6231  IsoString& ToSeparated( const C& c, const_c_string separator )
6232  {
6233  return ToSeparated( c, IsoString( separator ) );
6234  }
6235 
6252  template <class C, class AF>
6253  IsoString& ToSeparated( const C& c, const_c_string separator, AF append )
6254  {
6255  return ToSeparated( c, IsoString( separator ), append );
6256  }
6257 
6267  template <class C>
6269  {
6270  return ToSeparated( c, IsoCharTraits::Comma() );
6271  }
6272 
6282  template <class C>
6284  {
6285  return ToSeparated( c, IsoCharTraits::Colon() );
6286  }
6287 
6297  template <class C>
6299  {
6300  return ToSeparated( c, IsoCharTraits::Blank() );
6301  }
6302 
6312  template <class C>
6313  IsoString& ToTabSeparated( const C& c )
6314  {
6315  return ToSeparated( c, IsoCharTraits::Tab() );
6316  }
6317 
6327  template <class C>
6329  {
6330  return ToSeparated( c, IsoCharTraits::LF() );
6331  }
6332 
6342  template <class C>
6344  {
6345  return ToSeparated( c, IsoCharTraits::Null() );
6346  }
6347 
6356  template <class C>
6357  IsoString& ToHyphenated( const C& c )
6358  {
6359  return ToSeparated( c, IsoCharTraits::Hyphen() );
6360  }
6361 
6362  // -------------------------------------------------------------------------
6363 
6379 
6387  {
6388  return IsoString( *this ).ToEncodedHTMLSpecialChars();
6389  }
6390 
6408 
6417  {
6418  return IsoString( *this ).ToDecodedHTMLSpecialChars();
6419  }
6420 
6432  static IsoString ToURLEncoded( const void* data, size_type length );
6433 
6446  template <class C>
6447  static IsoString ToURLEncoded( const C& c )
6448  {
6449  return ToURLEncoded( c.Begin(), c.Length()*sizeof( *c.Begin() ) );
6450  }
6451 
6462 
6470  {
6471  return IsoString( *this ).ToURLEncoded();
6472  }
6473 
6485  static ByteArray FromURLEncoded( const void* data, size_type length );
6486 
6494  template <class C>
6495  static ByteArray FromURLEncoded( const C& c )
6496  {
6497  return FromURLEncoded( c.Begin(), c.Length()*sizeof( *c.Begin() ) );
6498  }
6499 
6512  {
6513  return FromURLEncoded( Begin(), Length() );
6514  }
6515 
6527  static IsoString ToURLDecoded( const void* data, size_type length );
6528 
6535  template <class C>
6536  static IsoString ToURLDecoded( const C& c )
6537  {
6538  return ToURLDecoded( c.Begin(), c.Length()*sizeof( *c.Begin() ) );
6539  }
6540 
6551 
6559  {
6560  return IsoString( *this ).ToURLDecoded();
6561  }
6562 
6563  // -------------------------------------------------------------------------
6564 
6565 #ifdef __PCL_QT_INTERFACE
6566 
6567  operator QString() const
6568  {
6569  return QString( c_str() );
6570  }
6571 
6572  operator QAnyStringView() const
6573  {
6574  return QAnyStringView( c_str() );
6575  }
6576 
6577  operator QByteArray() const
6578  {
6579  return QByteArray( c_str() );
6580  }
6581 
6582  operator QDate() const
6583  {
6584  return QDate::fromString( c_str(), PCL_QDATE_FMT_STR );
6585  }
6586 
6587  operator QDateTime() const
6588  {
6589  return QDateTime::fromString( c_str(), PCL_QDATETIME_FMT_STR );
6590  }
6591 
6592 #endif
6593 
6606  {
6607  va_list paramList;
6608  va_start( paramList, fmt );
6609 
6610  (void)VFormat( fmt, paramList );
6611 
6612  va_end( paramList );
6613  return *this;
6614  }
6615 
6628  {
6629  va_list paramList;
6630  va_start( paramList, fmt );
6631 
6632  (void)AppendVFormat( fmt, paramList );
6633 
6634  va_end( paramList );
6635  return *this;
6636  }
6637 
6650  int VFormat( const_c_string fmt, va_list paramList );
6651 
6664  int AppendVFormat( const_c_string fmt, va_list paramList );
6665 
6666  // -------------------------------------------------------------------------
6667 
6678  {
6679  ustring_base s;
6680  s.SetLength( Length() );
6681  uchar_iterator p = s.Begin();
6682  for ( const_iterator i = m_data->string; i < m_data->end; ++p, ++i )
6683  *p = uchar_type( *i );
6684  return s;
6685  }
6686 
6694  ustring_base UTF8ToUTF16( size_type i = 0, size_type n = maxPos ) const; // implemented inline after String
6695 
6712 
6713 #ifdef __PCL_QT_INTERFACE
6714 
6715  QString ToQString() const
6716  {
6717  return operator QString();
6718  }
6719 
6720  QByteArray ToQByteArray() const
6721  {
6722  return operator QByteArray();
6723  }
6724 
6725  QDate ToQDate() const
6726  {
6727  return operator QDate();
6728  }
6729 
6730  QDateTime ToQDateTime() const
6731  {
6732  return operator QDateTime();
6733  }
6734 
6735 #endif
6736 
6747  bool ToBool() const;
6748 
6763  bool TryToBool( bool& value ) const noexcept;
6764 
6778  float ToFloat() const;
6779 
6794  bool TryToFloat( float& value ) const noexcept;
6795 
6819  double ToDouble() const;
6820 
6835  bool TryToDouble( double& value ) const noexcept;
6836 
6854  long ToInt() const
6855  {
6856  return ToInt( 0 );
6857  }
6858 
6882  bool TryToInt( int& value ) const noexcept
6883  {
6884  return TryToInt( value, 0 );
6885  }
6886 
6913  long ToInt( int base ) const;
6914 
6932  bool TryToInt( int& value, int base ) const noexcept;
6933 
6951  unsigned long ToUInt() const
6952  {
6953  return ToUInt( 0 );
6954  }
6955 
6979  bool TryToUInt( unsigned& value ) const noexcept
6980  {
6981  return TryToUInt( value, 0 );
6982  }
6983 
7000  unsigned long ToUInt( int base ) const;
7001 
7019  bool TryToUInt( unsigned& value, int base ) const noexcept;
7020 
7035  long long ToInt64() const
7036  {
7037  return ToInt64( 0 );
7038  }
7039 
7064  bool TryToInt64( long long& value ) const noexcept
7065  {
7066  return TryToInt64( value, 0 );
7067  }
7068 
7083  long long ToInt64( int base ) const;
7084 
7102  bool TryToInt64( long long& value, int base ) const noexcept;
7103 
7118  unsigned long long ToUInt64() const
7119  {
7120  return ToUInt64( 0 );
7121  }
7122 
7147  bool TryToUInt64( unsigned long long& value ) const noexcept
7148  {
7149  return TryToUInt64( value, 0 );
7150  }
7151 
7166  unsigned long long ToUInt64( int base ) const;
7167 
7185  bool TryToUInt64( unsigned long long& value, int base ) const noexcept;
7186 
7218  Array<float> ParseListOfFloat( char separator = ',', size_type maxCount = ~size_type( 0 ) ) const;
7219 
7251  Array<double> ParseListOfDouble( char separator = ',', size_type maxCount = ~size_type( 0 ) ) const;
7252 
7253 #ifndef __PCL_NO_STRING_VECTOR
7254 
7262  GenericVector<float> ParseListOfFloatAsVector( char separator = ',', int maxCount = int_max ) const;
7263 
7271  GenericVector<double> ParseListOfDoubleAsVector( char separator = ',', int maxCount = int_max ) const;
7272 
7273 #endif // !__PCL_NO_STRING_VECTOR
7274 
7302  double SexagesimalToDouble( const IsoString& separator = ':' ) const
7303  {
7304  int sign, s1, s2; double s3;
7305  ParseSexagesimal( sign, s1, s2, s3, separator );
7306  return sign*(s1 + (s2 + s3/60)/60);
7307  }
7308 
7309  double SexagesimalToDouble( char separator ) const
7310  {
7311  return SexagesimalToDouble( IsoString( separator ) );
7312  }
7313 
7314  double SexagesimalToDouble( const ustring_base& separator ) const
7315  {
7316  return SexagesimalToDouble( IsoString( separator ) );
7317  }
7318 
7332  double SexagesimalToDouble( const Array<char_type>& separators ) const
7333  {
7334  int sign, s1, s2; double s3;
7335  ParseSexagesimal( sign, s1, s2, s3, separators );
7336  return sign*(s1 + (s2 + s3/60)/60);
7337  }
7338 
7356  bool TrySexagesimalToDouble( double& value, const IsoString& separator = ':' ) const noexcept
7357  {
7358  int sign, s1, s2; double s3;
7359  if ( TryParseSexagesimal( sign, s1, s2, s3, separator ) )
7360  {
7361  value = sign*(s1 + (s2 + s3/60)/60);
7362  return true;
7363  }
7364  return false;
7365  }
7366 
7367  bool TrySexagesimalToDouble( double& value, char separator ) const noexcept
7368  {
7369  return TrySexagesimalToDouble( value, IsoString( separator ) );
7370  }
7371 
7372  bool TrySexagesimalToDouble( double& value, const ustring_base& separator ) const noexcept
7373  {
7374  return TrySexagesimalToDouble( value, IsoString( separator ) );
7375  }
7376 
7389  bool TrySexagesimalToDouble( double& value, const Array<char_type>& separators ) const noexcept
7390  {
7391  int sign, s1, s2; double s3;
7392  if ( TryParseSexagesimal( sign, s1, s2, s3, separators ) )
7393  {
7394  value = sign*(s1 + (s2 + s3/60)/60);
7395  return true;
7396  }
7397  return false;
7398  }
7399 
7426  void ParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const IsoString& separator = ':' ) const;
7427 
7428  void ParseSexagesimal( int& sign, int& s1, int& s2, double& s3, char separator ) const
7429  {
7430  ParseSexagesimal( sign, s1, s2, s3, IsoString( separator ) );
7431  }
7432 
7433  void ParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const ustring_base& separator ) const
7434  {
7435  ParseSexagesimal( sign, s1, s2, s3, IsoString( separator ) );
7436  }
7437 
7452  void ParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const Array<char_type>& separators ) const;
7453 
7468  bool TryParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const IsoString& separator = ':' ) const noexcept;
7469 
7470  bool TryParseSexagesimal( int& sign, int& s1, int& s2, double& s3, char separator ) const noexcept
7471  {
7472  return TryParseSexagesimal( sign, s1, s2, s3, IsoString( separator ) );
7473  }
7474 
7475  bool TryParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const ustring_base& separator ) const noexcept
7476  {
7477  return TryParseSexagesimal( sign, s1, s2, s3, IsoString( separator ) );
7478  }
7479 
7493  bool TryParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const Array<char_type>& separators ) const noexcept;
7494 
7519  static IsoString ToSexagesimal( int sign, double s1, double s2, double s3,
7521 
7533  {
7534  return ToSexagesimal( (d < 0) ? -1 : +1, Abs( d ), 0, 0, options );
7535  }
7536 
7560  void ParseISO8601DateTime( int& year, int& month, int& day, double& dayf, double& tz ) const;
7561 
7575  bool TryParseISO8601DateTime( int& year, int& month, int& day, double& dayf, double& tz ) const noexcept;
7576 
7600  static IsoString ToISO8601DateTime( int year, int month, int day, double dayf, double tz = 0,
7602 
7614 
7626 
7634  static IsoString ToHex( const void* data, size_type length );
7635 
7648  template <class C>
7649  static IsoString ToHex( const C& c )
7650  {
7651  return ToHex( c.Begin(), c.Length()*sizeof( *c.Begin() ) );
7652  }
7653 
7665  static IsoString ToBase64( const void* data, size_type length );
7666 
7680  template <class C>
7681  static IsoString ToBase64( const C& c )
7682  {
7683  return ToBase64( c.Begin(), c.Length()*sizeof( *c.Begin() ) );
7684  }
7685 
7700  static IsoString ToBase64URL( const void* data, size_type length )
7701  {
7702  IsoString b64 = ToBase64( data, length );
7703  b64.DeleteChar( '=' );
7704  for ( char_type& c : b64 )
7705  if ( c == '+' )
7706  c = '-';
7707  else if ( c == '/' )
7708  c = '_';
7709  return b64;
7710  }
7711 
7723  template <class C>
7724  static IsoString ToBase64URL( const C& c )
7725  {
7726  return ToBase64URL( c.Begin(), c.Length()*sizeof( *c.Begin() ) );
7727  }
7728 
7737  {
7738  return ByteArray( Begin(), End() );
7739  }
7740 
7754 
7768 
7779  {
7780  IsoString b64 = *this;
7781  for ( char_type& c : b64 )
7782  if ( c == '-' )
7783  c = '+';
7784  else if ( c == '_' )
7785  c = '/';
7786  return b64.FromBase64();
7787  }
7788 
7795  static IsoString Random( size_type n, RandomizationOptions options = RandomizationOption::Default );
7796 
7808  static IsoString UUID();
7809 };
7810 
7811 // ----------------------------------------------------------------------------
7812 
7822 {
7823  IsoString s( s1 );
7824  s.Append( s2 );
7825  return s;
7826 }
7827 
7834 {
7835  s1.Append( s2 );
7836  return IsoString( std::move( s1 ) );
7837 }
7838 
7845 {
7846  s1.Append( s2 );
7847  return std::move( s1 );
7848 }
7849 
7856 {
7857  s2.Prepend( s1 );
7858  return IsoString( std::move( s2 ) );
7859 }
7860 
7867 {
7868  s2.Prepend( s1 );
7869  return std::move( s2 );
7870 }
7871 
7878 {
7879  s1.Append( s2 );
7880  return IsoString( std::move( s1 ) );
7881 }
7882 
7889 {
7890  s1.Append( s2 );
7891  return std::move( s1 );
7892 }
7893 
7900 {
7901  s1.Append( s2 );
7902  return IsoString( std::move( s1 ) );
7903 }
7904 
7911 {
7912  s1.Append( s2 );
7913  return std::move( s1 );
7914 }
7915 
7916 // ----------------------------------------------------------------------------
7917 
7924 {
7925  IsoString s = s1;
7926  s.Append( t2 );
7927  return s;
7928 }
7929 
7936 {
7937  s1.Append( t2 );
7938  return IsoString( std::move( s1 ) );
7939 }
7940 
7947 {
7948  s1.Append( t2 );
7949  return std::move( s1 );
7950 }
7951 
7958 {
7959  IsoString s = s2;
7960  s.Prepend( t1 );
7961  return s;
7962 }
7963 
7970 {
7971  s2.Prepend( t1 );
7972  return IsoString( std::move( s2 ) );
7973 }
7974 
7981 {
7982  s2.Prepend( t1 );
7983  return std::move( s2 );
7984 }
7985 
7986 // ----------------------------------------------------------------------------
7987 
7994 {
7995  IsoString s = s1;
7996  s.Append( c2 );
7997  return s;
7998 }
7999 
8006 {
8007  s1.Append( c2 );
8008  return IsoString( std::move( s1 ) );
8009 }
8010 
8017 {
8018  s1.Append( c2 );
8019  return std::move( s1 );
8020 }
8021 
8028 {
8029  IsoString s = s2;
8030  s.Prepend( c1 );
8031  return s;
8032 }
8033 
8040 {
8041  s2.Prepend( c1 );
8042  return IsoString( std::move( s2 ) );
8043 }
8044 
8051 {
8052  s2.Prepend( c1 );
8053  return std::move( s2 );
8054 }
8055 
8056 // ----------------------------------------------------------------------------
8057 
8064 {
8065  s1.Append( s2 );
8066  return s1;
8067 }
8068 
8075 {
8076  s1.Append( s2 );
8077  return s1;
8078 }
8079 
8086 {
8087  s1.Append( t2 );
8088  return s1;
8089 }
8090 
8097 {
8098  s1.Append( t2 );
8099  return s1;
8100 }
8101 
8108 {
8109  s1.Append( c2 );
8110  return s1;
8111 }
8112 
8119 {
8120  s1.Append( c2 );
8121  return s1;
8122 }
8123 
8124 // ----------------------------------------------------------------------------
8125 
8126 #ifndef __PCL_NO_STRING_OSTREAM
8127 
8128 inline std::ostream& operator <<( std::ostream& o, const IsoString& s )
8129 {
8130  return o << s.c_str();
8131 }
8132 
8133 #endif
8134 
8135 // ----------------------------------------------------------------------------
8136 
8147 class PCL_CLASS String : public GenericString<char16_type, CharTraits, PCL_STRING_ALLOCATOR>
8148 {
8149 public:
8150 
8155 
8160 
8165 
8170 
8175 
8180 
8185 
8190 
8195 
8200 
8201  /*
8202  * Null-terminated UTF-16 string - C++11 compatibility.
8203  */
8204  using c16_string = char16_t*;
8205 
8206  /*
8207  * Immutable null-terminated UTF-16 string - C++11 compatibility.
8208  */
8209  using const_c16_string = const char16_t*;
8210 
8215 
8220 
8225 
8230 
8239 
8244 
8249 
8254 
8259 
8264 
8269 
8270  // -------------------------------------------------------------------------
8271 
8275  String() = default;
8276 
8281  String( const string_base& s )
8282  : string_base( s )
8283  {
8284  }
8285 
8289  String( const String& ) = default;
8290 
8296  : string_base( std::move( s ) )
8297  {
8298  }
8299 
8303  String( String&& ) = default;
8304 
8309  String( const string8_base& s )
8310  {
8311  Assign( s );
8312  }
8313 
8318  : string_base( t )
8319  {
8320  }
8321 
8327  : string_base( t, i, n )
8328  {
8329  }
8330 
8335  : string_base( c, n )
8336  {
8337  }
8338 
8344  : string_base( i, j )
8345  {
8346  }
8347 
8356  String( std::initializer_list<char_type> l )
8357  : String( l.begin(), l.end() )
8358  {
8359  }
8360 
8365  String( const char16_t* t )
8366  : string_base( reinterpret_cast<const_iterator>( t ) )
8367  {
8368  }
8369 
8375  String( const char16_t* t, size_type i, size_type n )
8376  : string_base( reinterpret_cast<const_iterator>( t ), i, n )
8377  {
8378  }
8379 
8383  String( char16_t c, size_type n )
8384  : string_base( char_type( c ), n )
8385  {
8386  }
8387 
8392  String( const wchar_t* t )
8393  {
8394  Assign( t );
8395  }
8396 
8402  String( const wchar_t* t, size_type i, size_type n )
8403  {
8404  Assign( t, i, n );
8405  }
8406 
8410  String( wchar_t c, size_type n )
8411  : string_base( char_type( c ), n )
8412  {
8413  }
8414 
8420  {
8421  Assign( t );
8422  }
8423 
8430  {
8431  Assign( t, i, n );
8432  }
8433 
8439  {
8440  Assign( i, j );
8441  }
8442 
8451  String( std::initializer_list<char8_type> l )
8452  : String( l.begin(), l.end() )
8453  {
8454  }
8455 
8461  : string_base( char_type( c ), n )
8462  {
8463  }
8464 
8468  explicit
8469  String( bool x )
8470  {
8471  Assign( x ? "true" : "false" );
8472  }
8473 
8478  explicit
8479  String( short x )
8480  {
8481  (void)Format( L"%hd", x );
8482  }
8483 
8488  explicit
8489  String( unsigned short x )
8490  {
8491  (void)Format( L"%hu", x );
8492  }
8493 
8498  explicit
8499  String( int x )
8500  {
8501  (void)Format( L"%i", x );
8502  }
8503 
8508  explicit
8509  String( unsigned int x )
8510  {
8511  (void)Format( L"%u", x );
8512  }
8513 
8518  explicit
8519  String( long x )
8520  {
8521  (void)Format( L"%ld", x );
8522  }
8523 
8528  explicit
8529  String( unsigned long x )
8530  {
8531  (void)Format( L"%lu", x );
8532  }
8533 
8538  explicit
8539  String( long long x )
8540  {
8541  (void)Format( L"%lli", x );
8542  }
8543 
8548  explicit
8549  String( unsigned long long x )
8550  {
8551  (void)Format( L"%llu", x );
8552  }
8553 
8558  explicit
8559  String( float x )
8560  {
8561  (void)Format( L"%.7g", x );
8562  }
8563 
8568  explicit
8569  String( double x )
8570  {
8571  (void)Format( L"%.16g", x );
8572  }
8573 
8578  explicit
8579  String( long double x )
8580  {
8581 #ifdef _MSC_VER
8582  (void)Format( L"%.16Lg", x );
8583 #else
8584  (void)Format( L"%.18Lg", x );
8585 #endif
8586  }
8587 
8588 #ifndef __PCL_NO_STRING_COMPLEX
8589 
8593  explicit
8595  {
8596  (void)Format( L"{%.7g,%.7g}", x.Real(), x.Imag() );
8597  }
8598 
8602  explicit
8604  {
8605  (void)Format( L"{%.16g,%.16g}", x.Real(), x.Imag() );
8606  }
8607 
8611  explicit
8613  {
8614 #ifdef _MSC_VER
8615  (void)Format( L"{%.16Lg,%.16Lg}", x.Real(), x.Imag() );
8616 #else
8617  (void)Format( L"{%.18Lg,%.18Lg}", x.Real(), x.Imag() );
8618 #endif
8619  }
8620 
8621 #endif // !__PCL_NO_STRING_COMPLEX
8622 
8623 #ifdef __PCL_QT_INTERFACE
8624 
8625  explicit
8626  String( const QString& qs )
8627  : string_base( qs.isEmpty() ? iterator( nullptr ) : iterator( PCL_GET_CHAR16PTR_FROM_QSTRING( qs ) ) )
8628  {
8629  }
8630 
8631  explicit
8632  String( const QDate& qd )
8633  : string_base( iterator( PCL_GET_CHAR16PTR_FROM_QSTRING( qd.toString( PCL_QDATE_FMT_STR ) ) ) )
8634  {
8635  }
8636 
8637  explicit
8638  String( const QDateTime& qdt )
8639  : string_base( iterator( PCL_GET_CHAR16PTR_FROM_QSTRING( qdt.toString( PCL_QDATETIME_FMT_STR ) ) ) )
8640  {
8641  }
8642 
8643 #endif
8644 
8645  // -------------------------------------------------------------------------
8646 
8650  String& operator =( const String& s )
8651  {
8652  string_base::Assign( s );
8653  return *this;
8654  }
8655 
8659  String& operator =( String&& s )
8660  {
8661  string_base::Transfer( s );
8662  return *this;
8663  }
8664 
8669  String& operator =( const string_base& s )
8670  {
8671  string_base::Assign( s );
8672  return *this;
8673  }
8674 
8679  String& operator =( string_base&& s )
8680  {
8681  string_base::Transfer( s );
8682  return *this;
8683  }
8684 
8689  String& operator =( const string8_base& s )
8690  {
8691  Assign( s );
8692  return *this;
8693  }
8694 
8699  String& operator =( const_iterator t )
8700  {
8701  string_base::Assign( t );
8702  return *this;
8703  }
8704 
8709  String& operator =( char_type c )
8710  {
8711  string_base::Assign( c );
8712  return *this;
8713  }
8714 
8719  String& operator =( const char16_t* t )
8720  {
8721  string_base::Assign( reinterpret_cast<const_iterator>( t ) );
8722  return *this;
8723  }
8724 
8729  String& operator =( char16_t c )
8730  {
8731  string_base::Assign( char_type( c ) );
8732  return *this;
8733  }
8734 
8739  String& operator =( const wchar_t* t )
8740  {
8741  Assign( t );
8742  return *this;
8743  }
8744 
8749  String& operator =( wchar_t c )
8750  {
8751  Assign( c );
8752  return *this;
8753  }
8754 
8759  String& operator =( const_c_string8 t )
8760  {
8761  Assign( t );
8762  return *this;
8763  }
8764 
8769  String& operator =( char8_type c )
8770  {
8771  Assign( c );
8772  return *this;
8773  }
8774 
8775 #ifdef __PCL_QT_INTERFACE
8776 
8777  String& operator =( const QString& qs )
8778  {
8779  if ( qs.isEmpty() )
8780  Clear();
8781  else
8782  string_base::Assign( PCL_GET_CHAR16PTR_FROM_QSTRING( qs ) );
8783  return *this;
8784  }
8785 
8786  String& operator =( const QDate& qd )
8787  {
8788  string_base::Assign( PCL_GET_CHAR16PTR_FROM_QSTRING( qd.toString( PCL_QDATE_FMT_STR ) ) );
8789  return *this;
8790  }
8791 
8792  String& operator =( const QDateTime& qdt )
8793  {
8794  string_base::Assign( PCL_GET_CHAR16PTR_FROM_QSTRING( qdt.toString( PCL_QDATETIME_FMT_STR ) ) );
8795  return *this;
8796  }
8797 
8798 #endif
8799 
8800  // -------------------------------------------------------------------------
8801 
8805  void Assign( const String& s )
8806  {
8807  string_base::Assign( s );
8808  }
8809 
8814  void Assign( const String& s, size_type i, size_type n )
8815  {
8816  string_base::Assign( s, i, n );
8817  }
8818 
8823  {
8824  string_base::Assign( t );
8825  }
8826 
8832  {
8833  string_base::Assign( i, j );
8834  }
8835 
8842  void Assign( std::initializer_list<char_type> l )
8843  {
8844  Assign( l.begin(), l.end() );
8845  }
8846 
8853  {
8854  string_base::Assign( t, i, n );
8855  }
8856 
8860  void Assign( char_type c, size_type n = 1 )
8861  {
8862  string_base::Assign( c, n );
8863  }
8864 
8868  void Assign( const char16_t* t )
8869  {
8870  string_base::Assign( reinterpret_cast<const_iterator>( t ) );
8871  }
8872 
8878  void Assign( const char16_t* t, size_type i, size_type n )
8879  {
8880  string_base::Assign( reinterpret_cast<const_iterator>( t ), i, n );
8881  }
8882 
8886  void Assign( char16_t c, size_type n = 1 )
8887  {
8888  string_base::Assign( char_type( c ), n );
8889  }
8890 
8894  void Assign( const wchar_t* t );
8895 
8901  void Assign( const wchar_t* t, size_type i, size_type n );
8902 
8906  void Assign( wchar_t c, size_type n = 1 )
8907  {
8908  string_base::Assign( char_type( c ), n );
8909  }
8910 
8914  void Assign( const string8_base& s )
8915  {
8916  size_type n = s.Length();
8917  if ( n > 0 )
8918  {
8919  MaybeReallocate( n );
8920  const_char8_iterator t = s.Begin();
8921  PCL_IVDEP
8922  for ( iterator i = m_data->string; i < m_data->end; ++i, ++t )
8923  *i = char_type( uint8( *t ) );
8924  }
8925  else
8926  Clear();
8927  }
8928 
8934  {
8935  size_type n = char8_traits::Length( t );
8936  if ( n > 0 )
8937  {
8938  MaybeReallocate( n );
8939  PCL_IVDEP
8940  for ( iterator i = m_data->string; i < m_data->end; ++i, ++t )
8941  *i = char_type( uint8( *t ) );
8942  }
8943  else
8944  Clear();
8945  }
8946 
8953  {
8954  size_type len = char8_traits::Length( t );
8955  if ( i < len )
8956  {
8957  n = pcl::Min( n, len-i );
8958  MaybeReallocate( n );
8959  t += i;
8960  PCL_IVDEP
8961  for ( iterator i = m_data->string; i < m_data->end; ++i, ++t )
8962  *i = char_type( uint8( *t ) );
8963  }
8964  else
8965  Clear();
8966  }
8967 
8973  {
8974  if ( p < q )
8975  {
8976  MaybeReallocate( q - p );
8977  PCL_IVDEP
8978  for ( iterator i = m_data->string; i < m_data->end; ++i, ++p )
8979  *i = char_type( uint8( *p ) );
8980  }
8981  else
8982  Clear();
8983  }
8984 
8992  void Assign( std::initializer_list<char8_type> l )
8993  {
8994  Assign( l.begin(), l.end() );
8995  }
8996 
9000  void Assign( char8_type c, size_type n = 1 )
9001  {
9002  string_base::Assign( char_type( c ), n );
9003  }
9004 
9005  // -------------------------------------------------------------------------
9006 
9007  void Insert( size_type i, const String& s )
9008  {
9009  string_base::Insert( i, s );
9010  }
9011 
9012  void Insert( size_type i, const_iterator p, const_iterator q )
9013  {
9014  string_base::Insert( i, p, q );
9015  }
9016 
9017  void Insert( size_type i, const_iterator t )
9018  {
9019  string_base::Insert( i, t );
9020  }
9021 
9022  void Insert( size_type i, const_iterator t, size_type n )
9023  {
9024  string_base::Insert( i, t, n );
9025  }
9026 
9027  void Insert( size_type i, char_type c, size_type n = 1 )
9028  {
9029  string_base::Insert( i, c, n );
9030  }
9031 
9032  void Insert( size_type i, const char16_t* t )
9033  {
9034  string_base::Insert( i, reinterpret_cast<const_iterator>( t ) );
9035  }
9036 
9037  void Insert( size_type i, char16_t c, size_type n = 1 )
9038  {
9039  string_base::Insert( i, String( c, n ) );
9040  }
9041 
9042  void Insert( size_type i, const wchar_t* t )
9043  {
9044 #ifdef __PCL_WINDOWS
9045  string_base::Insert( i, reinterpret_cast<const_iterator>( t ) );
9046 #else
9047  string_base::Insert( i, String( t ) );
9048 #endif
9049  }
9050 
9051  void Insert( size_type i, wchar_t c, size_type n = 1 )
9052  {
9053  string_base::Insert( i, String( c, n ) );
9054  }
9055 
9056  void Insert( size_type i, const string8_base& s, size_type n )
9057  {
9058  n = pcl::Min( n, s.Length() );
9059  if ( n > 0 )
9060  {
9061  UninitializedGrow( i, n ); // -> 0 <= i <= len
9062  const_char8_iterator t = s.Begin();
9063  PCL_IVDEP
9064  for ( iterator p = m_data->string+i, q = p+n; p < q; ++p, ++t )
9065  *p = char_type( uint8( *t ) );
9066  }
9067  }
9068 
9069  void Insert( size_type i, const string8_base& s )
9070  {
9071  size_type n = s.Length();
9072  if ( n > 0 )
9073  {
9074  UninitializedGrow( i, n ); // -> 0 <= i <= len
9075  const_char8_iterator t = s.Begin();
9076  PCL_IVDEP
9077  for ( iterator p = m_data->string+i, q = p+n; p < q; ++p, ++t )
9078  *p = char_type( uint8( *t ) );
9079  }
9080  }
9081 
9082  void Insert( size_type i, const_c_string8 t )
9083  {
9084  size_type n = char8_traits::Length( t );
9085  if ( n > 0 )
9086  {
9087  UninitializedGrow( i, n ); // -> 0 <= i <= len
9088  PCL_IVDEP
9089  for ( iterator p = m_data->string+i, q = p+n; p < q; ++p, ++t )
9090  *p = char_type( uint8( *t ) );
9091  }
9092  }
9093 
9094  void Insert( size_type i, const_c_string8 t, size_type n )
9095  {
9096  n = pcl::Min( n, char8_traits::Length( t ) );
9097  if ( n > 0 )
9098  {
9099  UninitializedGrow( i, n ); // -> 0 <= i <= len
9100  PCL_IVDEP
9101  for ( iterator p = m_data->string+i, q = p+n; p < q; ++p, ++t )
9102  *p = char_type( uint8( *t ) );
9103  }
9104  }
9105 
9106  void Insert( size_type i, const_char8_iterator p, const_char8_iterator q )
9107  {
9108  if ( p < q )
9109  {
9110  size_type n = q - p;
9111  UninitializedGrow( i, n ); // -> 0 <= i <= len
9112  PCL_IVDEP
9113  for ( iterator r = m_data->string+i, s = r+n; r < s; ++r, ++p )
9114  *r = char_type( uint8( *p ) );
9115  }
9116  }
9117 
9118  void Insert( size_type i, char8_type c, size_type n = 1 )
9119  {
9120  string_base::Insert( i, char_type( c ), n );
9121  }
9122 
9123  // -------------------------------------------------------------------------
9124 
9125  void Append( const String& s )
9126  {
9127  string_base::Append( s );
9128  }
9129 
9130  String& operator +=( const String& s )
9131  {
9132  Append( s );
9133  return *this;
9134  }
9135 
9136  void Append( const_iterator i, const_iterator j )
9137  {
9138  string_base::Append( i, j );
9139  }
9140 
9141  void Append( const_iterator t )
9142  {
9143  string_base::Append( t );
9144  }
9145 
9146  String& operator +=( const_iterator t )
9147  {
9148  Append( t );
9149  return *this;
9150  }
9151 
9152  void Append( const_iterator t, size_type n )
9153  {
9154  string_base::Append( t, n );
9155  }
9156 
9157  void Append( char_type c, size_type n = 1 )
9158  {
9159  string_base::Append( c, n );
9160  }
9161 
9162  String& operator +=( char_type c )
9163  {
9164  Append( c );
9165  return *this;
9166  }
9167 
9168  void Append( const char16_t* t )
9169  {
9170  string_base::Append( reinterpret_cast<const_iterator>( t ) );
9171  }
9172 
9173  String& operator +=( const char16_t* t )
9174  {
9175  Append( t );
9176  return *this;
9177  }
9178 
9179  void Append( char16_t c, size_type n = 1 )
9180  {
9181  string_base::Append( char_type( c ), n );
9182  }
9183 
9184  String& operator +=( char16_t c )
9185  {
9186  Append( c );
9187  return *this;
9188  }
9189 
9190  void Append( const wchar_t* t )
9191  {
9192 #ifdef __PCL_WINDOWS
9193  string_base::Append( reinterpret_cast<const_iterator>( t ) );
9194 #else
9195  string_base::Append( String( t ) );
9196 #endif
9197  }
9198 
9199  String& operator +=( const wchar_t* t )
9200  {
9201  Append( t );
9202  return *this;
9203  }
9204 
9205  void Append( wchar_t c, size_type n = 1 )
9206  {
9207  string_base::Append( char_type( c ), n );
9208  }
9209 
9210  String& operator +=( wchar_t c )
9211  {
9212  Append( c );
9213  return *this;
9214  }
9215 
9216  void Append( const string8_base& s )
9217  {
9218  Insert( maxPos, s );
9219  }
9220 
9221  String& operator +=( const string8_base& s )
9222  {
9223  Append( s );
9224  return *this;
9225  }
9226 
9227  void Append( const string8_base& s, size_type n )
9228  {
9229  Insert( maxPos, s, n );
9230  }
9231 
9232  void Append( const_c_string8 t )
9233  {
9234  Insert( maxPos, t );
9235  }
9236 
9237  String& operator +=( const_c_string8 t )
9238  {
9239  Append( t );
9240  return *this;
9241  }
9242 
9243  void Append( const_c_string8 t, size_type n )
9244  {
9245  Insert( maxPos, t, n );
9246  }
9247 
9248  void Append( const_char8_iterator p, const_char8_iterator q )
9249  {
9250  Insert( maxPos, p, q );
9251  }
9252 
9253  void Append( char8_type c, size_type n = 1 )
9254  {
9255  string_base::Append( char_type( c ), n );
9256  }
9257 
9258  String& operator +=( char8_type c )
9259  {
9260  Append( c );
9261  return *this;
9262  }
9263 
9264  void Add( const String& s )
9265  {
9266  Append( s );
9267  }
9268 
9269  void Add( const_iterator i, const_iterator j )
9270  {
9271  Append( i, j );
9272  }
9273 
9274  void Add( const_iterator t )
9275  {
9276  Append( t );
9277  }
9278 
9279  void Add( const_iterator t, size_type n )
9280  {
9281  Append( t, n );
9282  }
9283 
9284  void Add( char_type c, size_type n = 1 )
9285  {
9286  Append( c, n );
9287  }
9288 
9289  void Add( const char16_t* t )
9290  {
9291  Append( t );
9292  }
9293 
9294  void Add( char16_t c, size_type n = 1 )
9295  {
9296  Append( c, n );
9297  }
9298 
9299  void Add( const wchar_t* t )
9300  {
9301  Append( t );
9302  }
9303 
9304  void Add( wchar_t c, size_type n = 1 )
9305  {
9306  Append( c, n );
9307  }
9308 
9309  void Add( const string8_base& s )
9310  {
9311  Append( s );
9312  }
9313 
9314  void Add( const string8_base& s, size_type n )
9315  {
9316  Append( s, n );
9317  }
9318 
9319  void Add( const_c_string8 t )
9320  {
9321  Append( t );
9322  }
9323 
9324  void Add( const_c_string8 t, size_type n )
9325  {
9326  Append( t, n );
9327  }
9328 
9329  void Add( const_char8_iterator p, const_char8_iterator q )
9330  {
9331  Append( p, q );
9332  }
9333 
9334  void Add( char8_type c, size_type n = 1 )
9335  {
9336  Append( c, n );
9337  }
9338 
9339  // -------------------------------------------------------------------------
9340 
9341  void Prepend( const String& s )
9342  {
9343  string_base::Prepend( s );
9344  }
9345 
9346  String& operator -=( const String& s )
9347  {
9348  Prepend( s );
9349  return *this;
9350  }
9351 
9352  void Prepend( const_iterator i, const_iterator j )
9353  {
9354  string_base::Prepend( i, j );
9355  }
9356 
9357  void Prepend( const_iterator t )
9358  {
9359  string_base::Prepend( t );
9360  }
9361 
9362  String& operator -=( const_iterator t )
9363  {
9364  Prepend( t );
9365  return *this;
9366  }
9367 
9368  void Prepend( const_iterator t, size_type n )
9369  {
9370  string_base::Prepend( t, n );
9371  }
9372 
9373  void Prepend( char_type c, size_type n = 1 )
9374  {
9375  string_base::Prepend( c, n );
9376  }
9377 
9378  String& operator -=( char_type c )
9379  {
9380  Prepend( c );
9381  return *this;
9382  }
9383 
9384  void Prepend( const char16_t* t )
9385  {
9386  string_base::Prepend( reinterpret_cast<const_iterator>( t ) );
9387  }
9388 
9389  String& operator -=( const char16_t* t )
9390  {
9391  Prepend( t );
9392  return *this;
9393  }
9394 
9395  void Prepend( char16_t c, size_type n = 1 )
9396  {
9397  string_base::Prepend( char_type( c ), n );
9398  }
9399 
9400  String& operator -=( char16_t c )
9401  {
9402  Prepend( c );
9403  return *this;
9404  }
9405 
9406  void Prepend( const wchar_t* t )
9407  {
9408 #ifdef __PCL_WINDOWS
9409  string_base::Prepend( reinterpret_cast<const_iterator>( t ) );
9410 #else
9411  string_base::Prepend( String( t ) );
9412 #endif
9413  }
9414 
9415  String& operator -=( const wchar_t* t )
9416  {
9417  Prepend( t );
9418  return *this;
9419  }
9420 
9421  void Prepend( wchar_t c, size_type n = 1 )
9422  {
9423  string_base::Prepend( char_type( c ), n );
9424  }
9425 
9426  String& operator -=( wchar_t c )
9427  {
9428  Prepend( c );
9429  return *this;
9430  }
9431 
9432  void Prepend( const string8_base& s )
9433  {
9434  Insert( 0, s );
9435  }
9436 
9437  String& operator -=( const string8_base& s )
9438  {
9439  Prepend( s );
9440  return *this;
9441  }
9442 
9443  void Prepend( const string8_base& s, size_type n )
9444  {
9445  Insert( 0, s, n );
9446  }
9447 
9448  void Prepend( const_c_string8 t )
9449  {
9450  Insert( 0, t );
9451  }
9452 
9453  String& operator -=( const_c_string8 t )
9454  {
9455  Prepend( t );
9456  return *this;
9457  }
9458 
9459  void Prepend( const_c_string8 t, size_type n )
9460  {
9461  Insert( 0, t, n );
9462  }
9463 
9464  void Prepend( const_char8_iterator p, const_char8_iterator q )
9465  {
9466  Insert( 0, p, q );
9467  }
9468 
9469  void Prepend( char8_type c, size_type n = 1 )
9470  {
9471  string_base::Prepend( String( c, n ) );
9472  }
9473 
9474  String& operator -=( char8_type c )
9475  {
9476  Prepend( c );
9477  return *this;
9478  }
9479 
9480  // -------------------------------------------------------------------------
9481 
9482  void Replace( size_type i, size_type n, const String& s )
9483  {
9484  string_base::Replace( i, n, s );
9485  }
9486 
9487  void Replace( size_type i, size_type n, const_iterator t )
9488  {
9489  string_base::Replace( i, n, t );
9490  }
9491 
9492  void Replace( size_type i, size_type n, char_type c, size_type nc = 1 )
9493  {
9494  string_base::Replace( i, n, c, nc );
9495  }
9496 
9497  void Replace( size_type i, size_type n, const char16_t* t )
9498  {
9499  string_base::Replace( i, n, reinterpret_cast<const_iterator>( t ) );
9500  }
9501 
9502  void Replace( size_type i, size_type n, char16_t c, size_type nc = 1 )
9503  {
9504  string_base::Replace( i, n, char_type( c ), nc );
9505  }
9506 
9507  void Replace( size_type i, size_type n, const wchar_t* t )
9508  {
9509 #ifdef __PCL_WINDOWS
9510  string_base::Replace( i, n, reinterpret_cast<const_iterator>( t ) );
9511 #else
9512  string_base::Replace( i, n, String( t ) );
9513 #endif
9514  }
9515 
9516  void Replace( size_type i, size_type n, wchar_t c, size_type nc = 1 )
9517  {
9518  string_base::Replace( i, n, char_type( c ), nc );
9519  }
9520 
9521  void Replace( size_type i, size_type n, const_c_string8 t )
9522  {
9523  if ( n > 0 )
9524  {
9525  size_type len = Length();
9526  if ( i < len )
9527  {
9528  n = pcl::Min( n, len-i );
9529  if ( n == len )
9530  Assign( t );
9531  else
9532  {
9533  size_type nt = char8_traits::Length( t );
9534  if ( nt > 0 )
9535  {
9536  if ( n < nt )
9537  UninitializedGrow( i, nt-n );
9538  else if ( nt < n )
9539  Delete( i, n-nt );
9540  else
9541  EnsureUnique();
9542 
9543  PCL_IVDEP
9544  for ( iterator p = m_data->string+i, q = p+nt; p < q; ++p, ++t )
9545  *p = char_type( *t );
9546  }
9547  else
9548  Delete( i, n );
9549  }
9550  }
9551  }
9552  }
9553 
9554  void Replace( size_type i, size_type n, char8_type c, size_type nc = 1 )
9555  {
9556  string_base::Replace( i, n, char_type( c ), nc );
9557  }
9558 
9559  // -------------------------------------------------------------------------
9560 
9561  void ReplaceChar( char_type c1, char_type c2, size_type i = 0, size_type n = maxPos )
9562  {
9563  string_base::ReplaceChar( c1, c2, i, n );
9564  }
9565 
9566  void ReplaceCharIC( char_type c1, char_type c2, size_type i = 0, size_type n = maxPos )
9567  {
9568  string_base::ReplaceCharIC( c1, c2, i, n );
9569  }
9570 
9571  void ReplaceChar( char16_t c1, char16_t c2, size_type i = 0, size_type n = maxPos )
9572  {
9573  string_base::ReplaceChar( char_type( c1 ), char_type( c2 ), i, n );
9574  }
9575 
9576  void ReplaceCharIC( char16_t c1, char16_t c2, size_type i = 0, size_type n = maxPos )
9577  {
9578  string_base::ReplaceCharIC( char_type( c1 ), char_type( c2 ), i, n );
9579  }
9580 
9581  void ReplaceChar( wchar_t c1, wchar_t c2, size_type i = 0, size_type n = maxPos )
9582  {
9583  string_base::ReplaceChar( char_type( c1 ), char_type( c2 ), i, n );
9584  }
9585 
9586  void ReplaceCharIC( wchar_t c1, wchar_t c2, size_type i = 0, size_type n = maxPos )
9587  {
9588  string_base::ReplaceCharIC( char_type( c1 ), char_type( c2 ), i, n );
9589  }
9590 
9591  void ReplaceChar( char8_type c1, char8_type c2, size_type i = 0, size_type n = maxPos )
9592  {
9593  string_base::ReplaceChar( char_type( c1 ), char_type( c2 ), i, n );
9594  }
9595 
9596  void ReplaceCharIC( char8_type c1, char8_type c2, size_type i = 0, size_type n = maxPos )
9597  {
9598  string_base::ReplaceCharIC( char_type( c1 ), char_type( c2 ), i, n );
9599  }
9600 
9601  // -------------------------------------------------------------------------
9602 
9603  void ReplaceString( const String& s1, const String& s2, size_type i = 0 )
9604  {
9605  string_base::ReplaceString( s1, s2, i );
9606  }
9607 
9608  void ReplaceStringIC( const String& s1, const String& s2, size_type i = 0 )
9609  {
9610  string_base::ReplaceStringIC( s1, s2, i );
9611  }
9612 
9613  void ReplaceString( const_iterator t1, const_iterator t2, size_type i = 0 )
9614  {
9615  string_base::ReplaceString( t1, t2, i );
9616  }
9617 
9618  void ReplaceStringIC( const_iterator t1, const_iterator t2, size_type i = 0 )
9619  {
9620  string_base::ReplaceStringIC( t1, t2, i );
9621  }
9622 
9623  void ReplaceString( const char16_t* t1, const char16_t* t2, size_type i = 0 )
9624  {
9625  string_base::ReplaceString( reinterpret_cast<const_iterator>( t1 ),
9626  reinterpret_cast<const_iterator>( t2 ), i );
9627  }
9628 
9629  void ReplaceStringIC( const char16_t* t1, const char16_t* t2, size_type i = 0 )
9630  {
9631  string_base::ReplaceStringIC( reinterpret_cast<const_iterator>( t1 ),
9632  reinterpret_cast<const_iterator>( t2 ), i );
9633  }
9634 
9635  void ReplaceString( const wchar_t* t1, const wchar_t* t2, size_type i = 0 )
9636  {
9637 #ifdef __PCL_WINDOWS
9638  string_base::ReplaceString( reinterpret_cast<const_iterator>( t1 ),
9639  reinterpret_cast<const_iterator>( t2 ), i );
9640 #else
9641  string_base::ReplaceString( String( t1 ), String( t2 ), i );
9642 #endif
9643  }
9644 
9645  void ReplaceStringIC( const wchar_t* t1, const wchar_t* t2, size_type i = 0 )
9646  {
9647 #ifdef __PCL_WINDOWS
9648  string_base::ReplaceStringIC( reinterpret_cast<const_iterator>( t1 ),
9649  reinterpret_cast<const_iterator>( t2 ), i );
9650 #else
9651  string_base::ReplaceStringIC( String( t1 ), String( t2 ), i );
9652 #endif
9653  }
9654 
9655  void ReplaceString( const_c_string8 t1, const_c_string8 t2, size_type i = 0 )
9656  {
9657  string_base::ReplaceString( String( t1 ), String( t2 ), i );
9658  }
9659 
9660  void ReplaceStringIC( const_c_string8 t1, const_c_string8 t2, size_type i = 0 )
9661  {
9662  string_base::ReplaceStringIC( String( t1 ), String( t2 ), i );
9663  }
9664 
9665  // -------------------------------------------------------------------------
9666 
9667  void DeleteChar( char_type c, size_type i = 0 )
9668  {
9669  string_base::DeleteChar( c, i );
9670  }
9671 
9672  void DeleteCharIC( char_type c, size_type i = 0 )
9673  {
9674  string_base::DeleteCharIC( c, i );
9675  }
9676 
9677  void DeleteChar( char16_t c, size_type i = 0 )
9678  {
9679  string_base::DeleteChar( char_type( c ), i );
9680  }
9681 
9682  void DeleteCharIC( char16_t c, size_type i = 0 )
9683  {
9684  string_base::DeleteCharIC( char_type( c ), i );
9685  }
9686 
9687  void DeleteChar( wchar_t c, size_type i = 0 )
9688  {
9689  string_base::DeleteChar( char_type( c ), i );
9690  }
9691 
9692  void DeleteCharIC( wchar_t c, size_type i = 0 )
9693  {
9694  string_base::DeleteCharIC( char_type( c ), i );
9695  }
9696 
9697  void DeleteChar( char8_type c, size_type i = 0 )
9698  {
9699  string_base::DeleteChar( char_type( c ), i );
9700  }
9701 
9702  void DeleteCharIC( char8_type c, size_type i = 0 )
9703  {
9704  string_base::DeleteCharIC( char_type( c ), i );
9705  }
9706 
9707  // -------------------------------------------------------------------------
9708 
9709  void DeleteString( const String& s, size_type i = 0 )
9710  {
9711  string_base::DeleteString( s, i );
9712  }
9713 
9714  void DeleteStringIC( const String& s, size_type i = 0 )
9715  {
9716  string_base::DeleteStringIC( s, i );
9717  }
9718 
9719  void DeleteString( const_iterator t, size_type i = 0 )
9720  {
9721  string_base::DeleteString( t, i );
9722  }
9723 
9724  void DeleteStringIC( const_iterator t, size_type i = 0 )
9725  {
9726  string_base::DeleteStringIC( t, i );
9727  }
9728 
9729  void DeleteString( const char16_t* t, size_type i = 0 )
9730  {
9731  string_base::DeleteString( reinterpret_cast<const_iterator>( t ), i );
9732  }
9733 
9734  void DeleteStringIC( const char16_t* t, size_type i = 0 )
9735  {
9736  string_base::DeleteStringIC( reinterpret_cast<const_iterator>( t ), i );
9737  }
9738 
9739  void DeleteString( const wchar_t* t, size_type i = 0 )
9740  {
9741 #ifdef __PCL_WINDOWS
9742  string_base::DeleteString( reinterpret_cast<const_iterator>( t ), i );
9743 #else
9744  string_base::DeleteString( String( t ), i );
9745 #endif
9746  }
9747 
9748  void DeleteStringIC( const wchar_t* t, size_type i = 0 )
9749  {
9750 #ifdef __PCL_WINDOWS
9751  string_base::DeleteStringIC( reinterpret_cast<const_iterator>( t ), i );
9752 #else
9753  string_base::DeleteStringIC( String( t ), i );
9754 #endif
9755  }
9756 
9757  void DeleteString( const_c_string8 t, size_type i = 0 )
9758  {
9759  string_base::DeleteString( String( t ), i );
9760  }
9761 
9762  void DeleteStringIC( const_c_string8 t, size_type i = 0 )
9763  {
9764  string_base::DeleteStringIC( String( t ), i );
9765  }
9766 
9767  // -------------------------------------------------------------------------
9768 
9769  bool StartsWith( const String& s ) const noexcept
9770  {
9771  return string_base::StartsWith( s );
9772  }
9773 
9774  bool StartsWith( const_iterator t ) const noexcept
9775  {
9776  return string_base::StartsWith( t );
9777  }
9778 
9779  bool StartsWith( char_type c ) const noexcept
9780  {
9781  return string_base::StartsWith( c );
9782  }
9783 
9784  bool StartsWithIC( const String& s ) const noexcept
9785  {
9786  return string_base::StartsWithIC( s );
9787  }
9788 
9789  bool StartsWithIC( const_iterator t ) const noexcept
9790  {
9791  return string_base::StartsWithIC( t );
9792  }
9793 
9794  bool StartsWithIC( char_type c ) const noexcept
9795  {
9796  return string_base::StartsWithIC( c );
9797  }
9798 
9799  bool StartsWith( const char16_t* t ) const noexcept
9800  {
9801  return string_base::StartsWith( reinterpret_cast<const_iterator>( t ) );
9802  }
9803 
9804  bool StartsWith( char16_t c ) const noexcept
9805  {
9806  return string_base::StartsWith( char_type( c ) );
9807  }
9808 
9809  bool StartsWithIC( const char16_t* t ) const noexcept
9810  {
9811  return string_base::StartsWithIC( reinterpret_cast<const_iterator>( t ) );
9812  }
9813 
9814  bool StartsWithIC( char16_t c ) const noexcept
9815  {
9816  return string_base::StartsWithIC( char_type( c ) );
9817  }
9818 
9819  bool StartsWith( const wchar_t* t ) const noexcept
9820  {
9821 #ifdef __PCL_WINDOWS
9822  return string_base::StartsWith( reinterpret_cast<const_iterator>( t ) );
9823 #else
9824  return string_base::StartsWith( String( t ) );
9825 #endif
9826  }
9827 
9828  bool StartsWith( wchar_t c ) const noexcept
9829  {
9830  return string_base::StartsWith( char_type( c ) );
9831  }
9832 
9833  bool StartsWithIC( const wchar_t* t ) const noexcept
9834  {
9835 #ifdef __PCL_WINDOWS
9836  return string_base::StartsWithIC( reinterpret_cast<const_iterator>( t ) );
9837 #else
9838  return string_base::StartsWithIC( String( t ) );
9839 #endif
9840  }
9841 
9842  bool StartsWithIC( wchar_t c ) const noexcept
9843  {
9844  return string_base::StartsWithIC( char_type( c ) );
9845  }
9846 
9847  bool StartsWith( const_c_string8 t ) const noexcept
9848  {
9849  size_type n = char8_traits::Length( t );
9850  if ( n == 0 || Length() < n )
9851  return false;
9852  for ( const_iterator p = m_data->string, q = p+n; p < q; ++p, ++t )
9853  if ( *p != char_type( *t ) )
9854  return false;
9855  return true;
9856  }
9857 
9858  bool StartsWith( char8_type c ) const noexcept
9859  {
9860  return string_base::StartsWith( char_type( c ) );
9861  }
9862 
9863  bool StartsWithIC( const_c_string8 t ) const noexcept
9864  {
9865  size_type n = char8_traits::Length( t );
9866  if ( n == 0 || Length() < n )
9867  return false;
9868  for ( const_iterator p = m_data->string, q = p+n; p < q; ++p, ++t )
9869  if ( char_traits::ToCaseFolded( *p ) != char_type( char8_traits::ToCaseFolded( *t ) ) )
9870  return false;
9871  return true;
9872  }
9873 
9874  bool StartsWithIC( char8_type c ) const noexcept
9875  {
9876  return string_base::StartsWithIC( char_type( c ) );
9877  }
9878 
9879  // -------------------------------------------------------------------------
9880 
9881  bool EndsWith( const String& s ) const noexcept
9882  {
9883  return string_base::EndsWith( s );
9884  }
9885 
9886  bool EndsWith( const_iterator t ) const noexcept
9887  {
9888  return string_base::EndsWith( t );
9889  }
9890 
9891  bool EndsWith( char_type c ) const noexcept
9892  {
9893  return string_base::EndsWith( c );
9894  }
9895 
9896  bool EndsWithIC( const String& s ) const noexcept
9897  {
9898  return string_base::EndsWithIC( s );
9899  }
9900 
9901  bool EndsWithIC( const_iterator t ) const noexcept
9902  {
9903  return string_base::EndsWithIC( t );
9904  }
9905 
9906  bool EndsWithIC( char_type c ) const noexcept
9907  {
9908  return string_base::EndsWithIC( c );
9909  }
9910 
9911  bool EndsWith( const char16_t* t ) const noexcept
9912  {
9913  return string_base::EndsWith( reinterpret_cast<const_iterator>( t ) );
9914  }
9915 
9916  bool EndsWith( char16_t c ) const noexcept
9917  {
9918  return string_base::EndsWith( char_type( c ) );
9919  }
9920 
9921  bool EndsWithIC( const char16_t* t ) const noexcept
9922  {
9923  return string_base::EndsWithIC( reinterpret_cast<const_iterator>( t ) );
9924  }
9925 
9926  bool EndsWithIC( char16_t c ) const noexcept
9927  {
9928  return string_base::EndsWithIC( char_type( c ) );
9929  }
9930 
9931  bool EndsWith( const wchar_t* t ) const noexcept
9932  {
9933 #ifdef __PCL_WINDOWS
9934  return string_base::EndsWith( reinterpret_cast<const_iterator>( t ) );
9935 #else
9936  return string_base::EndsWith( String( t ) );
9937 #endif
9938  }
9939 
9940  bool EndsWith( wchar_t c ) const noexcept
9941  {
9942  return string_base::EndsWith( char_type( c ) );
9943  }
9944 
9945  bool EndsWithIC( const wchar_t* t ) const noexcept
9946  {
9947 #ifdef __PCL_WINDOWS
9948  return string_base::EndsWithIC( reinterpret_cast<const_iterator>( t ) );
9949 #else
9950  return string_base::EndsWithIC( String( t ) );
9951 #endif
9952  }
9953 
9954  bool EndsWithIC( wchar_t c ) const noexcept
9955  {
9956  return string_base::EndsWithIC( char_type( c ) );
9957  }
9958 
9959  bool EndsWith( const_c_string8 t ) const noexcept
9960  {
9961  size_type n = char8_traits::Length( t );
9962  if ( n == 0 || Length() < n )
9963  return false;
9964  for ( const_iterator p = m_data->end-n; p < m_data->end; ++p, ++t )
9965  if ( *p != char_type( *t ) )
9966  return false;
9967  return true;
9968  }
9969 
9970  bool EndsWith( char8_type c ) const noexcept
9971  {
9972  return string_base::EndsWith( char_type( c ) );
9973  }
9974 
9975  bool EndsWithIC( const_c_string8 t ) const noexcept
9976  {
9977  size_type n = char8_traits::Length( t );
9978  if ( n == 0 || Length() < n )
9979  return false;
9980  for ( const_iterator p = m_data->end-n; p < m_data->end; ++p, ++t )
9981  if ( char_traits::ToCaseFolded( *p ) != char_type( char8_traits::ToCaseFolded( *t ) ) )
9982  return false;
9983  return true;
9984  }
9985 
9986  bool EndsWithIC( char8_type c ) const noexcept
9987  {
9988  return string_base::EndsWithIC( char_type( c ) );
9989  }
9990 
9991  // -------------------------------------------------------------------------
9992 
9993  size_type FindFirst( const String& s, size_type i = 0 ) const noexcept
9994  {
9995  return string_base::FindFirst( s, i );
9996  }
9997 
9998  size_type FindFirst( const_iterator t, size_type i = 0 ) const noexcept
9999  {
10000  return string_base::FindFirst( t, i );
10001  }
10002 
10003  size_type FindFirst( char_type c, size_type i = 0 ) const noexcept
10004  {
10005  return string_base::FindFirst( c, i );
10006  }
10007 
10008  size_type FindFirstIC( const String& s, size_type i = 0 ) const noexcept
10009  {
10010  return string_base::FindFirstIC( s, i );
10011  }
10012 
10013  size_type FindFirstIC( const_iterator t, size_type i = 0 ) const noexcept
10014  {
10015  return string_base::FindFirstIC( t, i );
10016  }
10017 
10018  size_type FindFirstIC( char_type c, size_type i = 0 ) const noexcept
10019  {
10020  return string_base::FindFirstIC( c, i );
10021  }
10022 
10023  size_type FindFirst( const char16_t* t, size_type i = 0 ) const noexcept
10024  {
10025  return string_base::FindFirst( reinterpret_cast<const_iterator>( t ), i );
10026  }
10027 
10028  size_type FindFirst( char16_t c, size_type i = 0 ) const noexcept
10029  {
10030  return string_base::FindFirst( char_type( c ), i );
10031  }
10032 
10033  size_type FindFirstIC( const char16_t* t, size_type i = 0 ) const noexcept
10034  {
10035  return string_base::FindFirstIC( reinterpret_cast<const_iterator>( t ), i );
10036  }
10037 
10038  size_type FindFirstIC( char16_t c, size_type i = 0 ) const noexcept
10039  {
10040  return string_base::FindFirstIC( char_type( c ), i );
10041  }
10042 
10043  size_type FindFirst( const wchar_t* t, size_type i = 0 ) const noexcept
10044  {
10045 #ifdef __PCL_WINDOWS
10046  return string_base::FindFirst( reinterpret_cast<const_iterator>( t ), i );
10047 #else
10048  return string_base::FindFirst( String( t ), i );
10049 #endif
10050  }
10051 
10052  size_type FindFirst( wchar_t c, size_type i = 0 ) const noexcept
10053  {
10054  return string_base::FindFirst( char_type( c ), i );
10055  }
10056 
10057  size_type FindFirstIC( const wchar_t* t, size_type i = 0 ) const noexcept
10058  {
10059 #ifdef __PCL_WINDOWS
10060  return string_base::FindFirstIC( reinterpret_cast<const_iterator>( t ), i );
10061 #else
10062  return string_base::FindFirstIC( String( t ), i );
10063 #endif
10064  }
10065 
10066  size_type FindFirstIC( wchar_t c, size_type i = 0 ) const noexcept
10067  {
10068  return string_base::FindFirstIC( char_type( c ), i );
10069  }
10070 
10071  size_type FindFirst( const_c_string8 t, size_type i = 0 ) const noexcept
10072  {
10073  return string_base::FindFirst( String( t ), i );
10074  }
10075 
10076  size_type FindFirst( char8_type c, size_type i = 0 ) const noexcept
10077  {
10078  return string_base::FindFirst( char_type( c ), i );
10079  }
10080 
10081  size_type FindFirstIC( const_c_string8 t, size_type i = 0 ) const noexcept
10082  {
10083  return string_base::FindFirstIC( String( t ), i );
10084  }
10085 
10086  size_type FindFirstIC( char8_type c, size_type i = 0 ) const noexcept
10087  {
10088  return string_base::FindFirstIC( char_type( c ), i );
10089  }
10090 
10091  //
10092 
10093  size_type Find( const String& s, size_type i = 0 ) const noexcept
10094  {
10095  return FindFirst( s, i );
10096  }
10097 
10098  size_type Find( const_iterator t, size_type i = 0 ) const noexcept
10099  {
10100  return FindFirst( t, i );
10101  }
10102 
10103  size_type Find( char_type c, size_type i = 0 ) const noexcept
10104  {
10105  return FindFirst( c, i );
10106  }
10107 
10108  size_type Find( const char16_t* t, size_type i = 0 ) const noexcept
10109  {
10110  return FindFirst( t, i );
10111  }
10112 
10113  size_type Find( char16_t c, size_type i = 0 ) const noexcept
10114  {
10115  return FindFirst( c, i );
10116  }
10117 
10118  size_type Find( const wchar_t* t, size_type i = 0 ) const noexcept
10119  {
10120  return FindFirst( t, i );
10121  }
10122 
10123  size_type Find( wchar_t c, size_type i = 0 ) const noexcept
10124  {
10125  return FindFirst( c, i );
10126  }
10127 
10128  size_type Find( const_c_string8 t, size_type i = 0 ) const noexcept
10129  {
10130  return FindFirst( t, i );
10131  }
10132 
10133  size_type Find( char8_type c, size_type i = 0 ) const noexcept
10134  {
10135  return FindFirst( c, i );
10136  }
10137 
10138  size_type FindIC( const String& s, size_type i = 0 ) const noexcept
10139  {
10140  return FindFirstIC( s, i );
10141  }
10142 
10143  size_type FindIC( const_iterator t, size_type i = 0 ) const noexcept
10144  {
10145  return FindFirstIC( t, i );
10146  }
10147 
10148  size_type FindIC( char_type c, size_type i = 0 ) const noexcept
10149  {
10150  return FindFirstIC( c, i );
10151  }
10152 
10153  size_type FindIC( const char16_t* t, size_type i = 0 ) const noexcept
10154  {
10155  return FindFirstIC( t, i );
10156  }
10157 
10158  size_type FindIC( char16_t c, size_type i = 0 ) const noexcept
10159  {
10160  return FindFirstIC( c, i );
10161  }
10162 
10163  size_type FindIC( const wchar_t* t, size_type i = 0 ) const noexcept
10164  {
10165  return FindFirstIC( t, i );
10166  }
10167 
10168  size_type FindIC( wchar_t c, size_type i = 0 ) const noexcept
10169  {
10170  return FindFirstIC( c, i );
10171  }
10172 
10173  size_type FindIC( const_c_string8 t, size_type i = 0 ) const noexcept
10174  {
10175  return FindFirstIC( t, i );
10176  }
10177 
10178  size_type FindIC( char8_type c, size_type i = 0 ) const noexcept
10179  {
10180  return FindFirstIC( c, i );
10181  }
10182 
10183  // -------------------------------------------------------------------------
10184 
10185  size_type FindLast( const String& s, size_type r = maxPos ) const noexcept
10186  {
10187  return string_base::FindLast( s, r );
10188  }
10189 
10190  size_type FindLast( const_iterator t, size_type r = maxPos ) const noexcept
10191  {
10192  return string_base::FindLast( t, r );
10193  }
10194 
10195  size_type FindLast( char_type c, size_type r = maxPos ) const noexcept
10196  {
10197  return string_base::FindLast( c, r );
10198  }
10199 
10200  size_type FindLastIC( const String& s, size_type r = maxPos ) const noexcept
10201  {
10202  return string_base::FindLastIC( s, r );
10203  }
10204 
10205  size_type FindLastIC( const_iterator t, size_type r = maxPos ) const noexcept
10206  {
10207  return string_base::FindLastIC( t, r );
10208  }
10209 
10210  size_type FindLastIC( char_type c, size_type r = maxPos ) const noexcept
10211  {
10212  return string_base::FindLastIC( c, r );
10213  }
10214 
10215  size_type FindLast( const char16_t* t, size_type r = maxPos ) const noexcept
10216  {
10217  return string_base::FindLast( reinterpret_cast<const_iterator>( t ), r );
10218  }
10219 
10220  size_type FindLast( char16_t c, size_type r = maxPos ) const noexcept
10221  {
10222  return string_base::FindLast( char_type( c ), r );
10223  }
10224 
10225  size_type FindLastIC( const char16_t* t, size_type r = maxPos ) const noexcept
10226  {
10227  return string_base::FindLastIC( reinterpret_cast<const_iterator>( t ), r );
10228  }
10229 
10230  size_type FindLastIC( char16_t c, size_type r = maxPos ) const noexcept
10231  {
10232  return string_base::FindLastIC( char_type( c ), r );
10233  }
10234 
10235  size_type FindLast( const wchar_t* t, size_type r = maxPos ) const noexcept
10236  {
10237 #ifdef __PCL_WINDOWS
10238  return string_base::FindLast( reinterpret_cast<const_iterator>( t ), r );
10239 #else
10240  return string_base::FindLast( String( t ), r );
10241 #endif
10242  }
10243 
10244  size_type FindLast( wchar_t c, size_type r = maxPos ) const noexcept
10245  {
10246  return string_base::FindLast( char_type( c ), r );
10247  }
10248 
10249  size_type FindLastIC( const wchar_t* t, size_type r = maxPos ) const noexcept
10250  {
10251 #ifdef __PCL_WINDOWS
10252  return string_base::FindLastIC( reinterpret_cast<const_iterator>( t ), r );
10253 #else
10254  return string_base::FindLastIC( String( t ), r );
10255 #endif
10256  }
10257 
10258  size_type FindLastIC( wchar_t c, size_type r = maxPos ) const noexcept
10259  {
10260  return string_base::FindLastIC( char_type( c ), r );
10261  }
10262 
10263  size_type FindLast( const_c_string8 t, size_type r = maxPos ) const noexcept
10264  {
10265  return string_base::FindLast( String( t ), r );
10266  }
10267 
10268  size_type FindLast( char8_type c, size_type r = maxPos ) const noexcept
10269  {
10270  return string_base::FindLast( char_type( c ), r );
10271  }
10272 
10273  size_type FindLastIC( const_c_string8 t, size_type r = maxPos ) const noexcept
10274  {
10275  return string_base::FindLastIC( String( t ), r );
10276  }
10277 
10278  size_type FindLastIC( char8_type c, size_type r = maxPos ) const noexcept
10279  {
10280  return string_base::FindLastIC( char_type( c ), r );
10281  }
10282 
10283  // -------------------------------------------------------------------------
10284 
10285  bool Contains( const String& s ) const noexcept
10286  {
10287  return string_base::Contains( s );
10288  }
10289 
10290  bool Contains( const_iterator t ) const noexcept
10291  {
10292  return string_base::Contains( t );
10293  }
10294 
10295  bool Contains( char_type c ) const noexcept
10296  {
10297  return string_base::Contains( c );
10298  }
10299 
10300  bool ContainsIC( const String& s ) const noexcept
10301  {
10302  return string_base::ContainsIC( s );
10303  }
10304 
10305  bool ContainsIC( const_iterator t ) const noexcept
10306  {
10307  return string_base::ContainsIC( t );
10308  }
10309 
10310  bool ContainsIC( char_type c ) const noexcept
10311  {
10312  return string_base::ContainsIC( c );
10313  }
10314 
10315  bool Contains( const char16_t* t ) const noexcept
10316  {
10317  return string_base::Contains( reinterpret_cast<const_iterator>( t ) );
10318  }
10319 
10320  bool Contains( char16_t c ) const noexcept
10321  {
10322  return string_base::Contains( char_type( c ) );
10323  }
10324 
10325  bool ContainsIC( const char16_t* t ) const noexcept
10326  {
10327  return string_base::ContainsIC( reinterpret_cast<const_iterator>( t ) );
10328  }
10329 
10330  bool ContainsIC( char16_t c ) const noexcept
10331  {
10332  return string_base::ContainsIC( char_type( c ) );
10333  }
10334 
10335  bool Contains( const wchar_t* t ) const noexcept
10336  {
10337 #ifdef __PCL_WINDOWS
10338  return string_base::Contains( reinterpret_cast<const_iterator>( t ) );
10339 #else
10340  return string_base::Contains( String( t ) );
10341 #endif
10342  }
10343 
10344  bool Contains( wchar_t c ) const noexcept
10345  {
10346  return string_base::Contains( char_type( c ) );
10347  }
10348 
10349  bool ContainsIC( const wchar_t* t ) const noexcept
10350  {
10351 #ifdef __PCL_WINDOWS
10352  return string_base::ContainsIC( reinterpret_cast<const_iterator>( t ) );
10353 #else
10354  return string_base::ContainsIC( String( t ) );
10355 #endif
10356  }
10357 
10358  bool ContainsIC( wchar_t c ) const noexcept
10359  {
10360  return string_base::ContainsIC( char_type( c ) );
10361  }
10362 
10363  bool Contains( const_c_string8 t ) const noexcept
10364  {
10365  return string_base::Contains( String( t ) );
10366  }
10367 
10368  bool Contains( char8_type c ) const noexcept
10369  {
10370  return string_base::Contains( char_type( c ) );
10371  }
10372 
10373  bool ContainsIC( const_c_string8 t ) const noexcept
10374  {
10375  return string_base::ContainsIC( String( t ) );
10376  }
10377 
10378  bool ContainsIC( char8_type c ) const noexcept
10379  {
10380  return string_base::ContainsIC( char_type( c ) );
10381  }
10382 
10383  // -------------------------------------------------------------------------
10384 
10385  int CompareCodePoints( const String& s, bool caseSensitive = true ) const noexcept
10386  {
10387  return string_base::CompareCodePoints( s, caseSensitive );
10388  }
10389 
10390  int CompareCodePoints( const_iterator t, bool caseSensitive = true ) const noexcept
10391  {
10392  return string_base::CompareCodePoints( t, caseSensitive );
10393  }
10394 
10395  int CompareCodePoints( char_type c, bool caseSensitive = true ) const noexcept
10396  {
10397  return string_base::CompareCodePoints( c, caseSensitive );
10398  }
10399 
10400  int CompareCodePoints( const char16_t* t, bool caseSensitive = true ) const noexcept
10401  {
10402  return string_base::CompareCodePoints( reinterpret_cast<const_iterator>( t ), caseSensitive );
10403  }
10404 
10405  int CompareCodePoints( char16_t c, bool caseSensitive = true ) const noexcept
10406  {
10407  return string_base::CompareCodePoints( char_type( c ), caseSensitive );
10408  }
10409 
10410  int CompareCodePoints( const wchar_t* t, bool caseSensitive = true ) const noexcept
10411  {
10412 #ifdef __PCL_WINDOWS
10413  return string_base::CompareCodePoints( reinterpret_cast<const_iterator>( t ), caseSensitive );
10414 #else
10415  return string_base::CompareCodePoints( String( t ), caseSensitive );
10416 #endif
10417  }
10418 
10419  int CompareCodePoints( wchar_t c, bool caseSensitive = true ) const noexcept
10420  {
10421  return string_base::CompareCodePoints( char_type( c ), caseSensitive );
10422  }
10423 
10424  int CompareCodePoints( const_c_string8 t, bool caseSensitive = true ) const noexcept
10425  {
10426  return string_base::CompareCodePoints( String( t ), caseSensitive );
10427  }
10428 
10429  int CompareCodePoints( char8_type c, bool caseSensitive = true ) const noexcept
10430  {
10431  return string_base::CompareCodePoints( char_type( c ), caseSensitive );
10432  }
10433 
10434  // -------------------------------------------------------------------------
10435 
10436  int Compare( const String& s, bool caseSensitive = true, bool localeAware = true ) const noexcept
10437  {
10438  return string_base::Compare( s, caseSensitive, localeAware );
10439  }
10440 
10441  int Compare( const_iterator t, bool caseSensitive = true, bool localeAware = true ) const noexcept
10442  {
10443  return string_base::Compare( t, caseSensitive, localeAware );
10444  }
10445 
10446  int Compare( char_type c, bool caseSensitive = true, bool localeAware = true ) const noexcept
10447  {
10448  return string_base::Compare( c, caseSensitive, localeAware );
10449  }
10450 
10451  int CompareIC( const String& s, bool localeAware = true ) const noexcept
10452  {
10453  return string_base::CompareIC( s, localeAware );
10454  }
10455 
10456  int CompareIC( const_iterator t, bool localeAware = true ) const noexcept
10457  {
10458  return string_base::CompareIC( t, localeAware );
10459  }
10460 
10461  int CompareIC( char_type c, bool localeAware = true ) const noexcept
10462  {
10463  return string_base::CompareIC( c, localeAware );
10464  }
10465 
10466  int Compare( const char16_t* t, bool caseSensitive = true, bool localeAware = true ) const noexcept
10467  {
10468  return string_base::Compare( reinterpret_cast<const_iterator>( t ), caseSensitive, localeAware );
10469  }
10470 
10471  int Compare( char16_t c, bool caseSensitive = true, bool localeAware = true ) const noexcept
10472  {
10473  return string_base::Compare( char_type( c ), caseSensitive, localeAware );
10474  }
10475 
10476  int CompareIC( const char16_t* t, bool localeAware = true ) const noexcept
10477  {
10478  return string_base::CompareIC( reinterpret_cast<const_iterator>( t ), localeAware );
10479  }
10480 
10481  int CompareIC( char16_t c, bool localeAware = true ) const noexcept
10482  {
10483  return string_base::CompareIC( char_type( c ), localeAware );
10484  }
10485 
10486  int Compare( const wchar_t* t, bool caseSensitive = true, bool localeAware = true ) const noexcept
10487  {
10488 #ifdef __PCL_WINDOWS
10489  return string_base::Compare( reinterpret_cast<const_iterator>( t ), caseSensitive, localeAware );
10490 #else
10491  return string_base::Compare( String( t ), caseSensitive, localeAware );
10492 #endif
10493  }
10494 
10495  int Compare( wchar_t c, bool caseSensitive = true, bool localeAware = true ) const noexcept
10496  {
10497  return string_base::Compare( char_type( c ), caseSensitive, localeAware );
10498  }
10499 
10500  int CompareIC( const wchar_t* t, bool localeAware = true ) const noexcept
10501  {
10502 #ifdef __PCL_WINDOWS
10503  return string_base::CompareIC( reinterpret_cast<const_iterator>( t ), localeAware );
10504 #else
10505  return string_base::CompareIC( String( t ), localeAware );
10506 #endif
10507  }
10508 
10509  int CompareIC( wchar_t c, bool localeAware = true ) const noexcept
10510  {
10511  return string_base::CompareIC( char_type( c ), localeAware );
10512  }
10513 
10514  int Compare( const_c_string8 t, bool caseSensitive = true, bool localeAware = true ) const noexcept
10515  {
10516  return string_base::Compare( String( t ), caseSensitive, localeAware );
10517  }
10518 
10519  int Compare( char8_type c, bool caseSensitive = true, bool localeAware = true ) const noexcept
10520  {
10521  return string_base::Compare( char_type( c ), caseSensitive, localeAware );
10522  }
10523 
10524  int CompareIC( const_c_string8 t, bool localeAware = true ) const noexcept
10525  {
10526  return string_base::CompareIC( String( t ), localeAware );
10527  }
10528 
10529  int CompareIC( char8_type c, bool localeAware = true ) const noexcept
10530  {
10531  return string_base::CompareIC( char_type( c ), localeAware );
10532  }
10533 
10534  // -------------------------------------------------------------------------
10535 
10536  bool WildMatch( const String& pattern, bool caseSensitive = true ) const noexcept
10537  {
10538  return string_base::WildMatch( pattern, caseSensitive );
10539  }
10540 
10541  bool WildMatchIC( const String& pattern ) const noexcept
10542  {
10543  return string_base::WildMatchIC( pattern );
10544  }
10545 
10546  bool WildMatch( const_iterator pattern, bool caseSensitive = true ) const noexcept
10547  {
10548  return string_base::WildMatch( pattern, caseSensitive );
10549  }
10550 
10551  bool WildMatchIC( const_iterator pattern ) const noexcept
10552  {
10553  return string_base::WildMatchIC( pattern );
10554  }
10555 
10556  bool WildMatch( const string8_base& pattern, bool caseSensitive = true ) const noexcept
10557  {
10558  return char_traits::WildMatch( m_data->string, Length(), pattern.Begin(), pattern.Length(), caseSensitive );
10559  }
10560 
10561  bool WildMatchIC( const string8_base& pattern ) const noexcept
10562  {
10563  return char_traits::WildMatch( m_data->string, Length(), pattern.Begin(), pattern.Length(), false/*caseSensitive*/ );
10564  }
10565 
10566  bool WildMatch( const_c_string8 pattern, bool caseSensitive = true ) const noexcept
10567  {
10568  return char_traits::WildMatch( m_data->string, Length(), pattern, char8_traits::Length( pattern ), caseSensitive );
10569  }
10570 
10571  bool WildMatchIC( const_c_string8 pattern ) const noexcept
10572  {
10573  return char_traits::WildMatch( m_data->string, Length(), pattern, char8_traits::Length( pattern ), false/*caseSensitive*/ );
10574  }
10575 
10576  // -------------------------------------------------------------------------
10577 
10578  String SetToLength( size_type n ) const
10579  {
10580  return string_base::SetToLength( n );
10581  }
10582 
10583  String ResizedToNullTerminated() const
10584  {
10585  return string_base::ResizedToNullTerminated();
10586  }
10587 
10588  String Squeezed() const
10589  {
10590  return string_base::Squeezed();
10591  }
10592 
10593  // -------------------------------------------------------------------------
10594 
10595  String Substring( size_type i, size_type n = maxPos ) const
10596  {
10597  return string_base::Substring( i, n );
10598  }
10599 
10600  String Left( size_type n ) const
10601  {
10602  return string_base::Left( n );
10603  }
10604 
10605  String Right( size_type n ) const
10606  {
10607  return string_base::Right( n );
10608  }
10609 
10610  String Suffix( size_type i ) const
10611  {
10612  return string_base::Suffix( i );
10613  }
10614 
10615  String Prefix( size_type i ) const
10616  {
10617  return string_base::Prefix( i );
10618  }
10619 
10620  // -------------------------------------------------------------------------
10621 
10622  template <class C>
10623  size_type Break( C& list, const String& s, bool trim = false, size_type i = 0 ) const
10624  {
10625  return string_base::Break( list, s, trim, i );
10626  }
10627 
10628  template <class C>
10629  size_type Break( C& list, const string8_base& s, bool trim = false, size_type i = 0 ) const
10630  {
10631  return string_base::Break( list, String( s ), trim, i );
10632  }
10633 
10634  template <class C>
10635  size_type Break( C& list, const_c_string8 s, bool trim = false, size_type i = 0 ) const
10636  {
10637  return string_base::Break( list, String( s ), trim, i );
10638  }
10639 
10640  template <class C>
10641  size_type Break( C& list, char_type c, bool trim = false, size_type i = 0 ) const
10642  {
10643  return string_base::Break( list, c, trim, i );
10644  }
10645 
10646  template <class C>
10647  size_type Break( C& list, char8_type c, bool trim = false, size_type i = 0 ) const
10648  {
10649  return string_base::Break( list, char_type( c ), trim, i );
10650  }
10651 
10652  template <class C, typename S>
10653  size_type Break( C& list, const Array<S>& ca, bool trim = false, size_type i = 0 ) const
10654  {
10655  return string_base::Break( list, ca, trim, i );
10656  }
10657 
10658  // -------------------------------------------------------------------------
10659 
10660  template <class C>
10661  size_type BreakIC( C& list, const String& s, bool trim = false, size_type i = 0 ) const
10662  {
10663  return string_base::BreakIC( list, s, trim, i );
10664  }
10665 
10666  template <class C>
10667  size_type BreakIC( C& list, const string8_base& s, bool trim = false, size_type i = 0 ) const
10668  {
10669  return string_base::BreakIC( list, String( s ), trim, i );
10670  }
10671 
10672  template <class C>
10673  size_type BreakIC( C& list, const_c_string8 s, bool trim = false, size_type i = 0 ) const
10674  {
10675  return string_base::BreakIC( list, String( s ), trim, i );
10676  }
10677 
10678  template <class C>
10679  size_type BreakIC( C& list, char_type c, bool trim = false, size_type i = 0 ) const
10680  {
10681  return string_base::BreakIC( list, c, trim, i );
10682  }
10683 
10684  template <class C>
10685  size_type BreakIC( C& list, char8_type c, bool trim = false, size_type i = 0 ) const
10686  {
10687  return string_base::BreakIC( list, char_type( c ), trim, i );
10688  }
10689 
10690  // -------------------------------------------------------------------------
10691 
10692  String Trimmed() const
10693  {
10694  return string_base::Trimmed();
10695  }
10696 
10697  String TrimmedLeft() const
10698  {
10699  return string_base::TrimmedLeft();
10700  }
10701 
10702  String TrimmedRight() const
10703  {
10704  return string_base::TrimmedRight();
10705  }
10706 
10707  // -------------------------------------------------------------------------
10708 
10709  String LeftJustified( size_type width, char_type fill = CharTraits::Blank() ) const
10710  {
10711  return string_base::LeftJustified( width, fill );
10712  }
10713 
10714  String RightJustified( size_type width, char_type fill = CharTraits::Blank() ) const
10715  {
10716  return string_base::RightJustified( width, fill );
10717  }
10718 
10719  String CenterJustified( size_type width, char_type fill = CharTraits::Blank() ) const
10720  {
10721  return string_base::CenterJustified( width, fill );
10722  }
10723 
10724  // -------------------------------------------------------------------------
10725 
10726  String Enclosed( char_type c ) const
10727  {
10728  return string_base::Enclosed( c );
10729  }
10730 
10731  String SingleQuoted() const
10732  {
10733  return string_base::SingleQuoted();
10734  }
10735 
10736  String DoubleQuoted() const
10737  {
10738  return string_base::DoubleQuoted();
10739  }
10740 
10741  String Unquoted() const
10742  {
10743  return string_base::Unquoted();
10744  }
10745 
10746  // -------------------------------------------------------------------------
10747 
10748  String CaseFolded() const
10749  {
10750  return string_base::CaseFolded();
10751  }
10752 
10753  String Lowercase() const
10754  {
10755  return string_base::Lowercase();
10756  }
10757 
10758  String Uppercase() const
10759  {
10760  return string_base::Uppercase();
10761  }
10762 
10763  // -------------------------------------------------------------------------
10764 
10765  String Reversed() const
10766  {
10767  return string_base::Reversed();
10768  }
10769 
10770  String Sorted() const
10771  {
10772  return string_base::Sorted();
10773  }
10774 
10775  template <class BP>
10776  String Sorted( BP p ) const
10777  {
10778  return string_base::Sorted( p );
10779  }
10780 
10781  // -------------------------------------------------------------------------
10782 
10792  template <class C>
10793  String& ToSeparated( const C& c, char_type separator )
10794  {
10795  Clear();
10796  return c.ToSeparated( *this, separator );
10797  }
10798 
10815  template <class C, class AF>
10816  String& ToSeparated( const C& c, char_type separator, AF append )
10817  {
10818  Clear();
10819  return c.ToSeparated( *this, separator, append );
10820  }
10821 
10831  template <class C>
10832  String& ToSeparated( const C& c, const String& separator )
10833  {
10834  Clear();
10835  return c.ToSeparated( *this, separator );
10836  }
10837 
10854  template <class C, class AF>
10855  String& ToSeparated( const C& c, const String& separator, AF append )
10856  {
10857  Clear();
10858  return c.ToSeparated( *this, separator, append );
10859  }
10860 
10870  template <class C>
10871  String& ToSeparated( const C& c, const_c_string separator )
10872  {
10873  return ToSeparated( c, String( separator ) );
10874  }
10875 
10892  template <class C, class AF>
10893  String& ToSeparated( const C& c, const_c_string separator, AF append )
10894  {
10895  return ToSeparated( c, String( separator ), append );
10896  }
10897 
10908  template <class C>
10909  String& ToSeparated( const C& c, const_c_string8 separator )
10910  {
10911  return ToSeparated( c, String( separator ) );
10912  }
10913 
10930  template <class C, class AF>
10931  String& ToSeparated( const C& c, const_c_string8 separator, AF append )
10932  {
10933  return ToSeparated( c, String( separator ), append );
10934  }
10935 
10945  template <class C>
10946  String& ToCommaSeparated( const C& c )
10947  {
10948  return ToSeparated( c, CharTraits::Comma() );
10949  }
10950 
10960  template <class C>
10961  String& ToColonSeparated( const C& c )
10962  {
10963  return ToSeparated( c, CharTraits::Colon() );
10964  }
10965 
10975  template <class C>
10976  String& ToSpaceSeparated( const C& c )
10977  {
10978  return ToSeparated( c, CharTraits::Blank() );
10979  }
10980 
10990  template <class C>
10991  String& ToTabSeparated( const C& c )
10992  {
10993  return ToSeparated( c, CharTraits::Tab() );
10994  }
10995 
11005  template <class C>
11007  {
11008  return ToSeparated( c, CharTraits::LF() );
11009  }
11010 
11020  template <class C>
11021  String& ToNullSeparated( const C& c )
11022  {
11023  return ToSeparated( c, CharTraits::Null() );
11024  }
11025 
11034  template <class C>
11035  String& ToHyphenated( const C& c )
11036  {
11037  return ToSeparated( c, CharTraits::Hyphen() );
11038  }
11039 
11040  // -------------------------------------------------------------------------
11041 
11057 
11065  {
11066  return String( *this ).ToEncodedHTMLSpecialChars();
11067  }
11068 
11086 
11095  {
11096  return String( *this ).ToDecodedHTMLSpecialChars();
11097  }
11098 
11099  // -------------------------------------------------------------------------
11100 
11101 #ifdef __PCL_QT_INTERFACE
11102 
11103  operator QString() const
11104  {
11105  return PCL_GET_QSTRING_FROM_CHAR16PTR( c_str() );
11106  }
11107 
11108  operator QDate() const
11109  {
11110  return QDate::fromString( operator QString(), PCL_QDATE_FMT_STR );
11111  }
11112 
11113  operator QDateTime() const
11114  {
11115  return QDateTime::fromString( operator QString(), PCL_QDATETIME_FMT_STR );
11116  }
11117 
11118 #endif
11119 
11132  {
11133  va_list paramList;
11134  va_start( paramList, fmt );
11135 
11136  (void)VFormat( fmt, paramList );
11137 
11138  va_end( paramList );
11139  return *this;
11140  }
11141 
11154  {
11155  va_list paramList;
11156  va_start( paramList, fmt );
11157 
11158  (void)AppendVFormat( fmt, paramList );
11159 
11160  va_end( paramList );
11161  return *this;
11162  }
11163 
11176  int VFormat( const_c_string8 fmt, va_list paramList )
11177  {
11178  IsoString s;
11179  int count = s.VFormat( fmt, paramList );
11180  Assign( s );
11181  return count;
11182  }
11183 
11196  int AppendVFormat( const_c_string8 fmt, va_list paramList )
11197  {
11198  IsoString s;
11199  int count = s.VFormat( fmt, paramList );
11200  Append( s );
11201  return count;
11202  }
11203 
11216  String& Format( const wchar_t* fmt, ... )
11217  {
11218  va_list paramList;
11219  va_start( paramList, fmt );
11220 
11221  (void)VFormat( fmt, paramList );
11222 
11223  va_end( paramList );
11224  return *this;
11225  }
11226 
11239  String& AppendFormat( const wchar_t* fmt, ... )
11240  {
11241  va_list paramList;
11242  va_start( paramList, fmt );
11243 
11244  (void)AppendVFormat( fmt, paramList );
11245 
11246  va_end( paramList );
11247  return *this;
11248  }
11249 
11263  int VFormat( const wchar_t* fmt, va_list paramList );
11264 
11278  int AppendVFormat( const wchar_t* fmt, va_list paramList );
11279 
11280  // -------------------------------------------------------------------------
11281 
11291  static String UTF8ToUTF16( const_c_string8 string, size_type i = 0, size_type n = maxPos );
11292 
11302  static IsoString UTF16ToUTF8( const_c_string string, size_type i = 0, size_type n = maxPos );
11303 
11313  static Array<uint32> UTF16ToUTF32( const_c_string string, size_type i = 0, size_type n = maxPos );
11314 
11324  static String UTF32ToUTF16( const uint32* string, size_type i = 0, size_type n = maxPos );
11325 
11326  // -------------------------------------------------------------------------
11327 
11338 
11349 
11357  IsoString ToUTF8( size_type i = 0, size_type n = maxPos ) const
11358  {
11359  return UTF16ToUTF8( Begin(), i, n );
11360  }
11361 
11377  IsoString ToMBS() const;
11378 
11388  {
11389 #ifdef __PCL_WINDOWS
11390  return ToMBS();
11391 #else
11392  return ToUTF8();
11393 #endif
11394  }
11395 
11416  Array<wchar_t> ToWCharArray( size_type i = 0, size_type n = maxPos ) const
11417  {
11418  if ( n > 0 )
11419  {
11420  size_type len = Length();
11421  if ( i < len )
11422  {
11423  n = pcl::Min( n, len-i );
11424  Array<wchar_t> a( n+1, wchar_t( 0 ) );
11425 #ifdef __PCL_WINDOWS
11426  char_traits::Copy( reinterpret_cast<iterator>( a.Begin() ), m_data->string+i, n );
11427 #else
11428  Array<wchar_t>::iterator w = a.Begin();
11429  for ( const_iterator s = m_data->string+i, e = s+n; s < e; ++w, ++s )
11430  *w = wchar_t( *s );
11431 #endif // __PCL_WINDOWS
11432  return a;
11433  }
11434  }
11435 
11436  return Array<wchar_t>( size_type( 1 ), wchar_t( 0 ) );
11437  }
11438 
11447  Array<uint32> ToUTF32( size_type i = 0, size_type n = maxPos ) const
11448  {
11449  return UTF16ToUTF32( Begin(), i, n );
11450  }
11451 
11452 #ifdef __PCL_QT_INTERFACE
11453 
11454  QString ToQString() const
11455  {
11456  return operator QString();
11457  }
11458 
11459  QDate ToQDate() const
11460  {
11461  return operator QDate();
11462  }
11463 
11464  QDateTime ToQDateTime() const
11465  {
11466  return operator QDateTime();
11467  }
11468 
11469 #endif
11470 
11481  bool ToBool() const;
11482 
11497  bool TryToBool( bool& value ) const noexcept;
11498 
11512  float ToFloat() const;
11513 
11528  bool TryToFloat( float& value ) const noexcept;
11529 
11553  double ToDouble() const;
11554 
11569  bool TryToDouble( double& value ) const noexcept;
11570 
11587  long ToInt() const
11588  {
11589  return ToInt( 0 );
11590  }
11591 
11615  bool TryToInt( int& value ) const noexcept
11616  {
11617  return TryToInt( value, 0 );
11618  }
11619 
11646  long ToInt( int base ) const;
11647 
11665  bool TryToInt( int& value, int base ) const noexcept;
11666 
11684  unsigned long ToUInt() const
11685  {
11686  return ToUInt( 0 );
11687  }
11688 
11712  bool TryToUInt( unsigned& value ) const noexcept
11713  {
11714  return TryToUInt( value, 0 );
11715  }
11716 
11733  unsigned long ToUInt( int base ) const;
11734 
11752  bool TryToUInt( unsigned& value, int base ) const noexcept;
11753 
11768  long long ToInt64() const
11769  {
11770  return ToInt64( 0 );
11771  }
11772 
11797  bool TryToInt64( long long& value ) const noexcept
11798  {
11799  return TryToInt64( value, 0 );
11800  }
11801 
11816  long long ToInt64( int base ) const;
11817 
11835  bool TryToInt64( long long& value, int base ) const noexcept;
11836 
11851  unsigned long long ToUInt64() const
11852  {
11853  return ToUInt64( 0 );
11854  }
11855 
11880  bool TryToUInt64( unsigned long long& value ) const noexcept
11881  {
11882  return TryToUInt64( value, 0 );
11883  }
11884 
11899  unsigned long long ToUInt64( int base ) const;
11900 
11918  bool TryToUInt64( unsigned long long& value, int base ) const noexcept;
11919 
11951  Array<float> ParseListOfFloat( char separator = ',', size_type maxCount = ~size_type( 0 ) ) const;
11952 
11984  Array<double> ParseListOfDouble( char separator = ',', size_type maxCount = ~size_type( 0 ) ) const;
11985 
11986 #ifndef __PCL_NO_STRING_VECTOR
11987 
11995  GenericVector<float> ParseListOfFloatAsVector( char separator = ',', int maxCount = int_max ) const;
11996 
12004  GenericVector<double> ParseListOfDoubleAsVector( char separator = ',', int maxCount = int_max ) const;
12005 
12006 #endif // !__PCL_NO_STRING_VECTOR
12007 
12035  double SexagesimalToDouble( const String& separator = ':' ) const
12036  {
12037  int sign, s1, s2; double s3;
12038  ParseSexagesimal( sign, s1, s2, s3, separator );
12039  return sign*(s1 + (s2 + s3/60)/60);
12040  }
12041 
12055  double SexagesimalToDouble( const Array<char_type>& separators ) const
12056  {
12057  int sign, s1, s2; double s3;
12058  ParseSexagesimal( sign, s1, s2, s3, separators );
12059  return sign*(s1 + (s2 + s3/60)/60);
12060  }
12061 
12079  bool TrySexagesimalToDouble( double& value, const String& separator = ':' ) const noexcept
12080  {
12081  int sign, s1, s2; double s3;
12082  if ( TryParseSexagesimal( sign, s1, s2, s3, separator ) )
12083  {
12084  value = sign*(s1 + (s2 + s3/60)/60);
12085  return true;
12086  }
12087  return false;
12088  }
12089 
12102  bool TrySexagesimalToDouble( double& value, const Array<char_type>& separators ) const noexcept
12103  {
12104  int sign, s1, s2; double s3;
12105  if ( TryParseSexagesimal( sign, s1, s2, s3, separators ) )
12106  {
12107  value = sign*(s1 + (s2 + s3/60)/60);
12108  return true;
12109  }
12110  return false;
12111  }
12112 
12139  void ParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const String& separator = ':' ) const;
12140 
12155  void ParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const Array<char_type>& separators ) const;
12156 
12171  bool TryParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const String& separator = ':' ) const noexcept;
12172 
12186  bool TryParseSexagesimal( int& sign, int& s1, int& s2, double& s3, const Array<char_type>& separators ) const noexcept;
12187 
12212  static String ToSexagesimal( int sign, double s1, double s2, double s3,
12214 
12225  static String ToSexagesimal( double d, const SexagesimalConversionOptions& options = SexagesimalConversionOptions() )
12226  {
12227  return ToSexagesimal( (d < 0) ? -1 : +1, Abs( d ), 0, 0, options );
12228  }
12229 
12253  void ParseISO8601DateTime( int& year, int& month, int& day, double& dayf, double& tz ) const;
12254 
12268  bool TryParseISO8601DateTime( int& year, int& month, int& day, double& dayf, double& tz ) const noexcept;
12269 
12293  static String ToISO8601DateTime( int year, int month, int day, double dayf, double tz = 0,
12295 
12307 
12319 
12326  static String Random( size_type n, RandomizationOptions options = RandomizationOption::Default );
12327 
12339  static String UUID();
12340 };
12341 
12342 // ----------------------------------------------------------------------------
12343 // ----------------------------------------------------------------------------
12344 
12346 {
12347  size_type len = uchar_traits::Length( t );
12348  if ( p < len )
12349  {
12350  m_data->Allocate( n = pcl::Min( n, len-p ) );
12351  t += p;
12352  for ( iterator i = m_data->string; i < m_data->end; ++i, ++t )
12353  *i = char_type( uint8( *t ) );
12354  }
12355 }
12356 
12358 {
12359  size_type len = uchar_traits::Length( t );
12360  if ( len > 0 )
12361  {
12362  MaybeReallocate( len );
12363  for ( iterator i = m_data->string; i < m_data->end; ++i, ++t )
12364  *i = char_type( uint8( *t ) );
12365  }
12366  else
12367  Clear();
12368 
12369  return *this;
12370 }
12371 
12373 {
12374  return String::UTF8ToUTF16( Begin(), i, n );
12375 }
12376 
12377 // ----------------------------------------------------------------------------
12378 
12388 inline String operator +( const String::string_base& s1, const String::string_base& s2 )
12389 {
12390  String s = s1;
12391  s.Append( s2 );
12392  return s;
12393 }
12394 
12400 inline String operator +( String::string_base&& s1, const String::string_base& s2 )
12401 {
12402  s1.Append( s2 );
12403  return String( std::move( s1 ) );
12404 }
12405 
12411 inline String operator +( String&& s1, const String::string_base& s2 )
12412 {
12413  s1.Append( s2 );
12414  return std::move( s1 );
12415 }
12416 
12422 inline String operator +( const String::string_base& s1, String::string_base&& s2 )
12423 {
12424  s2.Prepend( s1 );
12425  return String( std::move( s2 ) );
12426 }
12427 
12433 inline String operator +( const String::string_base& s1, String&& s2 )
12434 {
12435  s2.Prepend( s1 );
12436  return std::move( s2 );
12437 }
12438 
12444 inline String operator +( String::string_base&& s1, String::string_base&& s2 )
12445 {
12446  s1.Append( s2 );
12447  return String( std::move( s1 ) );
12448 }
12449 
12456 {
12457  s1.Append( s2 );
12458  return std::move( s1 );
12459 }
12460 
12467 {
12468  s1.Append( s2 );
12469  return String( std::move( s1 ) );
12470 }
12471 
12477 inline String operator +( String&& s1, String&& s2 )
12478 {
12479  s1.Append( s2 );
12480  return std::move( s1 );
12481 }
12482 
12483 // ----------------------------------------------------------------------------
12484 
12491 {
12492  String s = s1;
12493  s.Append( t2 );
12494  return s;
12495 }
12496 
12503 {
12504  s1.Append( t2 );
12505  return String( std::move( s1 ) );
12506 }
12507 
12514 {
12515  s1.Append( t2 );
12516  return std::move( s1 );
12517 }
12518 
12525 {
12526  String s = s2;
12527  s.Prepend( t1 );
12528  return s;
12529 }
12530 
12537 {
12538  s2.Prepend( t1 );
12539  return String( std::move( s2 ) );
12540 }
12541 
12548 {
12549  s2.Prepend( t1 );
12550  return std::move( s2 );
12551 }
12552 
12553 // ----------------------------------------------------------------------------
12554 
12560 inline String operator +( const String::string_base& s1, String::char_type c2 )
12561 {
12562  String s = s1;
12563  s.Append( c2 );
12564  return s;
12565 }
12566 
12572 inline String operator +( String::string_base&& s1, String::char_type c2 )
12573 {
12574  s1.Append( c2 );
12575  return String( std::move( s1 ) );
12576 }
12577 
12584 {
12585  s1.Append( c2 );
12586  return std::move( s1 );
12587 }
12588 
12594 inline String operator +( String::char_type c1, const String::string_base& s2 )
12595 {
12596  String s = s2;
12597  s.Prepend( c1 );
12598  return s;
12599 }
12600 
12606 inline String operator +( String::char_type c1, String::string_base&& s2 )
12607 {
12608  s2.Prepend( c1 );
12609  return String( std::move( s2 ) );
12610 }
12611 
12618 {
12619  s2.Prepend( c1 );
12620  return std::move( s2 );
12621 }
12622 
12623 // ----------------------------------------------------------------------------
12624 
12630 inline String operator +( const String::string_base& s1, const char16_t* t2 )
12631 {
12632  String s = s1;
12633  s.Append( t2 );
12634  return s;
12635 }
12636 
12642 inline String operator +( String::string_base&& s1, const char16_t* t2 )
12643 {
12644  String s = std::move( s1 );
12645  s.Append( t2 );
12646  return s;
12647 }
12648 
12654 inline String operator +( String&& s1, const char16_t* t2 )
12655 {
12656  s1.Append( t2 );
12657  return std::move( s1 );
12658 }
12659 
12665 inline String operator +( const char16_t* t1, const String::string_base& s2 )
12666 {
12667  String s = s2;
12668  s.Prepend( t1 );
12669  return s;
12670 }
12671 
12677 inline String operator +( const char16_t* t1, String::string_base&& s2 )
12678 {
12679  String s = std::move( s2 );
12680  s.Prepend( t1 );
12681  return s;
12682 }
12683 
12689 inline String operator +( const char16_t* t1, String&& s2 )
12690 {
12691  s2.Prepend( t1 );
12692  return std::move( s2 );
12693 }
12694 
12695 // ----------------------------------------------------------------------------
12696 
12702 inline String operator +( const String::string_base& s1, char16_t c2 )
12703 {
12704  String s = s1;
12705  s.Append( c2 );
12706  return s;
12707 }
12708 
12714 inline String operator +( String::string_base&& s1, char16_t c2 )
12715 {
12716  String s = std::move( s1 );
12717  s.Append( c2 );
12718  return s;
12719 }
12720 
12726 inline String operator +( String&& s1, char16_t c2 )
12727 {
12728  s1.Append( c2 );
12729  return std::move( s1 );
12730 }
12731 
12737 inline String operator +( char16_t c1, const String::string_base& s2 )
12738 {
12739  String s = s2;
12740  s.Prepend( c1 );
12741  return s;
12742 }
12743 
12749 inline String operator +( char16_t c1, String::string_base&& s2 )
12750 {
12751  String s = std::move( s2 );
12752  s.Prepend( c1 );
12753  return s;
12754 }
12755 
12761 inline String operator +( char16_t c1, String&& s2 )
12762 {
12763  s2.Prepend( c1 );
12764  return std::move( s2 );
12765 }
12766 
12767 // ----------------------------------------------------------------------------
12768 
12774 inline String operator +( const String::string_base& s1, const wchar_t* t2 )
12775 {
12776  String s = s1;
12777  s.Append( t2 );
12778  return s;
12779 }
12780 
12786 inline String operator +( String::string_base&& s1, const wchar_t* t2 )
12787 {
12788  String s = std::move( s1 );
12789  s.Append( t2 );
12790  return s;
12791 }
12792 
12798 inline String operator +( String&& s1, const wchar_t* t2 )
12799 {
12800  s1.Append( t2 );
12801  return std::move( s1 );
12802 }
12803 
12809 inline String operator +( const wchar_t* t1, const String::string_base& s2 )
12810 {
12811  String s = s2;
12812  s.Prepend( t1 );
12813  return s;
12814 }
12815 
12821 inline String operator +( const wchar_t* t1, String::string_base&& s2 )
12822 {
12823  String s = std::move( s2 );
12824  s.Prepend( t1 );
12825  return s;
12826 }
12827 
12833 inline String operator +( const wchar_t* t1, String&& s2 )
12834 {
12835  s2.Prepend( t1 );
12836  return std::move( s2 );
12837 }
12838 
12839 // ----------------------------------------------------------------------------
12840 
12846 inline String operator +( const String::string_base& s1, wchar_t c2 )
12847 {
12848  String s = s1;
12849  s.Append( c2 );
12850  return s;
12851 }
12852 
12858 inline String operator +( String::string_base&& s1, wchar_t c2 )
12859 {
12860  String s = std::move( s1 );
12861  s.Append( c2 );
12862  return s;
12863 }
12864 
12870 inline String operator +( String&& s1, wchar_t c2 )
12871 {
12872  s1.Append( c2 );
12873  return std::move( s1 );
12874 }
12875 
12881 inline String operator +( wchar_t c1, const String::string_base& s2 )
12882 {
12883  String s = s2;
12884  s.Prepend( c1 );
12885  return s;
12886 }
12887 
12893 inline String operator +( wchar_t c1, String::string_base&& s2 )
12894 {
12895  String s = std::move( s2 );
12896  s.Prepend( c1 );
12897  return s;
12898 }
12899 
12905 inline String operator +( wchar_t c1, String&& s2 )
12906 {
12907  s2.Prepend( c1 );
12908  return std::move( s2 );
12909 }
12910 
12911 // ----------------------------------------------------------------------------
12912 
12919 {
12920  String s = s1;
12921  s.Append( s2 );
12922  return s;
12923 }
12924 
12931 {
12932  String s = std::move( s1 );
12933  s.Append( s2 );
12934  return s;
12935 }
12936 
12943 {
12944  s1.Append( s2 );
12945  return std::move( s1 );
12946 }
12947 
12954 {
12955  String s = s2;
12956  s.Prepend( s1 );
12957  return s;
12958 }
12959 
12966 {
12967  String s = std::move( s2 );
12968  s.Prepend( s1 );
12969  return s;
12970 }
12971 
12978 {
12979  s2.Prepend( s1 );
12980  return std::move( s2 );
12981 }
12982 
12983 // ----------------------------------------------------------------------------
12984 
12991 {
12992  String s = s1;
12993  s.Append( t2 );
12994  return s;
12995 }
12996 
13003 {
13004  String s = std::move( s1 );
13005  s.Append( t2 );
13006  return s;
13007 }
13008 
13015 {
13016  s1.Append( t2 );
13017  return std::move( s1 );
13018 }
13019 
13026 {
13027  String s = s2;
13028  s.Prepend( t1 );
13029  return s;
13030 }
13031 
13038 {
13039  String s = std::move( s2 );
13040  s.Prepend( t1 );
13041  return s;
13042 }
13043 
13050 {
13051  s2.Prepend( t1 );
13052  return std::move( s2 );
13053 }
13054 
13055 // ----------------------------------------------------------------------------
13056 
13063 {
13064  String s = s1;
13065  s.Append( c2 );
13066  return s;
13067 }
13068 
13075 {
13076  s1.Append( String::char_type( c2 ) );
13077  return String( std::move( s1 ) );
13078 }
13079 
13086 {
13087  s1.Append( c2 );
13088  return std::move( s1 );
13089 }
13090 
13097 {
13098  String s = s2;
13099  s.Prepend( c1 );
13100  return s;
13101 }
13102 
13109 {
13110  s2.Prepend( String::char_type( c1 ) );
13111  return String( std::move( s2 ) );
13112 }
13113 
13120 {
13121  s2.Prepend( c1 );
13122  return std::move( s2 );
13123 }
13124 
13125 // ----------------------------------------------------------------------------
13126 
13133 {
13134  s1.Append( s2 );
13135  return s1;
13136 }
13137 
13143 inline String& operator <<( String&& s1, const String::string_base& s2 )
13144 {
13145  s1.Append( s2 );
13146  return s1;
13147 }
13148 
13155 {
13156  s1.Append( t2 );
13157  return s1;
13158 }
13159 
13166 {
13167  s1.Append( t2 );
13168  return s1;
13169 }
13170 
13177 {
13178  s1.Append( c2 );
13179  return s1;
13180 }
13181 
13188 {
13189  s1.Append( c2 );
13190  return s1;
13191 }
13192 
13198 inline String& operator <<( String& s1, const char16_t* t2 )
13199 {
13200  s1.Append( t2 );
13201  return s1;
13202 }
13203 
13209 inline String& operator <<( String&& s1, const char16_t* t2 )
13210 {
13211  s1.Append( t2 );
13212  return s1;
13213 }
13214 
13220 inline String& operator <<( String& s1, char16_t c2 )
13221 {
13222  s1.Append( c2 );
13223  return s1;
13224 }
13225 
13231 inline String& operator <<( String&& s1, char16_t c2 )
13232 {
13233  s1.Append( c2 );
13234  return s1;
13235 }
13236 
13242 inline String& operator <<( String& s1, const wchar_t* t2 )
13243 {
13244  s1.Append( t2 );
13245  return s1;
13246 }
13247 
13253 inline String& operator <<( String&& s1, const wchar_t* t2 )
13254 {
13255  s1.Append( t2 );
13256  return s1;
13257 }
13258 
13264 inline String& operator <<( String& s1, wchar_t c2 )
13265 {
13266  s1.Append( c2 );
13267  return s1;
13268 }
13269 
13275 inline String& operator <<( String&& s1, wchar_t c2 )
13276 {
13277  s1.Append( c2 );
13278  return s1;
13279 }
13280 
13287 {
13288  s1.Append( s2 );
13289  return s1;
13290 }
13291 
13297 inline String& operator <<( String&& s1, const String::string8_base& s2 )
13298 {
13299  s1.Append( s2 );
13300  return s1;
13301 }
13302 
13309 {
13310  s1.Append( t2 );
13311  return s1;
13312 }
13313 
13320 {
13321  s1.Append( t2 );
13322  return s1;
13323 }
13324 
13331 {
13332  s1.Append( c2 );
13333  return s1;
13334 }
13335 
13342 {
13343  s1.Append( c2 );
13344  return s1;
13345 }
13346 
13347 // ----------------------------------------------------------------------------
13348 
13357 inline bool operator ==( const String& s1, const char16_t* t2 ) noexcept
13358 {
13359  return s1.CompareCodePoints( t2 ) == 0;
13360 }
13361 
13368 inline bool operator <( const String& s1, const char16_t* t2 ) noexcept
13369 {
13370  return s1.CompareCodePoints( t2 ) < 0;
13371 }
13372 
13379 inline bool operator <=( const String& s1, const char16_t* t2 ) noexcept
13380 {
13381  return s1.CompareCodePoints( t2 ) <= 0;
13382 }
13383 
13390 inline bool operator >( const String& s1, const char16_t* t2 ) noexcept
13391 {
13392  return s1.CompareCodePoints( t2 ) > 0;
13393 }
13394 
13401 inline bool operator >=( const String& s1, const char16_t* t2 ) noexcept
13402 {
13403  return s1.CompareCodePoints( t2 ) >= 0;
13404 }
13405 
13406 // ----------------------------------------------------------------------------
13407 
13412 inline bool operator ==( const char16_t* t1, const String& s2 ) noexcept
13413 {
13414  return s2.CompareCodePoints( t1 ) == 0;
13415 }
13416 
13423 inline bool operator <( const char16_t* t1, const String& s2 ) noexcept
13424 {
13425  return s2.CompareCodePoints( t1 ) > 0;
13426 }
13427 
13434 inline bool operator <=( const char16_t* t1, const String& s2 ) noexcept
13435 {
13436  return s2.CompareCodePoints( t1 ) >= 0;
13437 }
13438 
13445 inline bool operator >( const char16_t* t1, const String& s2 ) noexcept
13446 {
13447  return s2.CompareCodePoints( t1 ) < 0;
13448 }
13449 
13456 inline bool operator >=( const char16_t* t1, const String& s2 ) noexcept
13457 {
13458  return s2.CompareCodePoints( t1 ) <= 0;
13459 }
13460 
13461 // ----------------------------------------------------------------------------
13462 
13467 inline bool operator ==( const String& s1, char16_t c2 ) noexcept
13468 {
13469  return s1.CompareCodePoints( c2 ) == 0;
13470 }
13471 
13478 inline bool operator <( const String& s1, char16_t c2 ) noexcept
13479 {
13480  return s1.CompareCodePoints( c2 ) < 0;
13481 }
13482 
13489 inline bool operator <=( const String& s1, char16_t c2 ) noexcept
13490 {
13491  return s1.CompareCodePoints( c2 ) <= 0;
13492 }
13493 
13500 inline bool operator >( const String& s1, char16_t c2 ) noexcept
13501 {
13502  return s1.CompareCodePoints( c2 ) > 0;
13503 }
13504 
13511 inline bool operator >=( const String& s1, char16_t c2 ) noexcept
13512 {
13513  return s1.CompareCodePoints( c2 ) >= 0;
13514 }
13515 
13516 // ----------------------------------------------------------------------------
13517 
13522 inline bool operator ==( char16_t c1, const String& s2 ) noexcept
13523 {
13524  return s2.CompareCodePoints( c1 ) == 0;
13525 }
13526 
13533 inline bool operator <( char16_t c1, const String& s2 ) noexcept
13534 {
13535  return s2.CompareCodePoints( c1 ) > 0;
13536 }
13537 
13544 inline bool operator <=( char16_t c1, const String& s2 ) noexcept
13545 {
13546  return s2.CompareCodePoints( c1 ) >= 0;
13547 }
13548 
13555 inline bool operator >( char16_t c1, const String& s2 ) noexcept
13556 {
13557  return s2.CompareCodePoints( c1 ) < 0;
13558 }
13559 
13566 inline bool operator >=( char16_t c1, const String& s2 ) noexcept
13567 {
13568  return s2.CompareCodePoints( c1 ) <= 0;
13569 }
13570 
13571 // ----------------------------------------------------------------------------
13572 
13577 inline bool operator ==( const String& s1, const wchar_t* t2 ) noexcept
13578 {
13579  return s1.CompareCodePoints( t2 ) == 0;
13580 }
13581 
13588 inline bool operator <( const String& s1, const wchar_t* t2 ) noexcept
13589 {
13590  return s1.CompareCodePoints( t2 ) < 0;
13591 }
13592 
13599 inline bool operator <=( const String& s1, const wchar_t* t2 ) noexcept
13600 {
13601  return s1.CompareCodePoints( t2 ) <= 0;
13602 }
13603 
13610 inline bool operator >( const String& s1, const wchar_t* t2 ) noexcept
13611 {
13612  return s1.CompareCodePoints( t2 ) > 0;
13613 }
13614 
13621 inline bool operator >=( const String& s1, const wchar_t* t2 ) noexcept
13622 {
13623  return s1.CompareCodePoints( t2 ) >= 0;
13624 }
13625 
13626 // ----------------------------------------------------------------------------
13627 
13632 inline bool operator ==( const wchar_t* t1, const String& s2 ) noexcept
13633 {
13634  return s2.CompareCodePoints( t1 ) == 0;
13635 }
13636 
13643 inline bool operator <( const wchar_t* t1, const String& s2 ) noexcept
13644 {
13645  return s2.CompareCodePoints( t1 ) > 0;
13646 }
13647 
13654 inline bool operator <=( const wchar_t* t1, const String& s2 ) noexcept
13655 {
13656  return s2.CompareCodePoints( t1 ) >= 0;
13657 }
13658 
13665 inline bool operator >( const wchar_t* t1, const String& s2 ) noexcept
13666 {
13667  return s2.CompareCodePoints( t1 ) < 0;
13668 }
13669 
13676 inline bool operator >=( const wchar_t* t1, const String& s2 ) noexcept
13677 {
13678  return s2.CompareCodePoints( t1 ) <= 0;
13679 }
13680 
13681 // ----------------------------------------------------------------------------
13682 
13687 inline bool operator ==( const String& s1, wchar_t c2 ) noexcept
13688 {
13689  return s1.CompareCodePoints( c2 ) == 0;
13690 }
13691 
13698 inline bool operator <( const String& s1, wchar_t c2 ) noexcept
13699 {
13700  return s1.CompareCodePoints( c2 ) < 0;
13701 }
13702 
13709 inline bool operator <=( const String& s1, wchar_t c2 ) noexcept
13710 {
13711  return s1.CompareCodePoints( c2 ) <= 0;
13712 }
13713 
13720 inline bool operator >( const String& s1, wchar_t c2 ) noexcept
13721 {
13722  return s1.CompareCodePoints( c2 ) > 0;
13723 }
13724 
13731 inline bool operator >=( const String& s1, wchar_t c2 ) noexcept
13732 {
13733  return s1.CompareCodePoints( c2 ) >= 0;
13734 }
13735 
13736 // ----------------------------------------------------------------------------
13737 
13742 inline bool operator ==( wchar_t c1, const String& s2 ) noexcept
13743 {
13744  return s2.CompareCodePoints( c1 ) == 0;
13745 }
13746 
13753 inline bool operator <( wchar_t c1, const String& s2 ) noexcept
13754 {
13755  return s2.CompareCodePoints( c1 ) > 0;
13756 }
13757 
13764 inline bool operator <=( wchar_t c1, const String& s2 ) noexcept
13765 {
13766  return s2.CompareCodePoints( c1 ) >= 0;
13767 }
13768 
13775 inline bool operator >( wchar_t c1, const String& s2 ) noexcept
13776 {
13777  return s2.CompareCodePoints( c1 ) < 0;
13778 }
13779 
13786 inline bool operator >=( wchar_t c1, const String& s2 ) noexcept
13787 {
13788  return s2.CompareCodePoints( c1 ) <= 0;
13789 }
13790 
13791 // ----------------------------------------------------------------------------
13792 
13797 inline bool operator ==( const String& s1, String::const_c_string8 t2 ) noexcept
13798 {
13799  return s1.CompareCodePoints( t2 ) == 0;
13800 }
13801 
13808 inline bool operator <( const String& s1, String::const_c_string8 t2 ) noexcept
13809 {
13810  return s1.CompareCodePoints( t2 ) < 0;
13811 }
13812 
13819 inline bool operator <=( const String& s1, String::const_c_string8 t2 ) noexcept
13820 {
13821  return s1.CompareCodePoints( t2 ) <= 0;
13822 }
13823 
13830 inline bool operator >( const String& s1, String::const_c_string8 t2 ) noexcept
13831 {
13832  return s1.CompareCodePoints( t2 ) > 0;
13833 }
13834 
13841 inline bool operator >=( const String& s1, String::const_c_string8 t2 ) noexcept
13842 {
13843  return s1.CompareCodePoints( t2 ) >= 0;
13844 }
13845 
13846 // ----------------------------------------------------------------------------
13847 
13852 inline bool operator ==( String::const_c_string8 t1, const String& s2 ) noexcept
13853 {
13854  return s2.CompareCodePoints( t1 ) == 0;
13855 }
13856 
13863 inline bool operator <( String::const_c_string8 t1, const String& s2 ) noexcept
13864 {
13865  return s2.CompareCodePoints( t1 ) > 0;
13866 }
13867 
13874 inline bool operator <=( String::const_c_string8 t1, const String& s2 ) noexcept
13875 {
13876  return s2.CompareCodePoints( t1 ) >= 0;
13877 }
13878 
13885 inline bool operator >( String::const_c_string8 t1, const String& s2 ) noexcept
13886 {
13887  return s2.CompareCodePoints( t1 ) < 0;
13888 }
13889 
13896 inline bool operator >=( String::const_c_string8 t1, const String& s2 ) noexcept
13897 {
13898  return s2.CompareCodePoints( t1 ) <= 0;
13899 }
13900 
13901 // ----------------------------------------------------------------------------
13902 
13907 inline bool operator ==( const String& s1, String::char8_type c2 ) noexcept
13908 {
13909  return s1.CompareCodePoints( c2 ) == 0;
13910 }
13911 
13918 inline bool operator <( const String& s1, String::char8_type c2 ) noexcept
13919 {
13920  return s1.CompareCodePoints( c2 ) < 0;
13921 }
13922 
13929 inline bool operator <=( const String& s1, String::char8_type c2 ) noexcept
13930 {
13931  return s1.CompareCodePoints( c2 ) <= 0;
13932 }
13933 
13940 inline bool operator >( const String& s1, String::char8_type c2 ) noexcept
13941 {
13942  return s1.CompareCodePoints( c2 ) > 0;
13943 }
13944 
13951 inline bool operator >=( const String& s1, String::char8_type c2 ) noexcept
13952 {
13953  return s1.CompareCodePoints( c2 ) >= 0;
13954 }
13955 
13956 // ----------------------------------------------------------------------------
13957 
13962 inline bool operator ==( String::char8_type c1, const String& s2 ) noexcept
13963 {
13964  return s2.CompareCodePoints( c1 ) == 0;
13965 }
13966 
13973 inline bool operator <( String::char8_type c1, const String& s2 ) noexcept
13974 {
13975  return s2.CompareCodePoints( c1 ) > 0;
13976 }
13977 
13984 inline bool operator <=( String::char8_type c1, const String& s2 ) noexcept
13985 {
13986  return s2.CompareCodePoints( c1 ) >= 0;
13987 }
13988 
13995 inline bool operator >( String::char8_type c1, const String& s2 ) noexcept
13996 {
13997  return s2.CompareCodePoints( c1 ) < 0;
13998 }
13999 
14006 inline bool operator >=( String::char8_type c1, const String& s2 ) noexcept
14007 {
14008  return s2.CompareCodePoints( c1 ) <= 0;
14009 }
14010 
14011 // ----------------------------------------------------------------------------
14012 
14013 #ifndef __PCL_NO_STRING_OSTREAM
14014 
14015 inline std::wostream& operator <<( std::wostream& o, const String& s )
14016 {
14017 #ifdef __PCL_WINDOWS
14018  return o << reinterpret_cast<const wchar_t*>( s.c_str() );
14019 #else
14020  Array<wchar_t> w = s.ToWCharArray();
14021  return o << w.Begin();
14022 #endif
14023 }
14024 
14025 inline std::ostream& operator <<( std::ostream& o, const String& s )
14026 {
14027  return o << s.ToUTF8();
14028 }
14029 
14030 #endif // __PCL_NO_STRING_OSTREAM
14031 
14032 // ----------------------------------------------------------------------------
14033 
14034 } // pcl
14035 
14036 #endif // __PCL_String_h
14037 
14038 // ----------------------------------------------------------------------------
14039 // EOF pcl/String.h - Released 2025-02-21T12:13:32Z
Provides memory allocation for PCL containers.
Definition: Allocator.h:132
Generic dynamic array.
Definition: Array.h:100
bool IsEmpty() const noexcept
Definition: Array.h:324
T * iterator
Definition: Array.h:125
A template instantiation of GenericCharTraits for char16_type.
Definition: CharTraits.h:1119
static size_type Length(const char_type *__restrict__ s) noexcept
Definition: CharTraits.h:1139
Generic complex number.
Definition: Complex.h:84
constexpr T Imag() const noexcept
Definition: Complex.h:136
constexpr T Real() const noexcept
Definition: Complex.h:120
Root base class of all PCL containers of objects.
Definition: Container.h:78
const T const_item_type
Definition: Container.h:89
A type-safe collection of enumerated flags.
Definition: Flags.h:85
static constexpr char_type LF() noexcept
Definition: CharTraits.h:599
static constexpr char_type Colon() noexcept
Definition: CharTraits.h:615
static constexpr char_type Null() noexcept
Definition: CharTraits.h:567
static constexpr char_type Hyphen() noexcept
Definition: CharTraits.h:631
static constexpr char_type Blank() noexcept
Definition: CharTraits.h:575
static constexpr char_type Comma() noexcept
Definition: CharTraits.h:607
static constexpr char_type Tab() noexcept
Definition: CharTraits.h:583
Generic character string.
Definition: String.h:493
size_type FindFirstIC(const_c_string t, size_type i=0) const noexcept
Definition: String.h:3058
iterator begin()
Definition: String.h:1125
void Append(const_c_string t)
Definition: String.h:1806
int CompareIC(const_c_string t, bool localeAware=true) const noexcept
Definition: String.h:3866
void JustifyCenter(size_type width, char_type fill=R::Blank())
Definition: String.h:3577
int CompareCodePoints(const GenericString< T, R1, A1 > &s, bool caseSensitive=true) const noexcept
Definition: String.h:3652
static size_type DeleteFreeList()
Definition: String.h:4286
iterator end()
Definition: String.h:1141
void Transfer(GenericString &s)
Definition: String.h:1224
bool WildMatch(const GenericString< T, R1, A1 > &pattern, bool caseSensitive=true) const noexcept
Definition: String.h:3916
bool ContainsIC(const_c_string t) const noexcept
Definition: String.h:3279
void ToUppercase()
Definition: String.h:4023
size_type FindFirst(const_c_string t, size_type i=0) const noexcept
Definition: String.h:3014
reverse_iterator ReverseBegin()
Definition: String.h:1050
const_iterator begin() const noexcept
Definition: String.h:1133
size_type Find(const GenericString< T, R1, A1 > &s, size_type i=0) const noexcept
Definition: String.h:3086
void ReplaceString(const GenericString< T, R1, A1 > &s1, const GenericString< T, R2, A2 > &s2, size_type i=0)
Definition: String.h:2101
bool StartsWith(const_c_string t) const noexcept
Definition: String.h:2860
void Insert(size_type i, const_c_string t, size_type n)
Definition: String.h:1741
const T * const_c_string
Definition: String.h:536
void JustifyRight(size_type width, char_type fill=R::Blank())
Definition: String.h:3558
void Fill(char_type c)
Definition: String.h:1398
void ReplaceChar(char_type c1, char_type c2, size_type i=0, size_type n=maxPos)
Definition: String.h:2078
bool StartsWithIC(char_type c) const noexcept
Definition: String.h:2913
void Add(const_c_string t, size_type n)
Definition: String.h:1852
void Swap(GenericString &s) noexcept
Definition: String.h:1384
void Insert(size_type i, char_type c, size_type n=1)
Definition: String.h:1753
GenericString ResizedToNullTerminated() const
Definition: String.h:1578
size_type BreakIC(C &list, char_type c, bool trim=false, size_type i=0) const
Definition: String.h:2781
GenericString SetToLength(size_type n) const
Definition: String.h:1553
void TrimRight()
Definition: String.h:3337
iterator At(size_type i)
Definition: String.h:899
void DeleteLeft(size_type i)
Definition: String.h:2194
void EnsureDoubleQuoted()
Definition: String.h:3467
bool EndsWithIC(char_type c) const noexcept
Definition: String.h:2991
void c_copy(iterator dst, size_type maxCharsToCopy, size_type i=0) const noexcept
Definition: String.h:1672
GenericString Left(size_type n) const
Definition: String.h:2318
void Prepend(const GenericString< T, R1, A1 > &s)
Definition: String.h:1890
size_type FindIC(const_c_string t, size_type i=0) const noexcept
Definition: String.h:3119
GenericString(char_type c, size_type n=1)
Definition: String.h:675
void Assign(const_iterator i, const_iterator j)
Definition: String.h:1317
char_type LastChar() const noexcept
Definition: String.h:2837
size_type FindLast(const_c_string t, size_type r=maxPos) const noexcept
Definition: String.h:3158
size_type FindIC(const GenericString< T, R1, A1 > &s, size_type i=0) const noexcept
Definition: String.h:3111
void Assign(const_c_string t, size_type i, size_type n)
Definition: String.h:1353
GenericString(const_iterator i, const_iterator j)
Definition: String.h:631
void Assign(const GenericString &s, size_type i, size_type n)
Definition: String.h:1273
void EnsureUnique()
Definition: String.h:730
GenericString Enclosed(char_type c) const
Definition: String.h:3430
GenericString Suffix(size_type i) const
Definition: String.h:2364
void Insert(size_type i, const_iterator p, const_iterator q)
Definition: String.h:1713
void EnsureEnclosed(char_type c)
Definition: String.h:3386
reverse_iterator ReverseEnd()
Definition: String.h:1084
void ToCaseFolded()
Definition: String.h:3995
size_type FindLast(const GenericString< T, R1, A1 > &s, size_type r=maxPos) const noexcept
Definition: String.h:3142
const_iterator end() const noexcept
Definition: String.h:1149
GenericString Lowercase() const
Definition: String.h:4048
bool Contains(const GenericString< T, R1, A1 > &s) const noexcept
Definition: String.h:3240
size_type FindFirst(char_type c, size_type i=0) const noexcept
Definition: String.h:3026
GenericString(std::initializer_list< char_type > l)
Definition: String.h:651
void Assign(std::initializer_list< char_type > l)
Definition: String.h:1335
void Insert(size_type i, const_c_string t)
Definition: String.h:1727
void Add(char_type c, size_type n=1)
Definition: String.h:1870
GenericString TrimmedRight() const
Definition: String.h:3373
void Reserve(size_type n)
Definition: String.h:1480
GenericString Unquoted() const
Definition: String.h:3522
bool StartsWith(char_type c) const noexcept
Definition: String.h:2874
uint64 Hash64(uint64 seed=0) const noexcept
Definition: String.h:4252
void Assign(const GenericString &s)
Definition: String.h:1193
void ReplaceStringIC(const GenericString< T, R1, A1 > &s1, const GenericString< T, R2, A2 > &s2, size_type i=0)
Definition: String.h:2125
bool StartsWithIC(const GenericString< T, R1, A1 > &s) const noexcept
Definition: String.h:2884
GenericString Uppercase() const
Definition: String.h:4059
bool ContainsIC(const GenericString< T, R1, A1 > &s) const noexcept
Definition: String.h:3268
size_type FindLast(char_type c, size_type r=maxPos) const noexcept
Definition: String.h:3170
GenericString SingleQuoted() const
Definition: String.h:3454
char_type FirstChar() const noexcept
Definition: String.h:2828
void Append(const_c_string t, size_type n)
Definition: String.h:1798
void Assign(const_c_string t)
Definition: String.h:1285
GenericString Trimmed() const
Definition: String.h:3349
void Prepend(const_c_string t)
Definition: String.h:1931
GenericString Sorted() const
Definition: String.h:4107
size_type IndexAt(const_iterator i) const noexcept
Definition: String.h:1115
GenericString LeftJustified(size_type width, char_type fill=R::Blank()) const
Definition: String.h:3595
size_type FindIC(char_type c, size_type i=0) const noexcept
Definition: String.h:3127
static size_type BytesPerChar() noexcept
Definition: String.h:750
void Replace(size_type i, size_type n, const GenericString< T, R1, A1 > &s)
Definition: String.h:1969
void DeleteCharIC(char_type c, size_type i=0)
Definition: String.h:2216
GenericString Right(size_type n) const
Definition: String.h:2340
bool IsValid() const noexcept
Definition: String.h:818
GenericString RightJustified(size_type width, char_type fill=R::Blank()) const
Definition: String.h:3608
bool WildMatch(const_c_string pattern, bool caseSensitive=true) const noexcept
Definition: String.h:3955
size_type Break(C &list, char_type c, bool trim=false, size_type i=0) const
Definition: String.h:2518
int Compare(const GenericString< T, R1, A1 > &s, bool caseSensitive=true, bool localeAware=true) const noexcept
Definition: String.h:3747
void ToLowercase()
Definition: String.h:4009
bool StartsWith(const GenericString< T, R1, A1 > &s) const noexcept
Definition: String.h:2846
size_type UpperBound() const noexcept
Definition: String.h:856
GenericString Prefix(size_type i) const
Definition: String.h:2375
const_reverse_iterator ReverseBegin() const noexcept
Definition: String.h:1066
int CompareCodePoints(const_c_string t, bool caseSensitive=true) const noexcept
Definition: String.h:3682
int CompareIC(char_type c, bool localeAware=true) const noexcept
Definition: String.h:3895
GenericString(const GenericString &s)
Definition: String.h:583
typename container_type::const_item_type const_item_type
Definition: String.h:506
bool EndsWithIC(const GenericString< T, R1, A1 > &s) const noexcept
Definition: String.h:2961
bool IsEmpty() const noexcept
Definition: String.h:830
bool HasWildcards() const noexcept
Definition: String.h:3983
bool IsValidIdentifier() const noexcept
Definition: String.h:4231
const allocator & Allocator() const noexcept
Definition: String.h:867
void DeleteRight(size_type i)
Definition: String.h:2185
bool StartsWithIC(const_c_string t) const noexcept
Definition: String.h:2898
int CompareCodePoints(char_type c, bool caseSensitive=true) const noexcept
Definition: String.h:3715
GenericString CenterJustified(size_type width, char_type fill=R::Blank()) const
Definition: String.h:3621
void Add(const_c_string t)
Definition: String.h:1861
void Append(const_iterator i, const_iterator j)
Definition: String.h:1789
void DeleteString(const GenericString< T, R1, A1 > &s, size_type i=0)
Definition: String.h:2227
void Fill(char_type c, size_type i, size_type n=maxPos)
Definition: String.h:1427
void Replace(size_type i, size_type n, const_c_string t)
Definition: String.h:2008
iterator Begin()
Definition: String.h:988
void SetLength(size_type n)
Definition: String.h:1523
bool WildMatchIC(const GenericString< T, R1, A1 > &pattern) const noexcept
Definition: String.h:3935
size_type FindFirstIC(char_type c, size_type i=0) const noexcept
Definition: String.h:3073
GenericString(const_c_string t, size_type i, size_type n)
Definition: String.h:660
bool IsAliasOf(const GenericString &s) const noexcept
Definition: String.h:716
bool IsNumeral() const noexcept
Definition: String.h:4155
void Prepend(const_iterator i, const_iterator j)
Definition: String.h:1913
GenericString Sorted(BP p) const
Definition: String.h:4134
void DeleteStringIC(const_c_string t, size_type i=0)
Definition: String.h:2262
void Prepend(char_type c, size_type n=1)
Definition: String.h:1949
bool IsSymbol() const noexcept
Definition: String.h:4177
bool EndsWith(char_type c) const noexcept
Definition: String.h:2951
void SecureFill(char c='\0') noexcept
Definition: String.h:1466
size_type Break(C &list, const Array< S > &ca, bool trim=false, size_type i=0) const
Definition: String.h:2583
bool Contains(char_type c) const noexcept
Definition: String.h:3256
GenericString Reversed() const
Definition: String.h:4084
void DeleteString(const_c_string t, size_type i=0)
Definition: String.h:2237
bool WildMatchIC(const_c_string pattern) const noexcept
Definition: String.h:3974
uint32 Hash32(uint32 seed=0) const noexcept
Definition: String.h:4265
uint64 Hash(uint64 seed=0) const noexcept
Definition: String.h:4274
size_type FindLastIC(char_type c, size_type r=maxPos) const noexcept
Definition: String.h:3226
void Delete(size_type i, size_type n=1)
Definition: String.h:2147
size_type FindFirstIC(const GenericString< T, R1, A1 > &s, size_type i=0) const noexcept
Definition: String.h:3043
size_type Size() const noexcept
Definition: String.h:761
void DeleteChar(char_type c, size_type i=0)
Definition: String.h:2204
void DeleteStringIC(const GenericString< T, R1, A1 > &s, size_type i=0)
Definition: String.h:2250
int CompareIC(const GenericString< T, R1, A1 > &s, bool localeAware=true) const noexcept
Definition: String.h:3840
GenericString(const_c_string t)
Definition: String.h:602
int Compare(const_c_string t, bool caseSensitive=true, bool localeAware=true) const noexcept
Definition: String.h:3780
const_iterator End() const noexcept
Definition: String.h:1032
size_type Break(C &list, const GenericString< T, R1, A1 > &s, bool trim=false, size_type i=0) const
Definition: String.h:2398
void JustifyLeft(size_type width, char_type fill=R::Blank())
Definition: String.h:3540
void ResizeToNullTerminated()
Definition: String.h:1569
void Add(const GenericString< T, R1, A1 > &s)
Definition: String.h:1834
size_type Find(const_c_string t, size_type i=0) const noexcept
Definition: String.h:3094
size_type LowerBound() const noexcept
Definition: String.h:843
void Append(const GenericString< T, R1, A1 > &s)
Definition: String.h:1766
void Replace(size_type i, size_type n, char_type c, size_type nc=1)
Definition: String.h:2043
bool Contains(const_c_string t) const noexcept
Definition: String.h:3248
void ReplaceString(const_c_string t1, const_c_string t2, size_type i=0)
Definition: String.h:2112
void ReplaceStringIC(const_c_string t1, const_c_string t2, size_type i=0)
Definition: String.h:2138
void SetAllocator(const allocator &a)
Definition: String.h:881
size_type FindFirst(const GenericString< T, R1, A1 > &s, size_type i=0) const noexcept
Definition: String.h:3002
size_type Capacity() const noexcept
Definition: String.h:785
iterator End()
Definition: String.h:1018
const T * const_iterator
Definition: String.h:546
size_type Available() const noexcept
Definition: String.h:797
void Prepend(const_c_string t, size_type n)
Definition: String.h:1922
void Sort(BP p)
Definition: String.h:4120
void ReplaceCharIC(char_type c1, char_type c2, size_type i=0, size_type n=maxPos)
Definition: String.h:2090
void EnsureSingleQuoted()
Definition: String.h:3443
bool ContainsIC(char_type c) const noexcept
Definition: String.h:3290
size_type BreakIC(C &list, const GenericString< T, R1, A1 > &s, bool trim=false, size_type i=0) const
Definition: String.h:2650
bool EndsWithIC(const_c_string t) const noexcept
Definition: String.h:2976
size_type Find(char_type c, size_type i=0) const noexcept
Definition: String.h:3102
size_type Break(C &list, const_c_string s, bool trim=false, size_type i=0) const
Definition: String.h:2458
void Transfer(GenericString &&s)
Definition: String.h:1239
GenericString DoubleQuoted() const
Definition: String.h:3478
size_type FindLastIC(const GenericString< T, R1, A1 > &s, size_type r=maxPos) const noexcept
Definition: String.h:3192
void Append(char_type c, size_type n=1)
Definition: String.h:1824
void Add(const_iterator i, const_iterator j)
Definition: String.h:1843
const_iterator Begin() const noexcept
Definition: String.h:1002
bool EndsWith(const_c_string t) const noexcept
Definition: String.h:2937
bool IsUnique() const noexcept
Definition: String.h:704
void Insert(size_type i, const GenericString< T, R1, A1 > &s)
Definition: String.h:1696
const_iterator At(size_type i) const noexcept
Definition: String.h:915
void Assign(char_type c, size_type n=1)
Definition: String.h:1370
GenericString CaseFolded() const
Definition: String.h:4037
c_string Release()
Definition: String.h:1648
GenericString Squeezed() const
Definition: String.h:1628
int Compare(char_type c, bool caseSensitive=true, bool localeAware=true) const noexcept
Definition: String.h:3814
const_c_string c_str() const noexcept
Definition: String.h:1162
GenericString Substring(size_type i, size_type n=maxPos) const
Definition: String.h:2293
const_reverse_iterator ReverseEnd() const noexcept
Definition: String.h:1100
GenericString TrimmedLeft() const
Definition: String.h:3361
size_type FindLastIC(const_c_string t, size_type r=maxPos) const noexcept
Definition: String.h:3211
size_type Length() const noexcept
Definition: String.h:772
bool IsValidIdentifier(distance_type &pos) const noexcept
Definition: String.h:4200
bool EndsWith(const GenericString< T, R1, A1 > &s) const noexcept
Definition: String.h:2922
GenericString(GenericString &&s)
Definition: String.h:592
size_type BreakIC(C &list, const_c_string s, bool trim=false, size_type i=0) const
Definition: String.h:2716
Generic vector of arbitrary length.
Definition: Vector.h:107
A template instantiation of GenericCharTraits for the char type.
Definition: CharTraits.h:859
Eight-bit string (ISO/IEC-8859-1 or UTF-8 string)
Definition: String.h:5445
IsoString(const ustring_base &s)
Definition: String.h:5598
IsoString(Complex< float > &x)
Definition: String.h:5820
IsoString EncodedHTMLSpecialChars() const
Definition: String.h:6386
IsoString()=default
IsoString(std::initializer_list< char_type > l)
Definition: String.h:5645
string_base::const_iterator const_iterator
Definition: String.h:5506
GenericVector< double > ParseListOfDoubleAsVector(char separator=',', int maxCount=int_max) const
IsoString(const ByteArray &B)
Definition: String.h:5685
bool TryToInt(int &value, int base) const noexcept
string_base::item_type item_type
Definition: String.h:5461
IsoString & ToSeparated(const C &c, char_type separator)
Definition: String.h:6153
static IsoString ToURLEncoded(const void *data, size_type length)
ustring_base::char_type uchar_type
Definition: String.h:5528
double ToDouble() const
IsoString & ToURLDecoded()
IsoString & ToCommaSeparated(const C &c)
Definition: String.h:6268
IsoString & ToHyphenated(const C &c)
Definition: String.h:6357
int AppendVFormat(const_c_string fmt, va_list paramList)
bool TryToUInt(unsigned &value) const noexcept
Definition: String.h:6979
IsoString & ToSeparated(const C &c, const_c_string separator)
Definition: String.h:6231
IsoString(short x)
Definition: String.h:5704
IsoString & ToSeparated(const C &c, char_type separator, AF append)
Definition: String.h:6176
IsoString URLDecoded() const
Definition: String.h:6558
string_base::iterator iterator
Definition: String.h:5501
bool TryToInt64(long long &value, int base) const noexcept
ByteArray FromBase64() const
IsoString(unsigned long x)
Definition: String.h:5754
IsoString(const string_base &s)
Definition: String.h:5566
static IsoString ToBase64URL(const void *data, size_type length)
Definition: String.h:7700
ByteArray FromHex() const
bool TryToInt64(long long &value) const noexcept
Definition: String.h:7064
IsoString(unsigned long long x)
Definition: String.h:5774
IsoString(long long x)
Definition: String.h:5764
static ByteArray FromURLEncoded(const C &c)
Definition: String.h:6495
string_base::const_item_type const_item_type
Definition: String.h:5466
IsoString(const IsoString &)=default
unsigned long long ToUInt64(int base) const
bool TryToDouble(double &value) const noexcept
IsoString(bool x)
Definition: String.h:5694
ustring_base::const_c_string const_c_ustring
Definition: String.h:5543
IsoString & ToEncodedHTMLSpecialChars()
void ParseISO8601DateTime(int &year, int &month, int &day, double &dayf, double &tz) const
IsoString(IsoString &&)=default
IsoString & ToColonSeparated(const C &c)
Definition: String.h:6283
bool TryToInt(int &value) const noexcept
Definition: String.h:6882
unsigned long long ToUInt64() const
Definition: String.h:7118
IsoString & ToSpaceSeparated(const C &c)
Definition: String.h:6298
IsoString(long double x)
Definition: String.h:5804
IsoString DecodedHTMLSpecialChars() const
Definition: String.h:6416
static IsoString ToURLDecoded(const C &c)
Definition: String.h:6536
IsoString & ToNewLineSeparated(const C &c)
Definition: String.h:6328
long ToInt(int base) const
IsoString(int x)
Definition: String.h:5724
static IsoString ToURLEncoded(const C &c)
Definition: String.h:6447
IsoString & ToSeparated(const C &c, const_c_string separator, AF append)
Definition: String.h:6253
IsoString(Complex< long double > &x)
Definition: String.h:5840
long long ToInt64() const
Definition: String.h:7035
static IsoString ToBase64(const void *data, size_type length)
IsoString & ToTabSeparated(const C &c)
Definition: String.h:6313
IsoString(unsigned int x)
Definition: String.h:5734
bool TryToUInt64(unsigned long long &value) const noexcept
Definition: String.h:7147
IsoString(Complex< double > &x)
Definition: String.h:5830
IsoString(string_base &&s)
Definition: String.h:5580
unsigned long ToUInt(int base) const
IsoString & ToDecodedHTMLSpecialChars()
IsoString(char_type c, size_type n=1)
Definition: String.h:5623
ByteArray ToByteArray() const
Definition: String.h:7736
static IsoString ToHex(const C &c)
Definition: String.h:7649
ByteArray FromBase64URL() const
Definition: String.h:7778
IsoString(const_c_ustring t)
Definition: String.h:5660
long ToInt() const
Definition: String.h:6854
IsoString & AppendFormat(const_c_string fmt,...)
Definition: String.h:6627
ustring_base::c_string c_ustring
Definition: String.h:5538
static IsoString ToHex(const void *data, size_type length)
GenericVector< float > ParseListOfFloatAsVector(char separator=',', int maxCount=int_max) const
IsoString(long x)
Definition: String.h:5744
static IsoString UUID()
IsoString & ToSeparated(const C &c, const IsoString &separator)
Definition: String.h:6192
static IsoString ToBase64URL(const C &c)
Definition: String.h:7724
IsoString & ToSeparated(const C &c, const IsoString &separator, AF append)
Definition: String.h:6215
static IsoString Random(size_type n, RandomizationOptions options=RandomizationOption::Default)
IsoString & Format(const_c_string fmt,...)
Definition: String.h:6605
ustring_base MBSToWCS() const
static ByteArray FromURLEncoded(const void *data, size_type length)
bool ToBool() const
Array< double > ParseListOfDouble(char separator=',', size_type maxCount=~size_type(0)) const
string_base::block_allocator block_allocator
Definition: String.h:5481
IsoString(const_iterator i, const_iterator j)
Definition: String.h:5632
string_base::char_type char_type
Definition: String.h:5471
static IsoString CurrentLocalISO8601DateTime(const ISO8601ConversionOptions &options=ISO8601ConversionOptions())
string_base::const_c_string const_c_string
Definition: String.h:5496
IsoString & ToURLEncoded()
IsoString(unsigned short x)
Definition: String.h:5714
static IsoString CurrentUTCISO8601DateTime(const ISO8601ConversionOptions &options=ISO8601ConversionOptions())
bool TryParseISO8601DateTime(int &year, int &month, int &day, double &dayf, double &tz) const noexcept
ustring_base UTF8ToUTF16(size_type i=0, size_type n=maxPos) const
Definition: String.h:12372
IsoString(const_c_string t)
Definition: String.h:5606
string_base::c_string c_string
Definition: String.h:5491
IsoString(double x)
Definition: String.h:5794
IsoString & ToNullSeparated(const C &c)
Definition: String.h:6343
IsoString & operator=(const IsoString &s)
Definition: String.h:5890
static IsoString ToURLDecoded(const void *data, size_type length)
static IsoString ToISO8601DateTime(int year, int month, int day, double dayf, double tz=0, const ISO8601ConversionOptions &options=ISO8601ConversionOptions())
unsigned long ToUInt() const
Definition: String.h:6951
IsoString URLEncoded() const
Definition: String.h:6469
ustring_base::const_iterator const_uchar_iterator
Definition: String.h:5553
ByteArray FromURLEncoded() const
Definition: String.h:6511
int VFormat(const_c_string fmt, va_list paramList)
Array< float > ParseListOfFloat(char separator=',', size_type maxCount=~size_type(0)) const
ustring_base::iterator uchar_iterator
Definition: String.h:5548
bool TryToBool(bool &value) const noexcept
bool TryToFloat(float &value) const noexcept
bool TryToUInt64(unsigned long long &value, int base) const noexcept
float ToFloat() const
long long ToInt64(int base) const
static IsoString ToBase64(const C &c)
Definition: String.h:7681
IsoString(const_c_string t, size_type i, size_type n)
Definition: String.h:5615
IsoString(float x)
Definition: String.h:5784
ustring_base ToString() const
Definition: String.h:6677
bool TryToUInt(unsigned &value, int base) const noexcept
A collection of string randomization options.
Definition: String.h:146
Reverse random access iterator.
Definition: Iterator.h:420
Unicode (UTF-16) string.
Definition: String.h:8148
String & ToNewLineSeparated(const C &c)
Definition: String.h:11006
String & ToSeparated(const C &c, const String &separator)
Definition: String.h:10832
String(float x)
Definition: String.h:8559
String(const_char8_iterator i, const_char8_iterator j)
Definition: String.h:8438
String(Complex< long double > &x)
Definition: String.h:8612
static IsoString UTF16ToUTF8(const_c_string string, size_type i=0, size_type n=maxPos)
static String CurrentUTCISO8601DateTime(const ISO8601ConversionOptions &options=ISO8601ConversionOptions())
bool TryToInt64(long long &value, int base) const noexcept
GenericVector< double > ParseListOfDoubleAsVector(char separator=',', int maxCount=int_max) const
static String CurrentLocalISO8601DateTime(const ISO8601ConversionOptions &options=ISO8601ConversionOptions())
bool TryToInt(int &value, int base) const noexcept
void Assign(const String &s, size_type i, size_type n)
Definition: String.h:8814
String(long x)
Definition: String.h:8519
void Assign(std::initializer_list< char_type > l)
Definition: String.h:8842
IsoString ToMBS() const
void Assign(const_iterator i, const_iterator j)
Definition: String.h:8831
IsoString ToLocal8Bit() const
Definition: String.h:11387
String(const wchar_t *t)
Definition: String.h:8392
string_base::char_type char_type
Definition: String.h:8174
String & ToSeparated(const C &c, char_type separator)
Definition: String.h:10793
long ToInt() const
Definition: String.h:11587
String(unsigned short x)
Definition: String.h:8489
int AppendVFormat(const_c_string8 fmt, va_list paramList)
Definition: String.h:11196
static String UTF32ToUTF16(const uint32 *string, size_type i=0, size_type n=maxPos)
String & ToSeparated(const C &c, char_type separator, AF append)
Definition: String.h:10816
Array< uint32 > ToUTF32(size_type i=0, size_type n=maxPos) const
Definition: String.h:11447
void ParseISO8601DateTime(int &year, int &month, int &day, double &dayf, double &tz) const
string8_base::const_c_string const_c_string8
Definition: String.h:8258
string_base::block_allocator block_allocator
Definition: String.h:8184
String & AppendFormat(const wchar_t *fmt,...)
Definition: String.h:11239
Array< wchar_t > ToWCharArray(size_type i=0, size_type n=maxPos) const
Definition: String.h:11416
String & ToSpaceSeparated(const C &c)
Definition: String.h:10976
string_base::iterator iterator
Definition: String.h:8214
String DecodedHTMLSpecialChars() const
Definition: String.h:11094
String(const wchar_t *t, size_type i, size_type n)
Definition: String.h:8402
long long ToInt64() const
Definition: String.h:11768
String & Format(const wchar_t *fmt,...)
Definition: String.h:11216
String(std::initializer_list< char_type > l)
Definition: String.h:8356
string_base::item_type item_type
Definition: String.h:8164
String(const_c_string8 t, size_type i, size_type n)
Definition: String.h:8429
String(String &&)=default
String & ToSeparated(const C &c, const_c_string8 separator)
Definition: String.h:10909
String(std::initializer_list< char8_type > l)
Definition: String.h:8451
bool ToBool() const
void Assign(const_char8_iterator p, const_char8_iterator q)
Definition: String.h:8972
GenericVector< float > ParseListOfFloatAsVector(char separator=',', int maxCount=int_max) const
bool TryToUInt64(unsigned long long &value) const noexcept
Definition: String.h:11880
float ToFloat() const
String(Complex< double > &x)
Definition: String.h:8603
String(double x)
Definition: String.h:8569
string8_base::c_string c_string8
Definition: String.h:8253
String(const_c_string8 t)
Definition: String.h:8419
void Assign(std::initializer_list< char8_type > l)
Definition: String.h:8992
void Assign(const wchar_t *t, size_type i, size_type n)
bool TryToFloat(float &value) const noexcept
void Assign(const string8_base &s)
Definition: String.h:8914
String & ToSeparated(const C &c, const String &separator, AF append)
Definition: String.h:10855
String(char_type c, size_type n)
Definition: String.h:8334
String(const string_base &s)
Definition: String.h:8281
void Assign(const_c_string8 t, size_type i, size_type n)
Definition: String.h:8952
String(bool x)
Definition: String.h:8469
String & ToSeparated(const C &c, const_c_string separator)
Definition: String.h:10871
String(char8_type c, size_type n=1)
Definition: String.h:8460
void Assign(const_iterator t, size_type i, size_type n)
Definition: String.h:8852
String(const String &)=default
string8_base::const_iterator const_char8_iterator
Definition: String.h:8268
string_base::const_iterator const_iterator
Definition: String.h:8219
IsoString ToIsoString() const
String(string_base &&s)
Definition: String.h:8295
String(int x)
Definition: String.h:8499
void Assign(const char16_t *t, size_type i, size_type n)
Definition: String.h:8878
String & AppendFormat(const_c_string8 fmt,...)
Definition: String.h:11153
void Assign(char_type c, size_type n=1)
Definition: String.h:8860
String()=default
bool TryParseISO8601DateTime(int &year, int &month, int &day, double &dayf, double &tz) const noexcept
bool TryToUInt(unsigned &value, int base) const noexcept
static Array< uint32 > UTF16ToUTF32(const_c_string string, size_type i=0, size_type n=maxPos)
String & Format(const_c_string8 fmt,...)
Definition: String.h:11131
String(const_iterator t)
Definition: String.h:8317
String(long double x)
Definition: String.h:8579
String & ToNullSeparated(const C &c)
Definition: String.h:11021
String(Complex< float > &x)
Definition: String.h:8594
String(const string8_base &s)
Definition: String.h:8309
Array< float > ParseListOfFloat(char separator=',', size_type maxCount=~size_type(0)) const
void Assign(char16_t c, size_type n=1)
Definition: String.h:8886
bool TryToInt64(long long &value) const noexcept
Definition: String.h:11797
long ToInt(int base) const
bool TryToUInt(unsigned &value) const noexcept
Definition: String.h:11712
bool TryToInt(int &value) const noexcept
Definition: String.h:11615
void Assign(const_iterator t)
Definition: String.h:8822
void Assign(char8_type c, size_type n=1)
Definition: String.h:9000
unsigned long ToUInt() const
Definition: String.h:11684
String & ToSeparated(const C &c, const_c_string separator, AF append)
Definition: String.h:10893
string8_base::char_type char8_type
Definition: String.h:8243
void Assign(const char16_t *t)
Definition: String.h:8868
String(const_iterator i, const_iterator j)
Definition: String.h:8343
String & ToHyphenated(const C &c)
Definition: String.h:11035
void Assign(wchar_t c, size_type n=1)
Definition: String.h:8906
string8_base::iterator char8_iterator
Definition: String.h:8263
long long ToInt64(int base) const
string_base::const_item_type const_item_type
Definition: String.h:8169
static String Random(size_type n, RandomizationOptions options=RandomizationOption::Default)
void Assign(const wchar_t *t)
String(const char16_t *t, size_type i, size_type n)
Definition: String.h:8375
int VFormat(const wchar_t *fmt, va_list paramList)
String & ToCommaSeparated(const C &c)
Definition: String.h:10946
int VFormat(const_c_string8 fmt, va_list paramList)
Definition: String.h:11176
static String ToISO8601DateTime(int year, int month, int day, double dayf, double tz=0, const ISO8601ConversionOptions &options=ISO8601ConversionOptions())
String(unsigned long long x)
Definition: String.h:8549
String & ToColonSeparated(const C &c)
Definition: String.h:10961
String(const char16_t *t)
Definition: String.h:8365
void Assign(const_c_string8 t)
Definition: String.h:8933
string_base::c_string c_string
Definition: String.h:8194
string_base::const_c_string const_c_string
Definition: String.h:8199
String(char16_t c, size_type n)
Definition: String.h:8383
void Assign(const String &s)
Definition: String.h:8805
bool TryToUInt64(unsigned long long &value, int base) const noexcept
GenericString< char16_type, CharTraits, PCL_STRING_ALLOCATOR > string_base
Definition: String.h:8154
bool TryToDouble(double &value) const noexcept
String EncodedHTMLSpecialChars() const
Definition: String.h:11064
unsigned long long ToUInt64(int base) const
String(const_iterator t, size_type i, size_type n)
Definition: String.h:8326
String(unsigned int x)
Definition: String.h:8509
String(wchar_t c, size_type n)
Definition: String.h:8410
String & ToSeparated(const C &c, const_c_string8 separator, AF append)
Definition: String.h:10931
int AppendVFormat(const wchar_t *fmt, va_list paramList)
String & ToEncodedHTMLSpecialChars()
String & ToTabSeparated(const C &c)
Definition: String.h:10991
String & ToDecodedHTMLSpecialChars()
String(unsigned long x)
Definition: String.h:8529
String(short x)
Definition: String.h:8479
IsoString ToUTF8(size_type i=0, size_type n=maxPos) const
Definition: String.h:11357
unsigned long long ToUInt64() const
Definition: String.h:11851
static String UTF8ToUTF16(const_c_string8 string, size_type i=0, size_type n=maxPos)
Array< double > ParseListOfDouble(char separator=',', size_type maxCount=~size_type(0)) const
double ToDouble() const
bool TryToBool(bool &value) const noexcept
static String UUID()
String(long long x)
Definition: String.h:8539
unsigned long ToUInt(int base) const
IsoString To7BitASCII() const
Array< T, A > & operator<<(Array< T, A > &x, const V &v)
Definition: Array.h:2313
bool operator==(const Array< T, A > &x1, const Array< T, A > &x2) noexcept
Definition: Array.h:2285
bool operator<(const Array< T, A > &x1, const Array< T, A > &x2) noexcept
Definition: Array.h:2296
Complex< T1 > operator*(const Complex< T1 > &c1, const Complex< T2 > &c2) noexcept
Definition: Complex.h:562
Complex< T1 > operator+(const Complex< T1 > &c1, const Complex< T2 > &c2) noexcept
Definition: Complex.h:478
T Abs(const Complex< T > &c) noexcept
Definition: Complex.h:443
constexpr bool operator<=(const T1 &a, const T2 &b)
Definition: Relational.h:96
constexpr bool operator>(const T1 &a, const T2 &b)
Definition: Relational.h:106
constexpr bool operator>=(const T1 &a, const T2 &b)
Definition: Relational.h:117
uint64 Hash64(const void *data, size_type size, uint64 seed=0) noexcept
Definition: Math.h:4830
uint32 Hash32(const void *data, size_type size, uint32 seed=0) noexcept
Definition: Math.h:4984
void Swap(GenericPoint< T > &p1, GenericPoint< T > &p2) noexcept
Definition: Point.h:1459
unsigned long long uint64
Definition: Defs.h:682
unsigned char uint8
Definition: Defs.h:642
unsigned int uint32
Definition: Defs.h:666
double SexagesimalToDouble(const IsoString &separator=':') const
Definition: String.h:7302
bool TryParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const String &separator=':') const noexcept
bool TrySexagesimalToDouble(double &value, const Array< char_type > &separators) const noexcept
Definition: String.h:7389
static IsoString ToSexagesimal(double d, const SexagesimalConversionOptions &options=SexagesimalConversionOptions())
Definition: String.h:7532
void ParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const Array< char_type > &separators) const
double SexagesimalToDouble(const Array< char_type > &separators) const
Definition: String.h:7332
bool TryParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const IsoString &separator=':') const noexcept
double SexagesimalToDouble(const String &separator=':') const
Definition: String.h:12035
void ParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const Array< char_type > &separators) const
bool TrySexagesimalToDouble(double &value, const Array< char_type > &separators) const noexcept
Definition: String.h:12102
bool TryParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const Array< char_type > &separators) const noexcept
static IsoString ToSexagesimal(int sign, double s1, double s2, double s3, const SexagesimalConversionOptions &options=SexagesimalConversionOptions())
double SexagesimalToDouble(const Array< char_type > &separators) const
Definition: String.h:12055
bool TrySexagesimalToDouble(double &value, const String &separator=':') const noexcept
Definition: String.h:12079
void ParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const String &separator=':') const
bool TrySexagesimalToDouble(double &value, const IsoString &separator=':') const noexcept
Definition: String.h:7356
void ParseSexagesimal(int &sign, int &s1, int &s2, double &s3, const IsoString &separator=':') const
ptrdiff_t distance_type
Definition: Defs.h:615
size_t size_type
Definition: Defs.h:609
void Sort(BI i, BI j)
Definition: Sort.h:520
constexpr const T & Min(const T &a, const T &b) noexcept
Definition: Utility.h:90
int Compare(FI1 i1, FI1 j1, FI2 i2, FI2 j2) noexcept
Definition: Utility.h:639
constexpr const T & Max(const T &a, const T &b) noexcept
Definition: Utility.h:119
PCL root namespace.
Definition: AbstractImage.h:77
A set of options specific for string representations of angles.
Definition: String.h:257
constexpr AngleConversionOptions(unsigned precision=2, unsigned width=3, char padding=' ')
Definition: String.h:265
A set of options specific for string representations of declination angles.
Definition: String.h:375
constexpr DecConversionOptions(unsigned precision=2, unsigned width=3, char padding=' ')
Definition: String.h:384
Formatting options for string representations of dates and times in ISO 8601 format....
Definition: String.h:459
constexpr ISO8601ConversionOptionsNoTimeZone(unsigned timeItems_=3, unsigned precision_=3)
Definition: String.h:463
Formatting options for string representations of dates and times in ISO 8601 format.
Definition: String.h:399
ISO8601ConversionOptions(const ISO8601ConversionOptions &)=default
constexpr ISO8601ConversionOptions(unsigned timeItems_=3, unsigned precision_=3, bool timeZone_=true, bool zuluTime_=true)
Definition: String.h:429
A set of options specific for string representations of latitude angles.
Definition: String.h:346
constexpr LatitudeConversionOptions(unsigned precision=2, unsigned width=3, char padding=' ')
Definition: String.h:355
A set of options specific for string representations of longitude angles.
Definition: String.h:286
constexpr LongitudeConversionOptions(unsigned precision=2, unsigned width=4, char padding=' ')
Definition: String.h:295
A set of options specific for string representations of right ascensions.
Definition: String.h:317
constexpr RAConversionOptions(unsigned precision=3, unsigned width=2, char padding=' ')
Definition: String.h:325
Formatting options for string sexagesimal representations.
Definition: String.h:170
constexpr SexagesimalConversionOptions(unsigned items_=3, unsigned precision_=2, bool sign_=false, unsigned width_=0, char separator_=':', char padding_=' ')
Definition: String.h:216
SexagesimalConversionOptions(const SexagesimalConversionOptions &)=default