1 ///////////////////////////////////////////////////////////////////////////// 
   3 // Purpose:     wxString class 
   4 // Author:      Vadim Zeitlin 
   8 // Copyright:   (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr> 
   9 // Licence:     wxWindows license 
  10 ///////////////////////////////////////////////////////////////////////////// 
  13   #pragma implementation "string.h" 
  18  *  1) all empty strings use g_strEmpty, nRefs = -1 (set in Init()) 
  19  *  2) AllocBuffer() sets nRefs to 1, Lock() increments it by one 
  20  *  3) Unlock() decrements nRefs and frees memory if it goes to 0 
  23 // =========================================================================== 
  24 // headers, declarations, constants 
  25 // =========================================================================== 
  27 // For compilers that support precompilation, includes "wx.h". 
  28 #include "wx/wxprec.h" 
  36   #include "wx/string.h" 
  38   #include "wx/thread.h" 
  49 // allocating extra space for each string consumes more memory but speeds up 
  50 // the concatenation operations (nLen is the current string's length) 
  51 // NB: EXTRA_ALLOC must be >= 0! 
  52 #define EXTRA_ALLOC       (19 - nLen % 16) 
  54 // --------------------------------------------------------------------------- 
  55 // static class variables definition 
  56 // --------------------------------------------------------------------------- 
  58 #if defined(__VISAGECPP__) && __IBMCPP__ >= 400 
  59 // must define this static for VA or else you get multiply defined symbols 
  61 const unsigned int wxSTRING_MAXLEN 
= UINT_MAX 
- 100; 
  64 #ifdef  wxSTD_STRING_COMPATIBILITY 
  65   const size_t wxString::npos 
= wxSTRING_MAXLEN
; 
  66 #endif // wxSTD_STRING_COMPATIBILITY 
  68 // ---------------------------------------------------------------------------- 
  70 // ---------------------------------------------------------------------------- 
  72 // for an empty string, GetStringData() will return this address: this 
  73 // structure has the same layout as wxStringData and it's data() method will 
  74 // return the empty string (dummy pointer) 
  79 } g_strEmpty 
= { {-1, 0, 0}, wxT('\0') }; 
  81 // empty C style string: points to 'string data' byte of g_strEmpty 
  82 extern const wxChar WXDLLEXPORT 
*wxEmptyString 
= &g_strEmpty
.dummy
; 
  84 // ---------------------------------------------------------------------------- 
  86 // ---------------------------------------------------------------------------- 
  88 #if defined(wxSTD_STRING_COMPATIBILITY) && wxUSE_STD_IOSTREAM 
  90 // MS Visual C++ version 5.0 provides the new STL headers as well as the old 
  93 // ATTN: you can _not_ use both of these in the same program! 
  95 wxSTD istream
& operator>>(wxSTD istream
& is
, wxString
& WXUNUSED(str
)) 
 100     streambuf 
*sb 
= is
.rdbuf(); 
 103       int ch 
= sb
->sbumpc (); 
 105         is
.setstate(ios::eofbit
); 
 108       else if ( isspace(ch
) ) { 
 120   if ( str
.length() == 0 ) 
 121     is
.setstate(ios::failbit
); 
 126 wxSTD ostream
& operator<<(wxSTD ostream
& os
, const wxString
& str
) 
 132 #endif  //std::string compatibility 
 134 // ---------------------------------------------------------------------------- 
 136 // ---------------------------------------------------------------------------- 
 138 // this small class is used to gather statistics for performance tuning 
 139 //#define WXSTRING_STATISTICS 
 140 #ifdef  WXSTRING_STATISTICS 
 144     Averager(const wxChar 
*sz
) { m_sz 
= sz
; m_nTotal 
= m_nCount 
= 0; } 
 146    { wxPrintf("wxString: average %s = %f\n", m_sz
, ((float)m_nTotal
)/m_nCount
); } 
 148     void Add(size_t n
) { m_nTotal 
+= n
; m_nCount
++; } 
 151     size_t m_nCount
, m_nTotal
; 
 153   } g_averageLength("allocation size"), 
 154     g_averageSummandLength("summand length"), 
 155     g_averageConcatHit("hit probability in concat"), 
 156     g_averageInitialLength("initial string length"); 
 158   #define STATISTICS_ADD(av, val) g_average##av.Add(val) 
 160   #define STATISTICS_ADD(av, val) 
 161 #endif // WXSTRING_STATISTICS 
 163 // =========================================================================== 
 164 // wxString class core 
 165 // =========================================================================== 
 167 // --------------------------------------------------------------------------- 
 169 // --------------------------------------------------------------------------- 
 171 // constructs string of <nLength> copies of character <ch> 
 172 wxString::wxString(wxChar ch
, size_t nLength
) 
 177     if ( !AllocBuffer(nLength
) ) { 
 178       wxFAIL_MSG( _T("out of memory in wxString::wxString") ); 
 183     // memset only works on chars 
 184     for ( size_t n 
= 0; n 
< nLength
; n
++ ) 
 187     memset(m_pchData
, ch
, nLength
); 
 192 // takes nLength elements of psz starting at nPos 
 193 void wxString::InitWith(const wxChar 
*psz
, size_t nPos
, size_t nLength
) 
 197   // if the length is not given, assume the string to be NUL terminated 
 198   if ( nLength 
== wxSTRING_MAXLEN 
) { 
 199     wxASSERT_MSG( nPos 
<= wxStrlen(psz
), _T("index out of bounds") ); 
 201     nLength 
= wxStrlen(psz 
+ nPos
); 
 204   STATISTICS_ADD(InitialLength
, nLength
); 
 207     // trailing '\0' is written in AllocBuffer() 
 208     if ( !AllocBuffer(nLength
) ) { 
 209       wxFAIL_MSG( _T("out of memory in wxString::InitWith") ); 
 212     memcpy(m_pchData
, psz 
+ nPos
, nLength
*sizeof(wxChar
)); 
 216 #ifdef  wxSTD_STRING_COMPATIBILITY 
 218 // poor man's iterators are "void *" pointers 
 219 wxString::wxString(const void *pStart
, const void *pEnd
) 
 221   InitWith((const wxChar 
*)pStart
, 0, 
 222            (const wxChar 
*)pEnd 
- (const wxChar 
*)pStart
); 
 225 #endif  //std::string compatibility 
 229 // from multibyte string 
 230 wxString::wxString(const char *psz
, wxMBConv
& conv
, size_t nLength
) 
 232     // first get the size of the buffer we need 
 236         // calculate the needed size ourselves or use a provide one 
 237         nLen 
= nLength 
== wxSTRING_MAXLEN 
? conv
.MB2WC(NULL
, psz
, 0) : nLength
; 
 241         // nothing to convert 
 246     if ( (nLen 
!= 0) && (nLen 
!= (size_t)-1) ) 
 248         if ( !AllocBuffer(nLen
) ) 
 250             wxFAIL_MSG( _T("out of memory in wxString::wxString") ); 
 254         // MB2WC wants the buffer size, not the string length 
 255         if ( conv
.MB2WC(m_pchData
, psz
, nLen 
+ 1) != (size_t)-1 ) 
 260         //else: the conversion failed -- leave the string empty (what else?) 
 270 wxString::wxString(const wchar_t *pwz
, wxMBConv
& conv
, size_t nLength
) 
 272     // first get the size of the buffer we need 
 276         // calculate the needed size ourselves or use a provide one 
 277         nLen 
= nLength 
== wxSTRING_MAXLEN 
? conv
.WC2MB(NULL
, pwz
, 0) : nLength
; 
 281         // nothing to convert 
 286     if ( (nLen 
!= 0) && (nLen 
!= (size_t)-1) ) 
 288         if ( !AllocBuffer(nLen
) ) 
 290             wxFAIL_MSG( _T("out of memory in wxString::wxString") ); 
 294         // WC2MB wants the buffer size, not the string length 
 295         if ( conv
.WC2MB(m_pchData
, pwz
, nLen 
+ 1) != (size_t)-1 ) 
 300         //else: the conversion failed -- leave the string empty (what else?) 
 305 #endif // wxUSE_WCHAR_T 
 307 #endif // Unicode/ANSI 
 309 // --------------------------------------------------------------------------- 
 311 // --------------------------------------------------------------------------- 
 313 // allocates memory needed to store a C string of length nLen 
 314 bool wxString::AllocBuffer(size_t nLen
) 
 316   // allocating 0 sized buffer doesn't make sense, all empty strings should 
 318   wxASSERT( nLen 
>  0 ); 
 320   // make sure that we don't overflow 
 321   wxASSERT( nLen 
< (INT_MAX 
/ sizeof(wxChar
)) - 
 322                    (sizeof(wxStringData
) + EXTRA_ALLOC 
+ 1) ); 
 324   STATISTICS_ADD(Length
, nLen
); 
 327   // 1) one extra character for '\0' termination 
 328   // 2) sizeof(wxStringData) for housekeeping info 
 329   wxStringData
* pData 
= (wxStringData
*) 
 330     malloc(sizeof(wxStringData
) + (nLen 
+ EXTRA_ALLOC 
+ 1)*sizeof(wxChar
)); 
 332   if ( pData 
== NULL 
) { 
 333     // allocation failures are handled by the caller 
 338   pData
->nDataLength  
= nLen
; 
 339   pData
->nAllocLength 
= nLen 
+ EXTRA_ALLOC
; 
 340   m_pchData           
= pData
->data();  // data starts after wxStringData 
 341   m_pchData
[nLen
]     = wxT('\0'); 
 345 // must be called before changing this string 
 346 bool wxString::CopyBeforeWrite() 
 348   wxStringData
* pData 
= GetStringData(); 
 350   if ( pData
->IsShared() ) { 
 351     pData
->Unlock();                // memory not freed because shared 
 352     size_t nLen 
= pData
->nDataLength
; 
 353     if ( !AllocBuffer(nLen
) ) { 
 354       // allocation failures are handled by the caller 
 357     memcpy(m_pchData
, pData
->data(), nLen
*sizeof(wxChar
)); 
 360   wxASSERT( !GetStringData()->IsShared() );  // we must be the only owner 
 365 // must be called before replacing contents of this string 
 366 bool wxString::AllocBeforeWrite(size_t nLen
) 
 368   wxASSERT( nLen 
!= 0 );  // doesn't make any sense 
 370   // must not share string and must have enough space 
 371   wxStringData
* pData 
= GetStringData(); 
 372   if ( pData
->IsShared() || pData
->IsEmpty() ) { 
 373     // can't work with old buffer, get new one 
 375     if ( !AllocBuffer(nLen
) ) { 
 376       // allocation failures are handled by the caller 
 381     if ( nLen 
> pData
->nAllocLength 
) { 
 382       // realloc the buffer instead of calling malloc() again, this is more 
 384       STATISTICS_ADD(Length
, nLen
); 
 388       pData 
= (wxStringData
*) 
 389           realloc(pData
, sizeof(wxStringData
) + (nLen 
+ 1)*sizeof(wxChar
)); 
 391       if ( pData 
== NULL 
) { 
 392         // allocation failures are handled by the caller 
 393         // keep previous data since reallocation failed 
 397       pData
->nAllocLength 
= nLen
; 
 398       m_pchData 
= pData
->data(); 
 401     // now we have enough space, just update the string length 
 402     pData
->nDataLength 
= nLen
; 
 405   wxASSERT( !GetStringData()->IsShared() );  // we must be the only owner 
 410 // allocate enough memory for nLen characters 
 411 bool wxString::Alloc(size_t nLen
) 
 413   wxStringData 
*pData 
= GetStringData(); 
 414   if ( pData
->nAllocLength 
<= nLen 
) { 
 415     if ( pData
->IsEmpty() ) { 
 418       wxStringData
* pData 
= (wxStringData
*) 
 419           malloc(sizeof(wxStringData
) + (nLen 
+ 1)*sizeof(wxChar
)); 
 421       if ( pData 
== NULL 
) { 
 422         // allocation failure handled by caller 
 427       pData
->nDataLength 
= 0; 
 428       pData
->nAllocLength 
= nLen
; 
 429       m_pchData 
= pData
->data();  // data starts after wxStringData 
 430       m_pchData
[0u] = wxT('\0'); 
 432     else if ( pData
->IsShared() ) { 
 433       pData
->Unlock();                // memory not freed because shared 
 434       size_t nOldLen 
= pData
->nDataLength
; 
 435       if ( !AllocBuffer(nLen
) ) { 
 436         // allocation failure handled by caller 
 439       memcpy(m_pchData
, pData
->data(), nOldLen
*sizeof(wxChar
)); 
 444       pData 
= (wxStringData 
*) 
 445         realloc(pData
, sizeof(wxStringData
) + (nLen 
+ 1)*sizeof(wxChar
)); 
 447       if ( pData 
== NULL 
) { 
 448         // allocation failure handled by caller 
 449         // keep previous data since reallocation failed 
 453       // it's not important if the pointer changed or not (the check for this 
 454       // is not faster than assigning to m_pchData in all cases) 
 455       pData
->nAllocLength 
= nLen
; 
 456       m_pchData 
= pData
->data(); 
 459   //else: we've already got enough 
 463 // shrink to minimal size (releasing extra memory) 
 464 bool wxString::Shrink() 
 466   wxStringData 
*pData 
= GetStringData(); 
 468   size_t nLen 
= pData
->nDataLength
; 
 469   void *p 
= realloc(pData
, sizeof(wxStringData
) + (nLen 
+ 1)*sizeof(wxChar
)); 
 472       wxFAIL_MSG( _T("out of memory reallocating wxString data") ); 
 473       // keep previous data since reallocation failed 
 479       // contrary to what one might believe, some realloc() implementation do 
 480       // move the memory block even when its size is reduced 
 481       pData 
= (wxStringData 
*)p
; 
 483       m_pchData 
= pData
->data(); 
 486   pData
->nAllocLength 
= nLen
; 
 491 // get the pointer to writable buffer of (at least) nLen bytes 
 492 wxChar 
*wxString::GetWriteBuf(size_t nLen
) 
 494   if ( !AllocBeforeWrite(nLen
) ) { 
 495     // allocation failure handled by caller 
 499   wxASSERT( GetStringData()->nRefs 
== 1 ); 
 500   GetStringData()->Validate(FALSE
); 
 505 // put string back in a reasonable state after GetWriteBuf 
 506 void wxString::UngetWriteBuf() 
 508   GetStringData()->nDataLength 
= wxStrlen(m_pchData
); 
 509   GetStringData()->Validate(TRUE
); 
 512 void wxString::UngetWriteBuf(size_t nLen
) 
 514   GetStringData()->nDataLength 
= nLen
; 
 515   GetStringData()->Validate(TRUE
); 
 518 // --------------------------------------------------------------------------- 
 520 // --------------------------------------------------------------------------- 
 522 // all functions are inline in string.h 
 524 // --------------------------------------------------------------------------- 
 525 // assignment operators 
 526 // --------------------------------------------------------------------------- 
 528 // helper function: does real copy 
 529 bool wxString::AssignCopy(size_t nSrcLen
, const wxChar 
*pszSrcData
) 
 531   if ( nSrcLen 
== 0 ) { 
 535     if ( !AllocBeforeWrite(nSrcLen
) ) { 
 536       // allocation failure handled by caller 
 539     memcpy(m_pchData
, pszSrcData
, nSrcLen
*sizeof(wxChar
)); 
 540     GetStringData()->nDataLength 
= nSrcLen
; 
 541     m_pchData
[nSrcLen
] = wxT('\0'); 
 546 // assigns one string to another 
 547 wxString
& wxString::operator=(const wxString
& stringSrc
) 
 549   wxASSERT( stringSrc
.GetStringData()->IsValid() ); 
 551   // don't copy string over itself 
 552   if ( m_pchData 
!= stringSrc
.m_pchData 
) { 
 553     if ( stringSrc
.GetStringData()->IsEmpty() ) { 
 558       GetStringData()->Unlock(); 
 559       m_pchData 
= stringSrc
.m_pchData
; 
 560       GetStringData()->Lock(); 
 567 // assigns a single character 
 568 wxString
& wxString::operator=(wxChar ch
) 
 570   if ( !AssignCopy(1, &ch
) ) { 
 571     wxFAIL_MSG( _T("out of memory in wxString::operator=(wxChar)") ); 
 578 wxString
& wxString::operator=(const wxChar 
*psz
) 
 580   if ( !AssignCopy(wxStrlen(psz
), psz
) ) { 
 581     wxFAIL_MSG( _T("out of memory in wxString::operator=(const wxChar *)") ); 
 588 // same as 'signed char' variant 
 589 wxString
& wxString::operator=(const unsigned char* psz
) 
 591   *this = (const char *)psz
; 
 596 wxString
& wxString::operator=(const wchar_t *pwz
) 
 606 // --------------------------------------------------------------------------- 
 607 // string concatenation 
 608 // --------------------------------------------------------------------------- 
 610 // add something to this string 
 611 bool wxString::ConcatSelf(int nSrcLen
, const wxChar 
*pszSrcData
) 
 613   STATISTICS_ADD(SummandLength
, nSrcLen
); 
 615   // concatenating an empty string is a NOP 
 617     wxStringData 
*pData 
= GetStringData(); 
 618     size_t nLen 
= pData
->nDataLength
; 
 619     size_t nNewLen 
= nLen 
+ nSrcLen
; 
 621     // alloc new buffer if current is too small 
 622     if ( pData
->IsShared() ) { 
 623       STATISTICS_ADD(ConcatHit
, 0); 
 625       // we have to allocate another buffer 
 626       wxStringData
* pOldData 
= GetStringData(); 
 627       if ( !AllocBuffer(nNewLen
) ) { 
 628           // allocation failure handled by caller 
 631       memcpy(m_pchData
, pOldData
->data(), nLen
*sizeof(wxChar
)); 
 634     else if ( nNewLen 
> pData
->nAllocLength 
) { 
 635       STATISTICS_ADD(ConcatHit
, 0); 
 637       // we have to grow the buffer 
 638       if ( !Alloc(nNewLen
) ) { 
 639           // allocation failure handled by caller 
 644       STATISTICS_ADD(ConcatHit
, 1); 
 646       // the buffer is already big enough 
 649     // should be enough space 
 650     wxASSERT( nNewLen 
<= GetStringData()->nAllocLength 
); 
 652     // fast concatenation - all is done in our buffer 
 653     memcpy(m_pchData 
+ nLen
, pszSrcData
, nSrcLen
*sizeof(wxChar
)); 
 655     m_pchData
[nNewLen
] = wxT('\0');          // put terminating '\0' 
 656     GetStringData()->nDataLength 
= nNewLen
; // and fix the length 
 658   //else: the string to append was empty 
 663  * concatenation functions come in 5 flavours: 
 665  *  char   + string      and      string + char 
 666  *  C str  + string      and      string + C str 
 669 wxString 
operator+(const wxString
& str1
, const wxString
& str2
) 
 671   wxASSERT( str1
.GetStringData()->IsValid() ); 
 672   wxASSERT( str2
.GetStringData()->IsValid() ); 
 680 wxString 
operator+(const wxString
& str
, wxChar ch
) 
 682   wxASSERT( str
.GetStringData()->IsValid() ); 
 690 wxString 
operator+(wxChar ch
, const wxString
& str
) 
 692   wxASSERT( str
.GetStringData()->IsValid() ); 
 700 wxString 
operator+(const wxString
& str
, const wxChar 
*psz
) 
 702   wxASSERT( str
.GetStringData()->IsValid() ); 
 705   if ( !s
.Alloc(wxStrlen(psz
) + str
.Len()) ) { 
 706     wxFAIL_MSG( _T("out of memory in wxString::operator+") ); 
 714 wxString 
operator+(const wxChar 
*psz
, const wxString
& str
) 
 716   wxASSERT( str
.GetStringData()->IsValid() ); 
 719   if ( !s
.Alloc(wxStrlen(psz
) + str
.Len()) ) { 
 720     wxFAIL_MSG( _T("out of memory in wxString::operator+") ); 
 728 // =========================================================================== 
 729 // other common string functions 
 730 // =========================================================================== 
 734 wxString 
wxString::FromAscii(const char *ascii
) 
 737        return wxEmptyString
; 
 739     size_t len 
= strlen( ascii 
); 
 744         wxStringBuffer 
buf(res
, len
); 
 750            if ( (*dest
++ = (wchar_t)(unsigned char)*ascii
++) == L
'\0' ) 
 758 wxString 
wxString::FromAscii(const char ascii
) 
 760     // What do we do with '\0' ? 
 763     res 
+= (wchar_t)(unsigned char) ascii
; 
 768 const wxCharBuffer 
wxString::ToAscii() const 
 770     // this will allocate enough space for the terminating NUL too 
 771     wxCharBuffer 
buffer(length()); 
 773     signed char *dest 
= (signed char *)buffer
.data(); 
 775     const wchar_t *pwc 
= c_str(); 
 778         *dest
++ = *pwc 
> SCHAR_MAX 
? '_' : *pwc
; 
 780         // the output string can't have embedded NULs anyhow, so we can safely 
 781         // stop at first of them even if we do have any 
 791 // --------------------------------------------------------------------------- 
 792 // simple sub-string extraction 
 793 // --------------------------------------------------------------------------- 
 795 // helper function: clone the data attached to this string 
 796 bool wxString::AllocCopy(wxString
& dest
, int nCopyLen
, int nCopyIndex
) const 
 798   if ( nCopyLen 
== 0 ) { 
 802     if ( !dest
.AllocBuffer(nCopyLen
) ) { 
 803       // allocation failure handled by caller 
 806     memcpy(dest
.m_pchData
, m_pchData 
+ nCopyIndex
, nCopyLen
*sizeof(wxChar
)); 
 811 // extract string of length nCount starting at nFirst 
 812 wxString 
wxString::Mid(size_t nFirst
, size_t nCount
) const 
 814   wxStringData 
*pData 
= GetStringData(); 
 815   size_t nLen 
= pData
->nDataLength
; 
 817   // default value of nCount is wxSTRING_MAXLEN and means "till the end" 
 818   if ( nCount 
== wxSTRING_MAXLEN 
) 
 820     nCount 
= nLen 
- nFirst
; 
 823   // out-of-bounds requests return sensible things 
 824   if ( nFirst 
+ nCount 
> nLen 
) 
 826     nCount 
= nLen 
- nFirst
; 
 831     // AllocCopy() will return empty string 
 836   if ( !AllocCopy(dest
, nCount
, nFirst
) ) { 
 837       wxFAIL_MSG( _T("out of memory in wxString::Mid") ); 
 843 // check that the tring starts with prefix and return the rest of the string 
 844 // in the provided pointer if it is not NULL, otherwise return FALSE 
 845 bool wxString::StartsWith(const wxChar 
*prefix
, wxString 
*rest
) const 
 847     wxASSERT_MSG( prefix
, _T("invalid parameter in wxString::StartsWith") ); 
 849     // first check if the beginning of the string matches the prefix: note 
 850     // that we don't have to check that we don't run out of this string as 
 851     // when we reach the terminating NUL, either prefix string ends too (and 
 852     // then it's ok) or we break out of the loop because there is no match 
 853     const wxChar 
*p 
= c_str(); 
 856         if ( *prefix
++ != *p
++ ) 
 865         // put the rest of the string into provided pointer 
 872 // extract nCount last (rightmost) characters 
 873 wxString 
wxString::Right(size_t nCount
) const 
 875   if ( nCount 
> (size_t)GetStringData()->nDataLength 
) 
 876     nCount 
= GetStringData()->nDataLength
; 
 879   if ( !AllocCopy(dest
, nCount
, GetStringData()->nDataLength 
- nCount
) ) { 
 880     wxFAIL_MSG( _T("out of memory in wxString::Right") ); 
 885 // get all characters after the last occurence of ch 
 886 // (returns the whole string if ch not found) 
 887 wxString 
wxString::AfterLast(wxChar ch
) const 
 890   int iPos 
= Find(ch
, TRUE
); 
 891   if ( iPos 
== wxNOT_FOUND 
) 
 894     str 
= c_str() + iPos 
+ 1; 
 899 // extract nCount first (leftmost) characters 
 900 wxString 
wxString::Left(size_t nCount
) const 
 902   if ( nCount 
> (size_t)GetStringData()->nDataLength 
) 
 903     nCount 
= GetStringData()->nDataLength
; 
 906   if ( !AllocCopy(dest
, nCount
, 0) ) { 
 907     wxFAIL_MSG( _T("out of memory in wxString::Left") ); 
 912 // get all characters before the first occurence of ch 
 913 // (returns the whole string if ch not found) 
 914 wxString 
wxString::BeforeFirst(wxChar ch
) const 
 917   for ( const wxChar 
*pc 
= m_pchData
; *pc 
!= wxT('\0') && *pc 
!= ch
; pc
++ ) 
 923 /// get all characters before the last occurence of ch 
 924 /// (returns empty string if ch not found) 
 925 wxString 
wxString::BeforeLast(wxChar ch
) const 
 928   int iPos 
= Find(ch
, TRUE
); 
 929   if ( iPos 
!= wxNOT_FOUND 
&& iPos 
!= 0 ) 
 930     str 
= wxString(c_str(), iPos
); 
 935 /// get all characters after the first occurence of ch 
 936 /// (returns empty string if ch not found) 
 937 wxString 
wxString::AfterFirst(wxChar ch
) const 
 941   if ( iPos 
!= wxNOT_FOUND 
) 
 942     str 
= c_str() + iPos 
+ 1; 
 947 // replace first (or all) occurences of some substring with another one 
 948 size_t wxString::Replace(const wxChar 
*szOld
, const wxChar 
*szNew
, bool bReplaceAll
) 
 950   size_t uiCount 
= 0;   // count of replacements made 
 952   size_t uiOldLen 
= wxStrlen(szOld
); 
 955   const wxChar 
*pCurrent 
= m_pchData
; 
 956   const wxChar 
*pSubstr
; 
 957   while ( *pCurrent 
!= wxT('\0') ) { 
 958     pSubstr 
= wxStrstr(pCurrent
, szOld
); 
 959     if ( pSubstr 
== NULL 
) { 
 960       // strTemp is unused if no replacements were made, so avoid the copy 
 964       strTemp 
+= pCurrent
;    // copy the rest 
 965       break;                  // exit the loop 
 968       // take chars before match 
 969       if ( !strTemp
.ConcatSelf(pSubstr 
- pCurrent
, pCurrent
) ) { 
 970         wxFAIL_MSG( _T("out of memory in wxString::Replace") ); 
 974       pCurrent 
= pSubstr 
+ uiOldLen
;  // restart after match 
 979       if ( !bReplaceAll 
) { 
 980         strTemp 
+= pCurrent
;    // copy the rest 
 981         break;                  // exit the loop 
 986   // only done if there were replacements, otherwise would have returned above 
 992 bool wxString::IsAscii() const 
 994   const wxChar 
*s 
= (const wxChar
*) *this; 
 996     if(!isascii(*s
)) return(FALSE
); 
1002 bool wxString::IsWord() const 
1004   const wxChar 
*s 
= (const wxChar
*) *this; 
1006     if(!wxIsalpha(*s
)) return(FALSE
); 
1012 bool wxString::IsNumber() const 
1014   const wxChar 
*s 
= (const wxChar
*) *this; 
1016      if ((s
[0] == '-') || (s
[0] == '+')) s
++; 
1018     if(!wxIsdigit(*s
)) return(FALSE
); 
1024 wxString 
wxString::Strip(stripType w
) const 
1027     if ( w 
& leading 
) s
.Trim(FALSE
); 
1028     if ( w 
& trailing 
) s
.Trim(TRUE
); 
1032 // --------------------------------------------------------------------------- 
1034 // --------------------------------------------------------------------------- 
1036 wxString
& wxString::MakeUpper() 
1038   if ( !CopyBeforeWrite() ) { 
1039     wxFAIL_MSG( _T("out of memory in wxString::MakeUpper") ); 
1043   for ( wxChar 
*p 
= m_pchData
; *p
; p
++ ) 
1044     *p 
= (wxChar
)wxToupper(*p
); 
1049 wxString
& wxString::MakeLower() 
1051   if ( !CopyBeforeWrite() ) { 
1052     wxFAIL_MSG( _T("out of memory in wxString::MakeLower") ); 
1056   for ( wxChar 
*p 
= m_pchData
; *p
; p
++ ) 
1057     *p 
= (wxChar
)wxTolower(*p
); 
1062 // --------------------------------------------------------------------------- 
1063 // trimming and padding 
1064 // --------------------------------------------------------------------------- 
1066 // some compilers (VC++ 6.0 not to name them) return TRUE for a call to 
1067 // isspace('ê') in the C locale which seems to be broken to me, but we have to 
1068 // live with this by checking that the character is a 7 bit one - even if this 
1069 // may fail to detect some spaces (I don't know if Unicode doesn't have 
1070 // space-like symbols somewhere except in the first 128 chars), it is arguably 
1071 // still better than trimming away accented letters 
1072 inline int wxSafeIsspace(wxChar ch
) { return (ch 
< 127) && wxIsspace(ch
); } 
1074 // trims spaces (in the sense of isspace) from left or right side 
1075 wxString
& wxString::Trim(bool bFromRight
) 
1077   // first check if we're going to modify the string at all 
1080         (bFromRight 
&& wxSafeIsspace(GetChar(Len() - 1))) || 
1081         (!bFromRight 
&& wxSafeIsspace(GetChar(0u))) 
1085     // ok, there is at least one space to trim 
1086     if ( !CopyBeforeWrite() ) { 
1087       wxFAIL_MSG( _T("out of memory in wxString::Trim") ); 
1093       // find last non-space character 
1094       wxChar 
*psz 
= m_pchData 
+ GetStringData()->nDataLength 
- 1; 
1095       while ( wxSafeIsspace(*psz
) && (psz 
>= m_pchData
) ) 
1098       // truncate at trailing space start 
1100       GetStringData()->nDataLength 
= psz 
- m_pchData
; 
1104       // find first non-space character 
1105       const wxChar 
*psz 
= m_pchData
; 
1106       while ( wxSafeIsspace(*psz
) ) 
1109       // fix up data and length 
1110       int nDataLength 
= GetStringData()->nDataLength 
- (psz 
- (const wxChar
*) m_pchData
); 
1111       memmove(m_pchData
, psz
, (nDataLength 
+ 1)*sizeof(wxChar
)); 
1112       GetStringData()->nDataLength 
= nDataLength
; 
1119 // adds nCount characters chPad to the string from either side 
1120 wxString
& wxString::Pad(size_t nCount
, wxChar chPad
, bool bFromRight
) 
1122   wxString 
s(chPad
, nCount
); 
1135 // truncate the string 
1136 wxString
& wxString::Truncate(size_t uiLen
) 
1138   if ( uiLen 
< Len() ) { 
1139     if ( !CopyBeforeWrite() ) { 
1140       wxFAIL_MSG( _T("out of memory in wxString::Truncate") ); 
1144     *(m_pchData 
+ uiLen
) = wxT('\0'); 
1145     GetStringData()->nDataLength 
= uiLen
; 
1147   //else: nothing to do, string is already short enough 
1152 // --------------------------------------------------------------------------- 
1153 // finding (return wxNOT_FOUND if not found and index otherwise) 
1154 // --------------------------------------------------------------------------- 
1157 int wxString::Find(wxChar ch
, bool bFromEnd
) const 
1159   const wxChar 
*psz 
= bFromEnd 
? wxStrrchr(m_pchData
, ch
) : wxStrchr(m_pchData
, ch
); 
1161   return (psz 
== NULL
) ? wxNOT_FOUND 
: psz 
- (const wxChar
*) m_pchData
; 
1164 // find a sub-string (like strstr) 
1165 int wxString::Find(const wxChar 
*pszSub
) const 
1167   const wxChar 
*psz 
= wxStrstr(m_pchData
, pszSub
); 
1169   return (psz 
== NULL
) ? wxNOT_FOUND 
: psz 
- (const wxChar
*) m_pchData
; 
1172 // ---------------------------------------------------------------------------- 
1173 // conversion to numbers 
1174 // ---------------------------------------------------------------------------- 
1176 bool wxString::ToLong(long *val
, int base
) const 
1178     wxCHECK_MSG( val
, FALSE
, _T("NULL pointer in wxString::ToLong") ); 
1179     wxASSERT_MSG( !base 
|| (base 
> 1 && base 
<= 36), _T("invalid base") ); 
1181     const wxChar 
*start 
= c_str(); 
1183     *val 
= wxStrtol(start
, &end
, base
); 
1185     // return TRUE only if scan was stopped by the terminating NUL and if the 
1186     // string was not empty to start with 
1187     return !*end 
&& (end 
!= start
); 
1190 bool wxString::ToULong(unsigned long *val
, int base
) const 
1192     wxCHECK_MSG( val
, FALSE
, _T("NULL pointer in wxString::ToULong") ); 
1193     wxASSERT_MSG( !base 
|| (base 
> 1 && base 
<= 36), _T("invalid base") ); 
1195     const wxChar 
*start 
= c_str(); 
1197     *val 
= wxStrtoul(start
, &end
, base
); 
1199     // return TRUE only if scan was stopped by the terminating NUL and if the 
1200     // string was not empty to start with 
1201     return !*end 
&& (end 
!= start
); 
1204 bool wxString::ToDouble(double *val
) const 
1206     wxCHECK_MSG( val
, FALSE
, _T("NULL pointer in wxString::ToDouble") ); 
1208     const wxChar 
*start 
= c_str(); 
1210     *val 
= wxStrtod(start
, &end
); 
1212     // return TRUE only if scan was stopped by the terminating NUL and if the 
1213     // string was not empty to start with 
1214     return !*end 
&& (end 
!= start
); 
1217 // --------------------------------------------------------------------------- 
1219 // --------------------------------------------------------------------------- 
1222 wxString 
wxString::Format(const wxChar 
*pszFormat
, ...) 
1225     va_start(argptr
, pszFormat
); 
1228     s
.PrintfV(pszFormat
, argptr
); 
1236 wxString 
wxString::FormatV(const wxChar 
*pszFormat
, va_list argptr
) 
1239     s
.PrintfV(pszFormat
, argptr
); 
1243 int wxString::Printf(const wxChar 
*pszFormat
, ...) 
1246   va_start(argptr
, pszFormat
); 
1248   int iLen 
= PrintfV(pszFormat
, argptr
); 
1255 int wxString::PrintfV(const wxChar
* pszFormat
, va_list argptr
) 
1260         wxChar 
*buf 
= GetWriteBuf(size 
+ 1); 
1267         int len 
= wxVsnprintf(buf
, size
, pszFormat
, argptr
); 
1269         // some implementations of vsnprintf() don't NUL terminate the string 
1270         // if there is not enough space for it so always do it manually 
1271         buf
[size
] = _T('\0'); 
1277             // ok, there was enough space 
1281         // still not enough, double it again 
1285     // we could have overshot 
1291 // ---------------------------------------------------------------------------- 
1292 // misc other operations 
1293 // ---------------------------------------------------------------------------- 
1295 // returns TRUE if the string matches the pattern which may contain '*' and 
1296 // '?' metacharacters (as usual, '?' matches any character and '*' any number 
1298 bool wxString::Matches(const wxChar 
*pszMask
) const 
1300     // I disable this code as it doesn't seem to be faster (in fact, it seems 
1301     // to be much slower) than the old, hand-written code below and using it 
1302     // here requires always linking with libregex even if the user code doesn't 
1304 #if 0 // wxUSE_REGEX 
1305     // first translate the shell-like mask into a regex 
1307     pattern
.reserve(wxStrlen(pszMask
)); 
1319                 pattern 
+= _T(".*"); 
1330                 // these characters are special in a RE, quote them 
1331                 // (however note that we don't quote '[' and ']' to allow 
1332                 // using them for Unix shell like matching) 
1333                 pattern 
+= _T('\\'); 
1337                 pattern 
+= *pszMask
; 
1345     return wxRegEx(pattern
, wxRE_NOSUB 
| wxRE_EXTENDED
).Matches(c_str()); 
1346 #else // !wxUSE_REGEX 
1347   // TODO: this is, of course, awfully inefficient... 
1349   // the char currently being checked 
1350   const wxChar 
*pszTxt 
= c_str(); 
1352   // the last location where '*' matched 
1353   const wxChar 
*pszLastStarInText 
= NULL
; 
1354   const wxChar 
*pszLastStarInMask 
= NULL
; 
1357   for ( ; *pszMask 
!= wxT('\0'); pszMask
++, pszTxt
++ ) { 
1358     switch ( *pszMask 
) { 
1360         if ( *pszTxt 
== wxT('\0') ) 
1363         // pszTxt and pszMask will be incremented in the loop statement 
1369           // remember where we started to be able to backtrack later 
1370           pszLastStarInText 
= pszTxt
; 
1371           pszLastStarInMask 
= pszMask
; 
1373           // ignore special chars immediately following this one 
1374           // (should this be an error?) 
1375           while ( *pszMask 
== wxT('*') || *pszMask 
== wxT('?') ) 
1378           // if there is nothing more, match 
1379           if ( *pszMask 
== wxT('\0') ) 
1382           // are there any other metacharacters in the mask? 
1384           const wxChar 
*pEndMask 
= wxStrpbrk(pszMask
, wxT("*?")); 
1386           if ( pEndMask 
!= NULL 
) { 
1387             // we have to match the string between two metachars 
1388             uiLenMask 
= pEndMask 
- pszMask
; 
1391             // we have to match the remainder of the string 
1392             uiLenMask 
= wxStrlen(pszMask
); 
1395           wxString 
strToMatch(pszMask
, uiLenMask
); 
1396           const wxChar
* pMatch 
= wxStrstr(pszTxt
, strToMatch
); 
1397           if ( pMatch 
== NULL 
) 
1400           // -1 to compensate "++" in the loop 
1401           pszTxt 
= pMatch 
+ uiLenMask 
- 1; 
1402           pszMask 
+= uiLenMask 
- 1; 
1407         if ( *pszMask 
!= *pszTxt 
) 
1413   // match only if nothing left 
1414   if ( *pszTxt 
== wxT('\0') ) 
1417   // if we failed to match, backtrack if we can 
1418   if ( pszLastStarInText 
) { 
1419     pszTxt 
= pszLastStarInText 
+ 1; 
1420     pszMask 
= pszLastStarInMask
; 
1422     pszLastStarInText 
= NULL
; 
1424     // don't bother resetting pszLastStarInMask, it's unnecessary 
1430 #endif // wxUSE_REGEX/!wxUSE_REGEX 
1433 // Count the number of chars 
1434 int wxString::Freq(wxChar ch
) const 
1438     for (int i 
= 0; i 
< len
; i
++) 
1440         if (GetChar(i
) == ch
) 
1446 // convert to upper case, return the copy of the string 
1447 wxString 
wxString::Upper() const 
1448 { wxString 
s(*this); return s
.MakeUpper(); } 
1450 // convert to lower case, return the copy of the string 
1451 wxString 
wxString::Lower() const { wxString 
s(*this); return s
.MakeLower(); } 
1453 int wxString::sprintf(const wxChar 
*pszFormat
, ...) 
1456     va_start(argptr
, pszFormat
); 
1457     int iLen 
= PrintfV(pszFormat
, argptr
); 
1462 // --------------------------------------------------------------------------- 
1463 // standard C++ library string functions 
1464 // --------------------------------------------------------------------------- 
1466 #ifdef  wxSTD_STRING_COMPATIBILITY 
1468 void wxString::resize(size_t nSize
, wxChar ch
) 
1470     size_t len 
= length(); 
1476     else if ( nSize 
> len 
) 
1478         *this += wxString(ch
, nSize 
- len
); 
1480     //else: we have exactly the specified length, nothing to do 
1483 void wxString::swap(wxString
& str
) 
1485     // this is slightly less efficient than fiddling with m_pchData directly, 
1486     // but it is still quite efficient as we don't copy the string here because 
1487     // ref count always stays positive 
1493 wxString
& wxString::insert(size_t nPos
, const wxString
& str
) 
1495   wxASSERT( str
.GetStringData()->IsValid() ); 
1496   wxASSERT( nPos 
<= Len() ); 
1498   if ( !str
.IsEmpty() ) { 
1500     wxChar 
*pc 
= strTmp
.GetWriteBuf(Len() + str
.Len()); 
1501     wxStrncpy(pc
, c_str(), nPos
); 
1502     wxStrcpy(pc 
+ nPos
, str
); 
1503     wxStrcpy(pc 
+ nPos 
+ str
.Len(), c_str() + nPos
); 
1504     strTmp
.UngetWriteBuf(); 
1511 size_t wxString::find(const wxString
& str
, size_t nStart
) const 
1513   wxASSERT( str
.GetStringData()->IsValid() ); 
1514   wxASSERT( nStart 
<= Len() ); 
1516   const wxChar 
*p 
= wxStrstr(c_str() + nStart
, str
); 
1518   return p 
== NULL 
? npos 
: p 
- c_str(); 
1521 // VC++ 1.5 can't cope with the default argument in the header. 
1522 #if !defined(__VISUALC__) || defined(__WIN32__) 
1523 size_t wxString::find(const wxChar
* sz
, size_t nStart
, size_t n
) const 
1525   return find(wxString(sz
, n
), nStart
); 
1529 // Gives a duplicate symbol (presumably a case-insensitivity problem) 
1530 #if !defined(__BORLANDC__) 
1531 size_t wxString::find(wxChar ch
, size_t nStart
) const 
1533   wxASSERT( nStart 
<= Len() ); 
1535   const wxChar 
*p 
= wxStrchr(c_str() + nStart
, ch
); 
1537   return p 
== NULL 
? npos 
: p 
- c_str(); 
1541 size_t wxString::rfind(const wxString
& str
, size_t nStart
) const 
1543   wxASSERT( str
.GetStringData()->IsValid() ); 
1544   wxASSERT( nStart 
== npos 
|| nStart 
<= Len() ); 
1546   // TODO could be made much quicker than that 
1547   const wxChar 
*p 
= c_str() + (nStart 
== npos 
? Len() : nStart
); 
1548   while ( p 
>= c_str() + str
.Len() ) { 
1549     if ( wxStrncmp(p 
- str
.Len(), str
, str
.Len()) == 0 ) 
1550       return p 
- str
.Len() - c_str(); 
1557 // VC++ 1.5 can't cope with the default argument in the header. 
1558 #if !defined(__VISUALC__) || defined(__WIN32__) 
1559 size_t wxString::rfind(const wxChar
* sz
, size_t nStart
, size_t n
) const 
1561     return rfind(wxString(sz
, n 
== npos 
? wxSTRING_MAXLEN 
: n
), nStart
); 
1564 size_t wxString::rfind(wxChar ch
, size_t nStart
) const 
1566     if ( nStart 
== npos 
) 
1572         wxASSERT( nStart 
<= Len() ); 
1575     const wxChar 
*p 
= wxStrrchr(c_str(), ch
); 
1580     size_t result 
= p 
- c_str(); 
1581     return ( result 
> nStart 
) ? npos 
: result
; 
1585 size_t wxString::find_first_of(const wxChar
* sz
, size_t nStart
) const 
1587     const wxChar 
*start 
= c_str() + nStart
; 
1588     const wxChar 
*firstOf 
= wxStrpbrk(start
, sz
); 
1590         return firstOf 
- c_str(); 
1595 size_t wxString::find_last_of(const wxChar
* sz
, size_t nStart
) const 
1597     if ( nStart 
== npos 
) 
1603         wxASSERT( nStart 
<= Len() ); 
1606     for ( const wxChar 
*p 
= c_str() + length() - 1; p 
>= c_str(); p
-- ) 
1608         if ( wxStrchr(sz
, *p
) ) 
1615 size_t wxString::find_first_not_of(const wxChar
* sz
, size_t nStart
) const 
1617     if ( nStart 
== npos 
) 
1623         wxASSERT( nStart 
<= Len() ); 
1626     size_t nAccept 
= wxStrspn(c_str() + nStart
, sz
); 
1627     if ( nAccept 
>= length() - nStart 
) 
1633 size_t wxString::find_first_not_of(wxChar ch
, size_t nStart
) const 
1635     wxASSERT( nStart 
<= Len() ); 
1637     for ( const wxChar 
*p 
= c_str() + nStart
; *p
; p
++ ) 
1646 size_t wxString::find_last_not_of(const wxChar
* sz
, size_t nStart
) const 
1648     if ( nStart 
== npos 
) 
1654         wxASSERT( nStart 
<= Len() ); 
1657     for ( const wxChar 
*p 
= c_str() + nStart 
- 1; p 
>= c_str(); p
-- ) 
1659         if ( !wxStrchr(sz
, *p
) ) 
1666 size_t wxString::find_last_not_of(wxChar ch
, size_t nStart
) const 
1668     if ( nStart 
== npos 
) 
1674         wxASSERT( nStart 
<= Len() ); 
1677     for ( const wxChar 
*p 
= c_str() + nStart 
- 1; p 
>= c_str(); p
-- ) 
1686 wxString
& wxString::erase(size_t nStart
, size_t nLen
) 
1688   wxString 
strTmp(c_str(), nStart
); 
1689   if ( nLen 
!= npos 
) { 
1690     wxASSERT( nStart 
+ nLen 
<= Len() ); 
1692     strTmp
.append(c_str() + nStart 
+ nLen
); 
1699 wxString
& wxString::replace(size_t nStart
, size_t nLen
, const wxChar 
*sz
) 
1701   wxASSERT_MSG( nStart 
+ nLen 
<= Len(), 
1702                 _T("index out of bounds in wxString::replace") ); 
1705   strTmp
.Alloc(Len());      // micro optimisation to avoid multiple mem allocs 
1708     strTmp
.append(c_str(), nStart
); 
1709   strTmp 
<< sz 
<< c_str() + nStart 
+ nLen
; 
1715 wxString
& wxString::replace(size_t nStart
, size_t nLen
, size_t nCount
, wxChar ch
) 
1717   return replace(nStart
, nLen
, wxString(ch
, nCount
)); 
1720 wxString
& wxString::replace(size_t nStart
, size_t nLen
, 
1721                             const wxString
& str
, size_t nStart2
, size_t nLen2
) 
1723   return replace(nStart
, nLen
, str
.substr(nStart2
, nLen2
)); 
1726 wxString
& wxString::replace(size_t nStart
, size_t nLen
, 
1727                         const wxChar
* sz
, size_t nCount
) 
1729   return replace(nStart
, nLen
, wxString(sz
, nCount
)); 
1732 #endif  //std::string compatibility 
1734 // ============================================================================ 
1736 // ============================================================================ 
1738 // size increment = min(50% of current size, ARRAY_MAXSIZE_INCREMENT) 
1739 #define   ARRAY_MAXSIZE_INCREMENT       4096 
1741 #ifndef   ARRAY_DEFAULT_INITIAL_SIZE    // also defined in dynarray.h 
1742 #define   ARRAY_DEFAULT_INITIAL_SIZE    (16) 
1745 #define   STRING(p)   ((wxString *)(&(p))) 
1748 void wxArrayString::Init(bool autoSort
) 
1752   m_pItems 
= (wxChar 
**) NULL
; 
1753   m_autoSort 
= autoSort
; 
1757 wxArrayString::wxArrayString(const wxArrayString
& src
) 
1759   Init(src
.m_autoSort
); 
1764 // assignment operator 
1765 wxArrayString
& wxArrayString::operator=(const wxArrayString
& src
) 
1772   m_autoSort 
= src
.m_autoSort
; 
1777 void wxArrayString::Copy(const wxArrayString
& src
) 
1779   if ( src
.m_nCount 
> ARRAY_DEFAULT_INITIAL_SIZE 
) 
1780     Alloc(src
.m_nCount
); 
1782   for ( size_t n 
= 0; n 
< src
.m_nCount
; n
++ ) 
1787 void wxArrayString::Grow(size_t nIncrement
) 
1789   // only do it if no more place 
1790   if ( (m_nSize 
- m_nCount
) < nIncrement 
) { 
1791     // if ARRAY_DEFAULT_INITIAL_SIZE were set to 0, the initially empty would 
1792     // be never resized! 
1793     #if ARRAY_DEFAULT_INITIAL_SIZE == 0 
1794       #error "ARRAY_DEFAULT_INITIAL_SIZE must be > 0!" 
1797     if ( m_nSize 
== 0 ) { 
1798       // was empty, alloc some memory 
1799       m_nSize 
= ARRAY_DEFAULT_INITIAL_SIZE
; 
1800       if (m_nSize 
< nIncrement
) 
1801           m_nSize 
= nIncrement
; 
1802       m_pItems 
= new wxChar 
*[m_nSize
]; 
1805       // otherwise when it's called for the first time, nIncrement would be 0 
1806       // and the array would never be expanded 
1807       // add 50% but not too much 
1808       size_t ndefIncrement 
= m_nSize 
< ARRAY_DEFAULT_INITIAL_SIZE
 
1809                           ? ARRAY_DEFAULT_INITIAL_SIZE 
: m_nSize 
>> 1; 
1810       if ( ndefIncrement 
> ARRAY_MAXSIZE_INCREMENT 
) 
1811         ndefIncrement 
= ARRAY_MAXSIZE_INCREMENT
; 
1812       if ( nIncrement 
< ndefIncrement 
) 
1813         nIncrement 
= ndefIncrement
; 
1814       m_nSize 
+= nIncrement
; 
1815       wxChar 
**pNew 
= new wxChar 
*[m_nSize
]; 
1817       // copy data to new location 
1818       memcpy(pNew
, m_pItems
, m_nCount
*sizeof(wxChar 
*)); 
1820       // delete old memory (but do not release the strings!) 
1821       wxDELETEA(m_pItems
); 
1828 void wxArrayString::Free() 
1830   for ( size_t n 
= 0; n 
< m_nCount
; n
++ ) { 
1831     STRING(m_pItems
[n
])->GetStringData()->Unlock(); 
1835 // deletes all the strings from the list 
1836 void wxArrayString::Empty() 
1843 // as Empty, but also frees memory 
1844 void wxArrayString::Clear() 
1851   wxDELETEA(m_pItems
); 
1855 wxArrayString::~wxArrayString() 
1859   wxDELETEA(m_pItems
); 
1862 // pre-allocates memory (frees the previous data!) 
1863 void wxArrayString::Alloc(size_t nSize
) 
1865   // only if old buffer was not big enough 
1866   if ( nSize 
> m_nSize 
) { 
1868     wxDELETEA(m_pItems
); 
1869     m_pItems 
= new wxChar 
*[nSize
]; 
1876 // minimizes the memory usage by freeing unused memory 
1877 void wxArrayString::Shrink() 
1879   // only do it if we have some memory to free 
1880   if( m_nCount 
< m_nSize 
) { 
1881     // allocates exactly as much memory as we need 
1882     wxChar 
**pNew 
= new wxChar 
*[m_nCount
]; 
1884     // copy data to new location 
1885     memcpy(pNew
, m_pItems
, m_nCount
*sizeof(wxChar 
*)); 
1891 // return a wxString[] as required for some control ctors. 
1892 wxString
* wxArrayString::GetStringArray() const 
1894     wxString 
*array 
= 0; 
1898         array 
= new wxString
[m_nCount
]; 
1899         for( size_t i 
= 0; i 
< m_nCount
; i
++ ) 
1900             array
[i
] = m_pItems
[i
]; 
1906 // searches the array for an item (forward or backwards) 
1907 int wxArrayString::Index(const wxChar 
*sz
, bool bCase
, bool bFromEnd
) const 
1910     // use binary search in the sorted array 
1911     wxASSERT_MSG( bCase 
&& !bFromEnd
, 
1912                   wxT("search parameters ignored for auto sorted array") ); 
1921       res 
= wxStrcmp(sz
, m_pItems
[i
]); 
1933     // use linear search in unsorted array 
1935       if ( m_nCount 
> 0 ) { 
1936         size_t ui 
= m_nCount
; 
1938           if ( STRING(m_pItems
[--ui
])->IsSameAs(sz
, bCase
) ) 
1945       for( size_t ui 
= 0; ui 
< m_nCount
; ui
++ ) { 
1946         if( STRING(m_pItems
[ui
])->IsSameAs(sz
, bCase
) ) 
1955 // add item at the end 
1956 size_t wxArrayString::Add(const wxString
& str
, size_t nInsert
) 
1959     // insert the string at the correct position to keep the array sorted 
1967       res 
= wxStrcmp(str
, m_pItems
[i
]); 
1978     wxASSERT_MSG( lo 
== hi
, wxT("binary search broken") ); 
1980     Insert(str
, lo
, nInsert
); 
1985     wxASSERT( str
.GetStringData()->IsValid() ); 
1989     for (size_t i 
= 0; i 
< nInsert
; i
++) 
1991         // the string data must not be deleted! 
1992         str
.GetStringData()->Lock(); 
1995         m_pItems
[m_nCount 
+ i
] = (wxChar 
*)str
.c_str(); // const_cast 
1997     size_t ret 
= m_nCount
; 
1998     m_nCount 
+= nInsert
; 
2003 // add item at the given position 
2004 void wxArrayString::Insert(const wxString
& str
, size_t nIndex
, size_t nInsert
) 
2006   wxASSERT( str
.GetStringData()->IsValid() ); 
2008   wxCHECK_RET( nIndex 
<= m_nCount
, wxT("bad index in wxArrayString::Insert") ); 
2009   wxCHECK_RET( m_nCount 
<= m_nCount 
+ nInsert
, 
2010                wxT("array size overflow in wxArrayString::Insert") ); 
2014   memmove(&m_pItems
[nIndex 
+ nInsert
], &m_pItems
[nIndex
], 
2015           (m_nCount 
- nIndex
)*sizeof(wxChar 
*)); 
2017   for (size_t i 
= 0; i 
< nInsert
; i
++) 
2019       str
.GetStringData()->Lock(); 
2020       m_pItems
[nIndex 
+ i
] = (wxChar 
*)str
.c_str(); 
2022   m_nCount 
+= nInsert
; 
2026 void wxArrayString::SetCount(size_t count
) 
2031     while ( m_nCount 
< count 
) 
2032         m_pItems
[m_nCount
++] = (wxChar 
*)s
.c_str(); 
2035 // removes item from array (by index) 
2036 void wxArrayString::Remove(size_t nIndex
, size_t nRemove
) 
2038   wxCHECK_RET( nIndex 
< m_nCount
, wxT("bad index in wxArrayString::Remove") ); 
2039   wxCHECK_RET( nIndex 
+ nRemove 
<= m_nCount
, 
2040                wxT("removing too many elements in wxArrayString::Remove") ); 
2043   for (size_t i 
= 0; i 
< nRemove
; i
++) 
2044       Item(nIndex 
+ i
).GetStringData()->Unlock(); 
2046   memmove(&m_pItems
[nIndex
], &m_pItems
[nIndex 
+ nRemove
], 
2047           (m_nCount 
- nIndex 
- nRemove
)*sizeof(wxChar 
*)); 
2048   m_nCount 
-= nRemove
; 
2051 // removes item from array (by value) 
2052 void wxArrayString::Remove(const wxChar 
*sz
) 
2054   int iIndex 
= Index(sz
); 
2056   wxCHECK_RET( iIndex 
!= wxNOT_FOUND
, 
2057                wxT("removing inexistent element in wxArrayString::Remove") ); 
2062 // ---------------------------------------------------------------------------- 
2064 // ---------------------------------------------------------------------------- 
2066 // we can only sort one array at a time with the quick-sort based 
2069   // need a critical section to protect access to gs_compareFunction and 
2070   // gs_sortAscending variables 
2071   static wxCriticalSection 
*gs_critsectStringSort 
= NULL
; 
2073   // call this before the value of the global sort vars is changed/after 
2074   // you're finished with them 
2075   #define START_SORT()     wxASSERT( !gs_critsectStringSort );                \ 
2076                            gs_critsectStringSort = new wxCriticalSection;     \ 
2077                            gs_critsectStringSort->Enter() 
2078   #define END_SORT()       gs_critsectStringSort->Leave();                    \ 
2079                            delete gs_critsectStringSort;                      \ 
2080                            gs_critsectStringSort = NULL 
2082   #define START_SORT() 
2084 #endif // wxUSE_THREADS 
2086 // function to use for string comparaison 
2087 static wxArrayString::CompareFunction gs_compareFunction 
= NULL
; 
2089 // if we don't use the compare function, this flag tells us if we sort the 
2090 // array in ascending or descending order 
2091 static bool gs_sortAscending 
= TRUE
; 
2093 // function which is called by quick sort 
2094 extern "C" int wxC_CALLING_CONV     
// LINKAGEMODE 
2095 wxStringCompareFunction(const void *first
, const void *second
) 
2097   wxString 
*strFirst 
= (wxString 
*)first
; 
2098   wxString 
*strSecond 
= (wxString 
*)second
; 
2100   if ( gs_compareFunction 
) { 
2101     return gs_compareFunction(*strFirst
, *strSecond
); 
2104     // maybe we should use wxStrcoll 
2105     int result 
= wxStrcmp(strFirst
->c_str(), strSecond
->c_str()); 
2107     return gs_sortAscending 
? result 
: -result
; 
2111 // sort array elements using passed comparaison function 
2112 void wxArrayString::Sort(CompareFunction compareFunction
) 
2116   wxASSERT( !gs_compareFunction 
);  // must have been reset to NULL 
2117   gs_compareFunction 
= compareFunction
; 
2121   // reset it to NULL so that Sort(bool) will work the next time 
2122   gs_compareFunction 
= NULL
; 
2127 void wxArrayString::Sort(bool reverseOrder
) 
2131   wxASSERT( !gs_compareFunction 
);  // must have been reset to NULL 
2132   gs_sortAscending 
= !reverseOrder
; 
2139 void wxArrayString::DoSort() 
2141   wxCHECK_RET( !m_autoSort
, wxT("can't use this method with sorted arrays") ); 
2143   // just sort the pointers using qsort() - of course it only works because 
2144   // wxString() *is* a pointer to its data 
2145   qsort(m_pItems
, m_nCount
, sizeof(wxChar 
*), wxStringCompareFunction
); 
2148 bool wxArrayString::operator==(const wxArrayString
& a
) const 
2150     if ( m_nCount 
!= a
.m_nCount 
) 
2153     for ( size_t n 
= 0; n 
< m_nCount
; n
++ ) 
2155         if ( Item(n
) != a
[n
] )