]> git.saurik.com Git - wxWidgets.git/blob - include/wx/string.h
changes for command line builds of osx-cocoa
[wxWidgets.git] / include / wx / string.h
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: wx/string.h
3 // Purpose: wxString class
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 29/01/98
7 // RCS-ID: $Id$
8 // Copyright: (c) 1998 Vadim Zeitlin <zeitlin@dptmaths.ens-cachan.fr>
9 // Licence: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
11
12 /*
13 Efficient string class [more or less] compatible with MFC CString,
14 wxWidgets version 1 wxString and std::string and some handy functions
15 missing from string.h.
16 */
17
18 #ifndef _WX_WXSTRING_H__
19 #define _WX_WXSTRING_H__
20
21 // ----------------------------------------------------------------------------
22 // headers
23 // ----------------------------------------------------------------------------
24
25 #include "wx/defs.h" // everybody should include this
26
27 #ifndef __WXPALMOS5__
28 #if defined(__WXMAC__) || defined(__VISAGECPP__)
29 #include <ctype.h>
30 #endif
31
32 #if defined(__VISAGECPP__) && __IBMCPP__ >= 400
33 // problem in VACPP V4 with including stdlib.h multiple times
34 // strconv includes it anyway
35 # include <stdio.h>
36 # include <string.h>
37 # include <stdarg.h>
38 # include <limits.h>
39 #else
40 # include <string.h>
41 # include <stdio.h>
42 # include <stdarg.h>
43 # include <limits.h>
44 # include <stdlib.h>
45 #endif
46
47 #ifdef HAVE_STRCASECMP_IN_STRINGS_H
48 #include <strings.h> // for strcasecmp()
49 #endif // HAVE_STRCASECMP_IN_STRINGS_H
50 #endif // ! __WXPALMOS5__
51
52 #include "wx/wxcrtbase.h" // for wxChar, wxStrlen() etc.
53 #include "wx/strvararg.h"
54 #include "wx/buffer.h" // for wxCharBuffer
55 #include "wx/strconv.h" // for wxConvertXXX() macros and wxMBConv classes
56 #include "wx/stringimpl.h"
57 #include "wx/stringops.h"
58 #include "wx/unichar.h"
59
60 // by default we cache the mapping of the positions in UTF-8 string to the byte
61 // offset as this results in noticeable performance improvements for loops over
62 // strings using indices; comment out this line to disable this
63 //
64 // notice that this optimization is well worth using even in debug builds as it
65 // changes asymptotic complexity of algorithms using indices to iterate over
66 // wxString back to expected linear from quadratic
67 //
68 // also notice that wxTLS_TYPE() (__declspec(thread) in this case) is unsafe to
69 // use in DLL build under pre-Vista Windows so we disable this code for now, if
70 // anybody really needs to use UTF-8 build under Windows with this optimization
71 // it would have to be re-tested and probably corrected
72 #if wxUSE_UNICODE_UTF8 && !defined(__WXMSW__)
73 #define wxUSE_STRING_POS_CACHE 1
74 #else
75 #define wxUSE_STRING_POS_CACHE 0
76 #endif
77
78 #if wxUSE_STRING_POS_CACHE
79 #include "wx/tls.h"
80
81 // change this 0 to 1 to enable additional (very expensive) asserts
82 // verifying that string caching logic works as expected
83 #if 0
84 #define wxSTRING_CACHE_ASSERT(cond) wxASSERT(cond)
85 #else
86 #define wxSTRING_CACHE_ASSERT(cond)
87 #endif
88 #endif // wxUSE_STRING_POS_CACHE
89
90 class WXDLLIMPEXP_FWD_BASE wxString;
91
92 // unless this symbol is predefined to disable the compatibility functions, do
93 // use them
94 #ifndef WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
95 #define WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER 1
96 #endif
97
98 namespace wxPrivate
99 {
100 template <typename T> struct wxStringAsBufHelper;
101 }
102
103 // ---------------------------------------------------------------------------
104 // macros
105 // ---------------------------------------------------------------------------
106
107 // casts [unfortunately!] needed to call some broken functions which require
108 // "char *" instead of "const char *"
109 #define WXSTRINGCAST (wxChar *)(const wxChar *)
110 #define wxCSTRINGCAST (wxChar *)(const wxChar *)
111 #define wxMBSTRINGCAST (char *)(const char *)
112 #define wxWCSTRINGCAST (wchar_t *)(const wchar_t *)
113
114 // ----------------------------------------------------------------------------
115 // constants
116 // ----------------------------------------------------------------------------
117
118 #if WXWIN_COMPATIBILITY_2_6
119
120 // deprecated in favour of wxString::npos, don't use in new code
121 //
122 // maximum possible length for a string means "take all string" everywhere
123 #define wxSTRING_MAXLEN wxString::npos
124
125 #endif // WXWIN_COMPATIBILITY_2_6
126
127 // ---------------------------------------------------------------------------
128 // global functions complementing standard C string library replacements for
129 // strlen() and portable strcasecmp()
130 //---------------------------------------------------------------------------
131
132 #if WXWIN_COMPATIBILITY_2_8
133 // Use wxXXX() functions from wxcrt.h instead! These functions are for
134 // backwards compatibility only.
135
136 // checks whether the passed in pointer is NULL and if the string is empty
137 wxDEPRECATED( inline bool IsEmpty(const char *p) );
138 inline bool IsEmpty(const char *p) { return (!p || !*p); }
139
140 // safe version of strlen() (returns 0 if passed NULL pointer)
141 wxDEPRECATED( inline size_t Strlen(const char *psz) );
142 inline size_t Strlen(const char *psz)
143 { return psz ? strlen(psz) : 0; }
144
145 // portable strcasecmp/_stricmp
146 wxDEPRECATED( inline int Stricmp(const char *psz1, const char *psz2) );
147 inline int Stricmp(const char *psz1, const char *psz2)
148 {
149 #if defined(__VISUALC__) && defined(__WXWINCE__)
150 register char c1, c2;
151 do {
152 c1 = tolower(*psz1++);
153 c2 = tolower(*psz2++);
154 } while ( c1 && (c1 == c2) );
155
156 return c1 - c2;
157 #elif defined(__VISUALC__) || ( defined(__MWERKS__) && defined(__INTEL__) )
158 return _stricmp(psz1, psz2);
159 #elif defined(__SC__)
160 return _stricmp(psz1, psz2);
161 #elif defined(__BORLANDC__)
162 return stricmp(psz1, psz2);
163 #elif defined(__WATCOMC__)
164 return stricmp(psz1, psz2);
165 #elif defined(__DJGPP__)
166 return stricmp(psz1, psz2);
167 #elif defined(__EMX__)
168 return stricmp(psz1, psz2);
169 #elif defined(__WXPM__)
170 return stricmp(psz1, psz2);
171 #elif defined(__WXPALMOS__) || \
172 defined(HAVE_STRCASECMP_IN_STRING_H) || \
173 defined(HAVE_STRCASECMP_IN_STRINGS_H) || \
174 defined(__GNUWIN32__)
175 return strcasecmp(psz1, psz2);
176 #elif defined(__MWERKS__) && !defined(__INTEL__)
177 register char c1, c2;
178 do {
179 c1 = tolower(*psz1++);
180 c2 = tolower(*psz2++);
181 } while ( c1 && (c1 == c2) );
182
183 return c1 - c2;
184 #else
185 // almost all compilers/libraries provide this function (unfortunately under
186 // different names), that's why we don't implement our own which will surely
187 // be more efficient than this code (uncomment to use):
188 /*
189 register char c1, c2;
190 do {
191 c1 = tolower(*psz1++);
192 c2 = tolower(*psz2++);
193 } while ( c1 && (c1 == c2) );
194
195 return c1 - c2;
196 */
197
198 #error "Please define string case-insensitive compare for your OS/compiler"
199 #endif // OS/compiler
200 }
201
202 #endif // WXWIN_COMPATIBILITY_2_8
203
204 // ----------------------------------------------------------------------------
205 // wxCStrData
206 // ----------------------------------------------------------------------------
207
208 // Lightweight object returned by wxString::c_str() and implicitly convertible
209 // to either const char* or const wchar_t*.
210 class WXDLLIMPEXP_BASE wxCStrData
211 {
212 private:
213 // Ctors; for internal use by wxString and wxCStrData only
214 wxCStrData(const wxString *str, size_t offset = 0, bool owned = false)
215 : m_str(str), m_offset(offset), m_owned(owned) {}
216
217 public:
218 // Ctor constructs the object from char literal; they are needed to make
219 // operator?: compile and they intentionally take char*, not const char*
220 inline wxCStrData(char *buf);
221 inline wxCStrData(wchar_t *buf);
222 inline wxCStrData(const wxCStrData& data);
223
224 inline ~wxCStrData();
225
226 // methods defined inline below must be declared inline or mingw32 3.4.5
227 // warns about "<symbol> defined locally after being referenced with
228 // dllimport linkage"
229 #if wxUSE_UNICODE_WCHAR
230 inline
231 #endif
232 const wchar_t* AsWChar() const;
233 operator const wchar_t*() const { return AsWChar(); }
234
235 #if !wxUSE_UNICODE || wxUSE_UTF8_LOCALE_ONLY
236 inline
237 #endif
238 const char* AsChar() const;
239 const unsigned char* AsUnsignedChar() const
240 { return (const unsigned char *) AsChar(); }
241 operator const char*() const { return AsChar(); }
242 operator const unsigned char*() const { return AsUnsignedChar(); }
243
244 operator const void*() const { return AsChar(); }
245
246 inline const wxCharBuffer AsCharBuf() const;
247 inline const wxWCharBuffer AsWCharBuf() const;
248
249 inline wxString AsString() const;
250
251 // returns the value as C string in internal representation (equivalent
252 // to AsString().wx_str(), but more efficient)
253 const wxStringCharType *AsInternal() const;
254
255 // allow expressions like "c_str()[0]":
256 inline wxUniChar operator[](size_t n) const;
257 wxUniChar operator[](int n) const { return operator[](size_t(n)); }
258 wxUniChar operator[](long n) const { return operator[](size_t(n)); }
259 #ifndef wxSIZE_T_IS_UINT
260 wxUniChar operator[](unsigned int n) const { return operator[](size_t(n)); }
261 #endif // size_t != unsigned int
262
263 // these operators are needed to emulate the pointer semantics of c_str():
264 // expressions like "wxChar *p = str.c_str() + 1;" should continue to work
265 // (we need both versions to resolve ambiguities):
266 wxCStrData operator+(int n) const
267 { return wxCStrData(m_str, m_offset + n, m_owned); }
268 wxCStrData operator+(long n) const
269 { return wxCStrData(m_str, m_offset + n, m_owned); }
270 wxCStrData operator+(size_t n) const
271 { return wxCStrData(m_str, m_offset + n, m_owned); }
272
273 // and these for "str.c_str() + (p2 - p1)" (it also works for any integer
274 // expression but it must be ptrdiff_t and not e.g. int to work in this
275 // example):
276 wxCStrData operator-(ptrdiff_t n) const
277 {
278 wxASSERT_MSG( n <= (ptrdiff_t)m_offset,
279 _T("attempt to construct address before the beginning of the string") );
280 return wxCStrData(m_str, m_offset - n, m_owned);
281 }
282
283 // this operator is needed to make expressions like "*c_str()" or
284 // "*(c_str() + 2)" work
285 inline wxUniChar operator*() const;
286
287 private:
288 const wxString *m_str;
289 size_t m_offset;
290 bool m_owned;
291
292 friend class WXDLLIMPEXP_FWD_BASE wxString;
293 };
294
295 // ----------------------------------------------------------------------------
296 // wxStringPrintfMixin
297 // ---------------------------------------------------------------------------
298
299 // NB: VC6 has a bug that causes linker errors if you have template methods
300 // in a class using __declspec(dllimport). The solution is to split such
301 // class into two classes, one that contains the template methods and does
302 // *not* use WXDLLIMPEXP_BASE and another class that contains the rest
303 // (with DLL linkage).
304 //
305 // We only do this for VC6 here, because the code is less efficient
306 // (Printf() has to use dynamic_cast<>) and because OpenWatcom compiler
307 // cannot compile this code.
308
309 #if defined(__VISUALC__) && __VISUALC__ < 1300
310 #define wxNEEDS_WXSTRING_PRINTF_MIXIN
311 #endif
312
313 #ifdef wxNEEDS_WXSTRING_PRINTF_MIXIN
314 // this class contains implementation of wxString's vararg methods, it's
315 // exported from wxBase DLL
316 class WXDLLIMPEXP_BASE wxStringPrintfMixinBase
317 {
318 protected:
319 wxStringPrintfMixinBase() {}
320
321 #if !wxUSE_UTF8_LOCALE_ONLY
322 int DoPrintfWchar(const wxChar *format, ...);
323 static wxString DoFormatWchar(const wxChar *format, ...);
324 #endif
325 #if wxUSE_UNICODE_UTF8
326 int DoPrintfUtf8(const char *format, ...);
327 static wxString DoFormatUtf8(const char *format, ...);
328 #endif
329 };
330
331 // this class contains template wrappers for wxString's vararg methods, it's
332 // intentionally *not* exported from the DLL in order to fix the VC6 bug
333 // described above
334 class wxStringPrintfMixin : public wxStringPrintfMixinBase
335 {
336 private:
337 // to further complicate things, we can't return wxString from
338 // wxStringPrintfMixin::Format() because wxString is not yet declared at
339 // this point; the solution is to use this fake type trait template - this
340 // way the compiler won't know the return type until Format() is used
341 // (this doesn't compile with Watcom, but VC6 compiles it just fine):
342 template<typename T> struct StringReturnType
343 {
344 typedef wxString type;
345 };
346
347 public:
348 // these are duplicated wxString methods, they're also declared below
349 // if !wxNEEDS_WXSTRING_PRINTF_MIXIN:
350
351 // static wxString Format(const wString& format, ...) ATTRIBUTE_PRINTF_1;
352 WX_DEFINE_VARARG_FUNC_SANS_N0(static typename StringReturnType<T1>::type,
353 Format, 1, (const wxFormatString&),
354 DoFormatWchar, DoFormatUtf8)
355 // We have to implement the version without template arguments manually
356 // because of the StringReturnType<> hack, although WX_DEFINE_VARARG_FUNC
357 // normally does it itself. It has to be a template so that we can use
358 // the hack, even though there's no real template parameter. We can't move
359 // it to wxStrig, because it would shadow these versions of Format() then.
360 template<typename T>
361 inline static typename StringReturnType<T>::type
362 Format(const T& fmt)
363 {
364 // NB: this doesn't compile if T is not (some form of) a string;
365 // this makes Format's prototype equivalent to
366 // Format(const wxFormatString& fmt)
367 return DoFormatWchar(wxFormatString(fmt));
368 }
369
370 // int Printf(const wxString& format, ...);
371 WX_DEFINE_VARARG_FUNC(int, Printf, 1, (const wxFormatString&),
372 DoPrintfWchar, DoPrintfUtf8)
373 // int sprintf(const wxString& format, ...) ATTRIBUTE_PRINTF_2;
374 WX_DEFINE_VARARG_FUNC(int, sprintf, 1, (const wxFormatString&),
375 DoPrintfWchar, DoPrintfUtf8)
376
377 protected:
378 wxStringPrintfMixin() : wxStringPrintfMixinBase() {}
379 };
380 #endif // wxNEEDS_WXSTRING_PRINTF_MIXIN
381
382
383 // ----------------------------------------------------------------------------
384 // wxString: string class trying to be compatible with std::string, MFC
385 // CString and wxWindows 1.x wxString all at once
386 // ---------------------------------------------------------------------------
387
388 #ifdef wxNEEDS_WXSTRING_PRINTF_MIXIN
389 // "non dll-interface class 'wxStringPrintfMixin' used as base interface
390 // for dll-interface class 'wxString'" -- this is OK in our case
391 #pragma warning (disable:4275)
392 #endif
393
394 #if wxUSE_UNICODE_UTF8
395 // see the comment near wxString::iterator for why we need this
396 class WXDLLIMPEXP_BASE wxStringIteratorNode
397 {
398 public:
399 wxStringIteratorNode()
400 : m_str(NULL), m_citer(NULL), m_iter(NULL), m_prev(NULL), m_next(NULL) {}
401 wxStringIteratorNode(const wxString *str,
402 wxStringImpl::const_iterator *citer)
403 { DoSet(str, citer, NULL); }
404 wxStringIteratorNode(const wxString *str, wxStringImpl::iterator *iter)
405 { DoSet(str, NULL, iter); }
406 ~wxStringIteratorNode()
407 { clear(); }
408
409 inline void set(const wxString *str, wxStringImpl::const_iterator *citer)
410 { clear(); DoSet(str, citer, NULL); }
411 inline void set(const wxString *str, wxStringImpl::iterator *iter)
412 { clear(); DoSet(str, NULL, iter); }
413
414 const wxString *m_str;
415 wxStringImpl::const_iterator *m_citer;
416 wxStringImpl::iterator *m_iter;
417 wxStringIteratorNode *m_prev, *m_next;
418
419 private:
420 inline void clear();
421 inline void DoSet(const wxString *str,
422 wxStringImpl::const_iterator *citer,
423 wxStringImpl::iterator *iter);
424
425 // the node belongs to a particular iterator instance, it's not copied
426 // when a copy of the iterator is made
427 DECLARE_NO_COPY_CLASS(wxStringIteratorNode)
428 };
429 #endif // wxUSE_UNICODE_UTF8
430
431 class WXDLLIMPEXP_BASE wxString
432 #ifdef wxNEEDS_WXSTRING_PRINTF_MIXIN
433 : public wxStringPrintfMixin
434 #endif
435 {
436 // NB: special care was taken in arranging the member functions in such order
437 // that all inline functions can be effectively inlined, verify that all
438 // performance critical functions are still inlined if you change order!
439 public:
440 // an 'invalid' value for string index, moved to this place due to a CW bug
441 static const size_t npos;
442
443 private:
444 // if we hadn't made these operators private, it would be possible to
445 // compile "wxString s; s = 17;" without any warnings as 17 is implicitly
446 // converted to char in C and we do have operator=(char)
447 //
448 // NB: we don't need other versions (short/long and unsigned) as attempt
449 // to assign another numeric type to wxString will now result in
450 // ambiguity between operator=(char) and operator=(int)
451 wxString& operator=(int);
452
453 // these methods are not implemented - there is _no_ conversion from int to
454 // string, you're doing something wrong if the compiler wants to call it!
455 //
456 // try `s << i' or `s.Printf("%d", i)' instead
457 wxString(int);
458
459
460 // buffer for holding temporary substring when using any of the methods
461 // that take (char*,size_t) or (wchar_t*,size_t) arguments:
462 template<typename T>
463 struct SubstrBufFromType
464 {
465 T data;
466 size_t len;
467
468 SubstrBufFromType(const T& data_, size_t len_)
469 : data(data_), len(len_)
470 {
471 wxASSERT_MSG( len != npos, "must have real length" );
472 }
473 };
474
475 #if wxUSE_UNICODE_UTF8
476 // even char* -> char* needs conversion, from locale charset to UTF-8
477 typedef SubstrBufFromType<wxCharBuffer> SubstrBufFromWC;
478 typedef SubstrBufFromType<wxCharBuffer> SubstrBufFromMB;
479 #elif wxUSE_UNICODE_WCHAR
480 typedef SubstrBufFromType<const wchar_t*> SubstrBufFromWC;
481 typedef SubstrBufFromType<wxWCharBuffer> SubstrBufFromMB;
482 #else
483 typedef SubstrBufFromType<const char*> SubstrBufFromMB;
484 typedef SubstrBufFromType<wxCharBuffer> SubstrBufFromWC;
485 #endif
486
487
488 // Functions implementing primitive operations on string data; wxString
489 // methods and iterators are implemented in terms of it. The differences
490 // between UTF-8 and wchar_t* representations of the string are mostly
491 // contained here.
492
493 #if wxUSE_UNICODE_UTF8
494 static SubstrBufFromMB ConvertStr(const char *psz, size_t nLength,
495 const wxMBConv& conv);
496 static SubstrBufFromWC ConvertStr(const wchar_t *pwz, size_t nLength,
497 const wxMBConv& conv);
498 #elif wxUSE_UNICODE_WCHAR
499 static SubstrBufFromMB ConvertStr(const char *psz, size_t nLength,
500 const wxMBConv& conv);
501 #else
502 static SubstrBufFromWC ConvertStr(const wchar_t *pwz, size_t nLength,
503 const wxMBConv& conv);
504 #endif
505
506 #if !wxUSE_UNICODE_UTF8 // wxUSE_UNICODE_WCHAR or !wxUSE_UNICODE
507 // returns C string encoded as the implementation expects:
508 #if wxUSE_UNICODE
509 static const wchar_t* ImplStr(const wchar_t* str)
510 { return str ? str : wxT(""); }
511 static const SubstrBufFromWC ImplStr(const wchar_t* str, size_t n)
512 { return SubstrBufFromWC(str, (str && n == npos) ? wxWcslen(str) : n); }
513 static wxWCharBuffer ImplStr(const char* str,
514 const wxMBConv& conv = wxConvLibc)
515 { return ConvertStr(str, npos, conv).data; }
516 static SubstrBufFromMB ImplStr(const char* str, size_t n,
517 const wxMBConv& conv = wxConvLibc)
518 { return ConvertStr(str, n, conv); }
519 #else
520 static const char* ImplStr(const char* str,
521 const wxMBConv& WXUNUSED(conv) = wxConvLibc)
522 { return str ? str : ""; }
523 static const SubstrBufFromMB ImplStr(const char* str, size_t n,
524 const wxMBConv& WXUNUSED(conv) = wxConvLibc)
525 { return SubstrBufFromMB(str, (str && n == npos) ? wxStrlen(str) : n); }
526 static wxCharBuffer ImplStr(const wchar_t* str)
527 { return ConvertStr(str, npos, wxConvLibc).data; }
528 static SubstrBufFromWC ImplStr(const wchar_t* str, size_t n)
529 { return ConvertStr(str, n, wxConvLibc); }
530 #endif
531
532 // translates position index in wxString to/from index in underlying
533 // wxStringImpl:
534 static size_t PosToImpl(size_t pos) { return pos; }
535 static void PosLenToImpl(size_t pos, size_t len,
536 size_t *implPos, size_t *implLen)
537 { *implPos = pos; *implLen = len; }
538 static size_t LenToImpl(size_t len) { return len; }
539 static size_t PosFromImpl(size_t pos) { return pos; }
540
541 // we don't want to define these as empty inline functions as it could
542 // result in noticeable (and quite unnecessary in non-UTF-8 build) slowdown
543 // in debug build where the inline functions are not effectively inlined
544 #define wxSTRING_INVALIDATE_CACHE()
545 #define wxSTRING_INVALIDATE_CACHED_LENGTH()
546 #define wxSTRING_UPDATE_CACHED_LENGTH(n)
547 #define wxSTRING_SET_CACHED_LENGTH(n)
548
549 #else // wxUSE_UNICODE_UTF8
550
551 static wxCharBuffer ImplStr(const char* str,
552 const wxMBConv& conv = wxConvLibc)
553 { return ConvertStr(str, npos, conv).data; }
554 static SubstrBufFromMB ImplStr(const char* str, size_t n,
555 const wxMBConv& conv = wxConvLibc)
556 { return ConvertStr(str, n, conv); }
557
558 static wxCharBuffer ImplStr(const wchar_t* str)
559 { return ConvertStr(str, npos, wxMBConvUTF8()).data; }
560 static SubstrBufFromWC ImplStr(const wchar_t* str, size_t n)
561 { return ConvertStr(str, n, wxMBConvUTF8()); }
562
563 #if wxUSE_STRING_POS_CACHE
564 // this is an extremely simple cache used by PosToImpl(): each cache element
565 // contains the string it applies to and the index corresponding to the last
566 // used position in this wxString in its m_impl string
567 //
568 // NB: notice that this struct (and nested Element one) must be a POD or we
569 // wouldn't be able to use a thread-local variable of this type, in
570 // particular it should have no ctor -- we rely on statics being
571 // initialized to 0 instead
572 struct Cache
573 {
574 enum { SIZE = 8 };
575
576 struct Element
577 {
578 const wxString *str; // the string to which this element applies
579 size_t pos, // the cached index in this string
580 impl, // the corresponding position in its m_impl
581 len; // cached length or npos if unknown
582
583 // reset cached index to 0
584 void ResetPos() { pos = impl = 0; }
585
586 // reset position and length
587 void Reset() { ResetPos(); len = npos; }
588 };
589
590 // cache the indices mapping for the last few string used
591 Element cached[SIZE];
592
593 // the last used index
594 unsigned lastUsed;
595 };
596
597 #ifndef wxHAS_COMPILER_TLS
598 // we must use an accessor function and not a static variable when the TLS
599 // variables support is implemented in the library (and not by the compiler)
600 // because the global s_cache variable could be not yet initialized when a
601 // ctor of another global object is executed and if that ctor uses any
602 // wxString methods, bad things happen
603 //
604 // however notice that this approach does not work when compiler TLS is used,
605 // at least not with g++ 4.1.2 under amd64 as it apparently compiles code
606 // using this accessor incorrectly when optimizations are enabled (-O2 is
607 // enough) -- luckily we don't need it then neither as static __thread
608 // variables are initialized by 0 anyhow then and so we can use the variable
609 // directly
610 static Cache& GetCache()
611 {
612 static wxTLS_TYPE(Cache) s_cache;
613
614 return wxTLS_VALUE(s_cache);
615 }
616
617 // this helper struct is used to ensure that GetCache() is called during
618 // static initialization time, i.e. before any threads creation, as otherwise
619 // the static s_cache construction inside GetCache() wouldn't be MT-safe
620 friend struct wxStrCacheInitializer;
621 #else // wxHAS_COMPILER_TLS
622 static wxTLS_TYPE(Cache) ms_cache;
623 static Cache& GetCache() { return wxTLS_VALUE(ms_cache); }
624 #endif // !wxHAS_COMPILER_TLS/wxHAS_COMPILER_TLS
625
626 static Cache::Element *GetCacheBegin() { return GetCache().cached; }
627 static Cache::Element *GetCacheEnd() { return GetCacheBegin() + Cache::SIZE; }
628 static unsigned& LastUsedCacheElement() { return GetCache().lastUsed; }
629
630 // this is used in debug builds only to provide a convenient function,
631 // callable from a debugger, to show the cache contents
632 friend struct wxStrCacheDumper;
633
634 // uncomment this to have access to some profiling statistics on program
635 // termination
636 //#define wxPROFILE_STRING_CACHE
637
638 #ifdef wxPROFILE_STRING_CACHE
639 static struct PosToImplCacheStats
640 {
641 unsigned postot, // total non-trivial calls to PosToImpl
642 poshits, // cache hits from PosToImpl()
643 mishits, // cached position beyond the needed one
644 sumpos, // sum of all positions, used to compute the
645 // average position after dividing by postot
646 sumofs, // sum of all offsets after using the cache, used to
647 // compute the average after dividing by hits
648 lentot, // number of total calls to length()
649 lenhits; // number of cache hits in length()
650 } ms_cacheStats;
651
652 friend struct wxStrCacheStatsDumper;
653
654 #define wxCACHE_PROFILE_FIELD_INC(field) ms_cacheStats.field++
655 #define wxCACHE_PROFILE_FIELD_ADD(field, val) ms_cacheStats.field += (val)
656 #else // !wxPROFILE_STRING_CACHE
657 #define wxCACHE_PROFILE_FIELD_INC(field)
658 #define wxCACHE_PROFILE_FIELD_ADD(field, val)
659 #endif // wxPROFILE_STRING_CACHE/!wxPROFILE_STRING_CACHE
660
661 // note: it could seem that the functions below shouldn't be inline because
662 // they are big, contain loops and so the compiler shouldn't be able to
663 // inline them anyhow, however moving them into string.cpp does decrease the
664 // code performance by ~5%, at least when using g++ 4.1 so do keep them here
665 // unless tests show that it's not advantageous any more
666
667 // return the pointer to the cache element for this string or NULL if not
668 // cached
669 Cache::Element *FindCacheElement() const
670 {
671 // profiling seems to show a small but consistent gain if we use this
672 // simple loop instead of starting from the last used element (there are
673 // a lot of misses in this function...)
674 for ( Cache::Element *c = GetCacheBegin(); c != GetCacheEnd(); c++ )
675 {
676 if ( c->str == this )
677 return c;
678 }
679
680 return NULL;
681 }
682
683 // unlike FindCacheElement(), this one always returns a valid pointer to the
684 // cache element for this string, it may have valid last cached position and
685 // its corresponding index in the byte string or not
686 Cache::Element *GetCacheElement() const
687 {
688 Cache::Element * const cacheBegin = GetCacheBegin();
689 Cache::Element * const cacheEnd = GetCacheEnd();
690 Cache::Element * const cacheStart = cacheBegin + LastUsedCacheElement();
691
692 // check the last used first, this does no (measurable) harm for a miss
693 // but does help for simple loops addressing the same string all the time
694 if ( cacheStart->str == this )
695 return cacheStart;
696
697 // notice that we're going to check cacheStart again inside this call but
698 // profiling shows that it's still faster to use a simple loop like
699 // inside FindCacheElement() than manually looping with wrapping starting
700 // from the cache entry after the start one
701 Cache::Element *c = FindCacheElement();
702 if ( !c )
703 {
704 // claim the next cache entry for this string
705 c = cacheStart;
706 if ( ++c == cacheEnd )
707 c = cacheBegin;
708
709 c->str = this;
710 c->Reset();
711
712 // and remember the last used element
713 LastUsedCacheElement() = c - cacheBegin;
714 }
715
716 return c;
717 }
718
719 size_t DoPosToImpl(size_t pos) const
720 {
721 wxCACHE_PROFILE_FIELD_INC(postot);
722
723 // NB: although the case of pos == 1 (and offset from cached position
724 // equal to 1) are common, nothing is gained by writing special code
725 // for handling them, the compiler (at least g++ 4.1 used) seems to
726 // optimize the code well enough on its own
727
728 wxCACHE_PROFILE_FIELD_ADD(sumpos, pos);
729
730 Cache::Element * const cache = GetCacheElement();
731
732 // cached position can't be 0 so if it is, it means that this entry was
733 // used for length caching only so far, i.e. it doesn't count as a hit
734 // from our point of view
735 if ( cache->pos )
736 {
737 wxCACHE_PROFILE_FIELD_INC(poshits);
738 }
739
740 if ( pos == cache->pos )
741 return cache->impl;
742
743 // this seems to happen only rarely so just reset the cache in this case
744 // instead of complicating code even further by seeking backwards in this
745 // case
746 if ( cache->pos > pos )
747 {
748 wxCACHE_PROFILE_FIELD_INC(mishits);
749
750 cache->ResetPos();
751 }
752
753 wxCACHE_PROFILE_FIELD_ADD(sumofs, pos - cache->pos);
754
755
756 wxStringImpl::const_iterator i(m_impl.begin() + cache->impl);
757 for ( size_t n = cache->pos; n < pos; n++ )
758 wxStringOperations::IncIter(i);
759
760 cache->pos = pos;
761 cache->impl = i - m_impl.begin();
762
763 wxSTRING_CACHE_ASSERT(
764 (int)cache->impl == (begin() + pos).impl() - m_impl.begin() );
765
766 return cache->impl;
767 }
768
769 void InvalidateCache()
770 {
771 Cache::Element * const cache = FindCacheElement();
772 if ( cache )
773 cache->Reset();
774 }
775
776 void InvalidateCachedLength()
777 {
778 Cache::Element * const cache = FindCacheElement();
779 if ( cache )
780 cache->len = npos;
781 }
782
783 void SetCachedLength(size_t len)
784 {
785 // we optimistically cache the length here even if the string wasn't
786 // present in the cache before, this seems to do no harm and the
787 // potential for avoiding length recomputation for long strings looks
788 // interesting
789 GetCacheElement()->len = len;
790 }
791
792 void UpdateCachedLength(ptrdiff_t delta)
793 {
794 Cache::Element * const cache = FindCacheElement();
795 if ( cache && cache->len != npos )
796 {
797 wxSTRING_CACHE_ASSERT( (ptrdiff_t)cache->len + delta >= 0 );
798
799 cache->len += delta;
800 }
801 }
802
803 #define wxSTRING_INVALIDATE_CACHE() InvalidateCache()
804 #define wxSTRING_INVALIDATE_CACHED_LENGTH() InvalidateCachedLength()
805 #define wxSTRING_UPDATE_CACHED_LENGTH(n) UpdateCachedLength(n)
806 #define wxSTRING_SET_CACHED_LENGTH(n) SetCachedLength(n)
807 #else // !wxUSE_STRING_POS_CACHE
808 size_t DoPosToImpl(size_t pos) const
809 {
810 return (begin() + pos).impl() - m_impl.begin();
811 }
812
813 #define wxSTRING_INVALIDATE_CACHE()
814 #define wxSTRING_INVALIDATE_CACHED_LENGTH()
815 #define wxSTRING_UPDATE_CACHED_LENGTH(n)
816 #define wxSTRING_SET_CACHED_LENGTH(n)
817 #endif // wxUSE_STRING_POS_CACHE/!wxUSE_STRING_POS_CACHE
818
819 size_t PosToImpl(size_t pos) const
820 {
821 return pos == 0 || pos == npos ? pos : DoPosToImpl(pos);
822 }
823
824 void PosLenToImpl(size_t pos, size_t len, size_t *implPos, size_t *implLen) const;
825
826 size_t LenToImpl(size_t len) const
827 {
828 size_t pos, len2;
829 PosLenToImpl(0, len, &pos, &len2);
830 return len2;
831 }
832
833 size_t PosFromImpl(size_t pos) const
834 {
835 if ( pos == 0 || pos == npos )
836 return pos;
837 else
838 return const_iterator(this, m_impl.begin() + pos) - begin();
839 }
840 #endif // !wxUSE_UNICODE_UTF8/wxUSE_UNICODE_UTF8
841
842 public:
843 // standard types
844 typedef wxUniChar value_type;
845 typedef wxUniChar char_type;
846 typedef wxUniCharRef reference;
847 typedef wxChar* pointer;
848 typedef const wxChar* const_pointer;
849
850 typedef size_t size_type;
851 typedef wxUniChar const_reference;
852
853 #if wxUSE_STD_STRING
854 #if wxUSE_UNICODE_UTF8
855 // random access is not O(1), as required by Random Access Iterator
856 #define WX_STR_ITERATOR_TAG std::bidirectional_iterator_tag
857 #else
858 #define WX_STR_ITERATOR_TAG std::random_access_iterator_tag
859 #endif
860 #define WX_DEFINE_ITERATOR_CATEGORY(cat) typedef cat iterator_category;
861 #else
862 // not defining iterator_category at all in this case is better than defining
863 // it as some dummy type -- at least it results in more intelligible error
864 // messages
865 #define WX_DEFINE_ITERATOR_CATEGORY(cat)
866 #endif
867
868 #define WX_STR_ITERATOR_IMPL(iterator_name, pointer_type, reference_type) \
869 private: \
870 typedef wxStringImpl::iterator_name underlying_iterator; \
871 public: \
872 WX_DEFINE_ITERATOR_CATEGORY(WX_STR_ITERATOR_TAG) \
873 typedef wxUniChar value_type; \
874 typedef int difference_type; \
875 typedef reference_type reference; \
876 typedef pointer_type pointer; \
877 \
878 reference operator[](size_t n) const { return *(*this + n); } \
879 \
880 iterator_name& operator++() \
881 { wxStringOperations::IncIter(m_cur); return *this; } \
882 iterator_name& operator--() \
883 { wxStringOperations::DecIter(m_cur); return *this; } \
884 iterator_name operator++(int) \
885 { \
886 iterator_name tmp = *this; \
887 wxStringOperations::IncIter(m_cur); \
888 return tmp; \
889 } \
890 iterator_name operator--(int) \
891 { \
892 iterator_name tmp = *this; \
893 wxStringOperations::DecIter(m_cur); \
894 return tmp; \
895 } \
896 \
897 iterator_name& operator+=(ptrdiff_t n) \
898 { \
899 m_cur = wxStringOperations::AddToIter(m_cur, n); \
900 return *this; \
901 } \
902 iterator_name& operator-=(ptrdiff_t n) \
903 { \
904 m_cur = wxStringOperations::AddToIter(m_cur, -n); \
905 return *this; \
906 } \
907 \
908 difference_type operator-(const iterator_name& i) const \
909 { return wxStringOperations::DiffIters(m_cur, i.m_cur); } \
910 \
911 bool operator==(const iterator_name& i) const \
912 { return m_cur == i.m_cur; } \
913 bool operator!=(const iterator_name& i) const \
914 { return m_cur != i.m_cur; } \
915 \
916 bool operator<(const iterator_name& i) const \
917 { return m_cur < i.m_cur; } \
918 bool operator>(const iterator_name& i) const \
919 { return m_cur > i.m_cur; } \
920 bool operator<=(const iterator_name& i) const \
921 { return m_cur <= i.m_cur; } \
922 bool operator>=(const iterator_name& i) const \
923 { return m_cur >= i.m_cur; } \
924 \
925 private: \
926 /* for internal wxString use only: */ \
927 underlying_iterator impl() const { return m_cur; } \
928 \
929 friend class wxString; \
930 friend class wxCStrData; \
931 \
932 private: \
933 underlying_iterator m_cur
934
935 class WXDLLIMPEXP_FWD_BASE const_iterator;
936
937 #if wxUSE_UNICODE_UTF8
938 // NB: In UTF-8 build, (non-const) iterator needs to keep reference
939 // to the underlying wxStringImpl, because UTF-8 is variable-length
940 // encoding and changing the value pointer to by an iterator (using
941 // its operator*) requires calling wxStringImpl::replace() if the old
942 // and new values differ in their encoding's length.
943 //
944 // Furthermore, the replace() call may invalid all iterators for the
945 // string, so we have to keep track of outstanding iterators and update
946 // them if replace() happens.
947 //
948 // This is implemented by maintaining linked list of iterators for every
949 // string and traversing it in wxUniCharRef::operator=(). Head of the
950 // list is stored in wxString. (FIXME-UTF8)
951
952 class WXDLLIMPEXP_BASE iterator
953 {
954 WX_STR_ITERATOR_IMPL(iterator, wxChar*, wxUniCharRef);
955
956 public:
957 iterator() {}
958 iterator(const iterator& i)
959 : m_cur(i.m_cur), m_node(i.str(), &m_cur) {}
960 iterator& operator=(const iterator& i)
961 {
962 if (&i != this)
963 {
964 m_cur = i.m_cur;
965 m_node.set(i.str(), &m_cur);
966 }
967 return *this;
968 }
969
970 reference operator*()
971 { return wxUniCharRef::CreateForString(*str(), m_cur); }
972
973 iterator operator+(ptrdiff_t n) const
974 { return iterator(str(), wxStringOperations::AddToIter(m_cur, n)); }
975 iterator operator-(ptrdiff_t n) const
976 { return iterator(str(), wxStringOperations::AddToIter(m_cur, -n)); }
977
978 private:
979 iterator(wxString *str, underlying_iterator ptr)
980 : m_cur(ptr), m_node(str, &m_cur) {}
981
982 wxString* str() const { return wx_const_cast(wxString*, m_node.m_str); }
983
984 wxStringIteratorNode m_node;
985
986 friend class const_iterator;
987 };
988
989 class WXDLLIMPEXP_BASE const_iterator
990 {
991 // NB: reference_type is intentionally value, not reference, the character
992 // may be encoded differently in wxString data:
993 WX_STR_ITERATOR_IMPL(const_iterator, const wxChar*, wxUniChar);
994
995 public:
996 const_iterator() {}
997 const_iterator(const const_iterator& i)
998 : m_cur(i.m_cur), m_node(i.str(), &m_cur) {}
999 const_iterator(const iterator& i)
1000 : m_cur(i.m_cur), m_node(i.str(), &m_cur) {}
1001
1002 const_iterator& operator=(const const_iterator& i)
1003 {
1004 if (&i != this)
1005 {
1006 m_cur = i.m_cur;
1007 m_node.set(i.str(), &m_cur);
1008 }
1009 return *this;
1010 }
1011 const_iterator& operator=(const iterator& i)
1012 { m_cur = i.m_cur; m_node.set(i.str(), &m_cur); return *this; }
1013
1014 reference operator*() const
1015 { return wxStringOperations::DecodeChar(m_cur); }
1016
1017 const_iterator operator+(ptrdiff_t n) const
1018 { return const_iterator(str(), wxStringOperations::AddToIter(m_cur, n)); }
1019 const_iterator operator-(ptrdiff_t n) const
1020 { return const_iterator(str(), wxStringOperations::AddToIter(m_cur, -n)); }
1021
1022 private:
1023 // for internal wxString use only:
1024 const_iterator(const wxString *str, underlying_iterator ptr)
1025 : m_cur(ptr), m_node(str, &m_cur) {}
1026
1027 const wxString* str() const { return m_node.m_str; }
1028
1029 wxStringIteratorNode m_node;
1030 };
1031
1032 size_t IterToImplPos(wxString::iterator i) const
1033 { return wxStringImpl::const_iterator(i.impl()) - m_impl.begin(); }
1034
1035 iterator GetIterForNthChar(size_t n)
1036 { return iterator(this, m_impl.begin() + PosToImpl(n)); }
1037 const_iterator GetIterForNthChar(size_t n) const
1038 { return const_iterator(this, m_impl.begin() + PosToImpl(n)); }
1039 #else // !wxUSE_UNICODE_UTF8
1040
1041 class WXDLLIMPEXP_BASE iterator
1042 {
1043 WX_STR_ITERATOR_IMPL(iterator, wxChar*, wxUniCharRef);
1044
1045 public:
1046 iterator() {}
1047 iterator(const iterator& i) : m_cur(i.m_cur) {}
1048
1049 reference operator*()
1050 { return wxUniCharRef::CreateForString(m_cur); }
1051
1052 iterator operator+(ptrdiff_t n) const
1053 { return iterator(wxStringOperations::AddToIter(m_cur, n)); }
1054 iterator operator-(ptrdiff_t n) const
1055 { return iterator(wxStringOperations::AddToIter(m_cur, -n)); }
1056
1057 private:
1058 // for internal wxString use only:
1059 iterator(underlying_iterator ptr) : m_cur(ptr) {}
1060 iterator(wxString *WXUNUSED(str), underlying_iterator ptr) : m_cur(ptr) {}
1061
1062 friend class const_iterator;
1063 };
1064
1065 class WXDLLIMPEXP_BASE const_iterator
1066 {
1067 // NB: reference_type is intentionally value, not reference, the character
1068 // may be encoded differently in wxString data:
1069 WX_STR_ITERATOR_IMPL(const_iterator, const wxChar*, wxUniChar);
1070
1071 public:
1072 const_iterator() {}
1073 const_iterator(const const_iterator& i) : m_cur(i.m_cur) {}
1074 const_iterator(const iterator& i) : m_cur(i.m_cur) {}
1075
1076 reference operator*() const
1077 { return wxStringOperations::DecodeChar(m_cur); }
1078
1079 const_iterator operator+(ptrdiff_t n) const
1080 { return const_iterator(wxStringOperations::AddToIter(m_cur, n)); }
1081 const_iterator operator-(ptrdiff_t n) const
1082 { return const_iterator(wxStringOperations::AddToIter(m_cur, -n)); }
1083
1084 private:
1085 // for internal wxString use only:
1086 const_iterator(underlying_iterator ptr) : m_cur(ptr) {}
1087 const_iterator(const wxString *WXUNUSED(str), underlying_iterator ptr)
1088 : m_cur(ptr) {}
1089 };
1090
1091 iterator GetIterForNthChar(size_t n) { return begin() + n; }
1092 const_iterator GetIterForNthChar(size_t n) const { return begin() + n; }
1093 #endif // wxUSE_UNICODE_UTF8/!wxUSE_UNICODE_UTF8
1094
1095 #undef WX_STR_ITERATOR_TAG
1096 #undef WX_STR_ITERATOR_IMPL
1097
1098 friend class iterator;
1099 friend class const_iterator;
1100
1101 template <typename T>
1102 class reverse_iterator_impl
1103 {
1104 public:
1105 typedef T iterator_type;
1106
1107 WX_DEFINE_ITERATOR_CATEGORY(typename T::iterator_category)
1108 typedef typename T::value_type value_type;
1109 typedef typename T::difference_type difference_type;
1110 typedef typename T::reference reference;
1111 typedef typename T::pointer *pointer;
1112
1113 reverse_iterator_impl() {}
1114 reverse_iterator_impl(iterator_type i) : m_cur(i) {}
1115 reverse_iterator_impl(const reverse_iterator_impl& ri)
1116 : m_cur(ri.m_cur) {}
1117
1118 iterator_type base() const { return m_cur; }
1119
1120 reference operator*() const { return *(m_cur-1); }
1121 reference operator[](size_t n) const { return *(*this + n); }
1122
1123 reverse_iterator_impl& operator++()
1124 { --m_cur; return *this; }
1125 reverse_iterator_impl operator++(int)
1126 { reverse_iterator_impl tmp = *this; --m_cur; return tmp; }
1127 reverse_iterator_impl& operator--()
1128 { ++m_cur; return *this; }
1129 reverse_iterator_impl operator--(int)
1130 { reverse_iterator_impl tmp = *this; ++m_cur; return tmp; }
1131
1132 // NB: explicit <T> in the functions below is to keep BCC 5.5 happy
1133 reverse_iterator_impl operator+(ptrdiff_t n) const
1134 { return reverse_iterator_impl<T>(m_cur - n); }
1135 reverse_iterator_impl operator-(ptrdiff_t n) const
1136 { return reverse_iterator_impl<T>(m_cur + n); }
1137 reverse_iterator_impl operator+=(ptrdiff_t n)
1138 { m_cur -= n; return *this; }
1139 reverse_iterator_impl operator-=(ptrdiff_t n)
1140 { m_cur += n; return *this; }
1141
1142 unsigned operator-(const reverse_iterator_impl& i) const
1143 { return i.m_cur - m_cur; }
1144
1145 bool operator==(const reverse_iterator_impl& ri) const
1146 { return m_cur == ri.m_cur; }
1147 bool operator!=(const reverse_iterator_impl& ri) const
1148 { return !(*this == ri); }
1149
1150 bool operator<(const reverse_iterator_impl& i) const
1151 { return m_cur > i.m_cur; }
1152 bool operator>(const reverse_iterator_impl& i) const
1153 { return m_cur < i.m_cur; }
1154 bool operator<=(const reverse_iterator_impl& i) const
1155 { return m_cur >= i.m_cur; }
1156 bool operator>=(const reverse_iterator_impl& i) const
1157 { return m_cur <= i.m_cur; }
1158
1159 private:
1160 iterator_type m_cur;
1161 };
1162
1163 typedef reverse_iterator_impl<iterator> reverse_iterator;
1164 typedef reverse_iterator_impl<const_iterator> const_reverse_iterator;
1165
1166 private:
1167 // used to transform an expression built using c_str() (and hence of type
1168 // wxCStrData) to an iterator into the string
1169 static const_iterator CreateConstIterator(const wxCStrData& data)
1170 {
1171 return const_iterator(data.m_str,
1172 (data.m_str->begin() + data.m_offset).impl());
1173 }
1174
1175 // in UTF-8 STL build, creation from std::string requires conversion under
1176 // non-UTF8 locales, so we can't have and use wxString(wxStringImpl) ctor;
1177 // instead we define dummy type that lets us have wxString ctor for creation
1178 // from wxStringImpl that couldn't be used by user code (in all other builds,
1179 // "standard" ctors can be used):
1180 #if wxUSE_UNICODE_UTF8 && wxUSE_STL_BASED_WXSTRING
1181 struct CtorFromStringImplTag {};
1182
1183 wxString(CtorFromStringImplTag* WXUNUSED(dummy), const wxStringImpl& src)
1184 : m_impl(src) {}
1185
1186 static wxString FromImpl(const wxStringImpl& src)
1187 { return wxString((CtorFromStringImplTag*)NULL, src); }
1188 #else
1189 #if !wxUSE_STL_BASED_WXSTRING
1190 wxString(const wxStringImpl& src) : m_impl(src) { }
1191 // else: already defined as wxString(wxStdString) below
1192 #endif
1193 static wxString FromImpl(const wxStringImpl& src) { return wxString(src); }
1194 #endif
1195
1196 public:
1197 // constructors and destructor
1198 // ctor for an empty string
1199 wxString() {}
1200
1201 // copy ctor
1202 wxString(const wxString& stringSrc) : m_impl(stringSrc.m_impl) { }
1203
1204 // string containing nRepeat copies of ch
1205 wxString(wxUniChar ch, size_t nRepeat = 1 )
1206 { assign(nRepeat, ch); }
1207 wxString(size_t nRepeat, wxUniChar ch)
1208 { assign(nRepeat, ch); }
1209 wxString(wxUniCharRef ch, size_t nRepeat = 1)
1210 { assign(nRepeat, ch); }
1211 wxString(size_t nRepeat, wxUniCharRef ch)
1212 { assign(nRepeat, ch); }
1213 wxString(char ch, size_t nRepeat = 1)
1214 { assign(nRepeat, ch); }
1215 wxString(size_t nRepeat, char ch)
1216 { assign(nRepeat, ch); }
1217 wxString(wchar_t ch, size_t nRepeat = 1)
1218 { assign(nRepeat, ch); }
1219 wxString(size_t nRepeat, wchar_t ch)
1220 { assign(nRepeat, ch); }
1221
1222 // ctors from char* strings:
1223 wxString(const char *psz)
1224 : m_impl(ImplStr(psz)) {}
1225 wxString(const char *psz, const wxMBConv& conv)
1226 : m_impl(ImplStr(psz, conv)) {}
1227 wxString(const char *psz, size_t nLength)
1228 { assign(psz, nLength); }
1229 wxString(const char *psz, const wxMBConv& conv, size_t nLength)
1230 {
1231 SubstrBufFromMB str(ImplStr(psz, nLength, conv));
1232 m_impl.assign(str.data, str.len);
1233 }
1234
1235 // and unsigned char*:
1236 wxString(const unsigned char *psz)
1237 : m_impl(ImplStr((const char*)psz)) {}
1238 wxString(const unsigned char *psz, const wxMBConv& conv)
1239 : m_impl(ImplStr((const char*)psz, conv)) {}
1240 wxString(const unsigned char *psz, size_t nLength)
1241 { assign((const char*)psz, nLength); }
1242 wxString(const unsigned char *psz, const wxMBConv& conv, size_t nLength)
1243 {
1244 SubstrBufFromMB str(ImplStr((const char*)psz, nLength, conv));
1245 m_impl.assign(str.data, str.len);
1246 }
1247
1248 // ctors from wchar_t* strings:
1249 wxString(const wchar_t *pwz)
1250 : m_impl(ImplStr(pwz)) {}
1251 wxString(const wchar_t *pwz, const wxMBConv& WXUNUSED(conv))
1252 : m_impl(ImplStr(pwz)) {}
1253 wxString(const wchar_t *pwz, size_t nLength)
1254 { assign(pwz, nLength); }
1255 wxString(const wchar_t *pwz, const wxMBConv& WXUNUSED(conv), size_t nLength)
1256 { assign(pwz, nLength); }
1257
1258 wxString(const wxCharBuffer& buf)
1259 { assign(buf.data()); } // FIXME-UTF8: fix for embedded NUL and buffer length
1260 wxString(const wxWCharBuffer& buf)
1261 { assign(buf.data()); } // FIXME-UTF8: fix for embedded NUL and buffer length
1262
1263 // NB: this version uses m_impl.c_str() to force making a copy of the
1264 // string, so that "wxString(str.c_str())" idiom for passing strings
1265 // between threads works
1266 wxString(const wxCStrData& cstr)
1267 : m_impl(cstr.AsString().m_impl.c_str()) { }
1268
1269 // as we provide both ctors with this signature for both char and unsigned
1270 // char string, we need to provide one for wxCStrData to resolve ambiguity
1271 wxString(const wxCStrData& cstr, size_t nLength)
1272 : m_impl(cstr.AsString().Mid(0, nLength).m_impl) {}
1273
1274 // and because wxString is convertible to wxCStrData and const wxChar *
1275 // we also need to provide this one
1276 wxString(const wxString& str, size_t nLength)
1277 { assign(str, nLength); }
1278
1279
1280 #if wxUSE_STRING_POS_CACHE
1281 ~wxString()
1282 {
1283 // we need to invalidate our cache entry as another string could be
1284 // recreated at the same address (unlikely, but still possible, with the
1285 // heap-allocated strings but perfectly common with stack-allocated ones)
1286 InvalidateCache();
1287 }
1288 #endif // wxUSE_STRING_POS_CACHE
1289
1290 // even if we're not built with wxUSE_STL == 1 it is very convenient to allow
1291 // implicit conversions from std::string to wxString and vice verse as this
1292 // allows to use the same strings in non-GUI and GUI code, however we don't
1293 // want to unconditionally add this ctor as it would make wx lib dependent on
1294 // libstdc++ on some Linux versions which is bad, so instead we ask the
1295 // client code to define this wxUSE_STD_STRING symbol if they need it
1296 #if wxUSE_STD_STRING
1297 #if wxUSE_UNICODE_WCHAR
1298 wxString(const wxStdWideString& str) : m_impl(str) {}
1299 #else // UTF-8 or ANSI
1300 wxString(const wxStdWideString& str)
1301 { assign(str.c_str(), str.length()); }
1302 #endif
1303
1304 #if !wxUSE_UNICODE // ANSI build
1305 // FIXME-UTF8: do this in UTF8 build #if wxUSE_UTF8_LOCALE_ONLY, too
1306 wxString(const std::string& str) : m_impl(str) {}
1307 #else // Unicode
1308 wxString(const std::string& str)
1309 { assign(str.c_str(), str.length()); }
1310 #endif
1311 #endif // wxUSE_STD_STRING
1312
1313 // Unlike ctor from std::string, we provide conversion to std::string only
1314 // if wxUSE_STL and not merely wxUSE_STD_STRING (which is on by default),
1315 // because it conflicts with operator const char/wchar_t*:
1316 #if wxUSE_STL
1317 #if wxUSE_UNICODE_WCHAR && wxUSE_STL_BASED_WXSTRING
1318 // wxStringImpl is std::string in the encoding we want
1319 operator const wxStdWideString&() const { return m_impl; }
1320 #else
1321 // wxStringImpl is either not std::string or needs conversion
1322 operator wxStdWideString() const
1323 // FIXME-UTF8: broken for embedded NULs
1324 { return wxStdWideString(wc_str()); }
1325 #endif
1326
1327 #if (!wxUSE_UNICODE || wxUSE_UTF8_LOCALE_ONLY) && wxUSE_STL_BASED_WXSTRING
1328 // wxStringImpl is std::string in the encoding we want
1329 operator const std::string&() const { return m_impl; }
1330 #else
1331 // wxStringImpl is either not std::string or needs conversion
1332 operator std::string() const
1333 // FIXME-UTF8: broken for embedded NULs
1334 { return std::string(mb_str()); }
1335 #endif
1336 #endif // wxUSE_STL
1337
1338 wxString Clone() const
1339 {
1340 // make a deep copy of the string, i.e. the returned string will have
1341 // ref count = 1 with refcounted implementation
1342 return wxString::FromImpl(wxStringImpl(m_impl.c_str(), m_impl.length()));
1343 }
1344
1345 // first valid index position
1346 const_iterator begin() const { return const_iterator(this, m_impl.begin()); }
1347 iterator begin() { return iterator(this, m_impl.begin()); }
1348 // position one after the last valid one
1349 const_iterator end() const { return const_iterator(this, m_impl.end()); }
1350 iterator end() { return iterator(this, m_impl.end()); }
1351
1352 // first element of the reversed string
1353 const_reverse_iterator rbegin() const
1354 { return const_reverse_iterator(end()); }
1355 reverse_iterator rbegin()
1356 { return reverse_iterator(end()); }
1357 // one beyond the end of the reversed string
1358 const_reverse_iterator rend() const
1359 { return const_reverse_iterator(begin()); }
1360 reverse_iterator rend()
1361 { return reverse_iterator(begin()); }
1362
1363 // std::string methods:
1364 #if wxUSE_UNICODE_UTF8
1365 size_t length() const
1366 {
1367 #if wxUSE_STRING_POS_CACHE
1368 wxCACHE_PROFILE_FIELD_INC(lentot);
1369
1370 Cache::Element * const cache = GetCacheElement();
1371
1372 if ( cache->len == npos )
1373 {
1374 // it's probably not worth trying to be clever and using cache->pos
1375 // here as it's probably 0 anyhow -- you usually call length() before
1376 // starting to index the string
1377 cache->len = end() - begin();
1378 }
1379 else
1380 {
1381 wxCACHE_PROFILE_FIELD_INC(lenhits);
1382
1383 wxSTRING_CACHE_ASSERT( (int)cache->len == end() - begin() );
1384 }
1385
1386 return cache->len;
1387 #else // !wxUSE_STRING_POS_CACHE
1388 return end() - begin();
1389 #endif // wxUSE_STRING_POS_CACHE/!wxUSE_STRING_POS_CACHE
1390 }
1391 #else
1392 size_t length() const { return m_impl.length(); }
1393 #endif
1394
1395 size_type size() const { return length(); }
1396 size_type max_size() const { return npos; }
1397
1398 bool empty() const { return m_impl.empty(); }
1399
1400 // NB: these methods don't have a well-defined meaning in UTF-8 case
1401 size_type capacity() const { return m_impl.capacity(); }
1402 void reserve(size_t sz) { m_impl.reserve(sz); }
1403
1404 void resize(size_t nSize, wxUniChar ch = wxT('\0'))
1405 {
1406 const size_t len = length();
1407 if ( nSize == len)
1408 return;
1409
1410 #if wxUSE_UNICODE_UTF8
1411 if ( nSize < len )
1412 {
1413 wxSTRING_INVALIDATE_CACHE();
1414
1415 // we can't use wxStringImpl::resize() for truncating the string as it
1416 // counts in bytes, not characters
1417 erase(nSize);
1418 return;
1419 }
1420
1421 // we also can't use (presumably more efficient) resize() if we have to
1422 // append characters taking more than one byte
1423 if ( !ch.IsAscii() )
1424 {
1425 append(nSize - len, ch);
1426 }
1427 else // can use (presumably faster) resize() version
1428 #endif // wxUSE_UNICODE_UTF8
1429 {
1430 wxSTRING_INVALIDATE_CACHED_LENGTH();
1431
1432 m_impl.resize(nSize, (wxStringCharType)ch);
1433 }
1434 }
1435
1436 wxString substr(size_t nStart = 0, size_t nLen = npos) const
1437 {
1438 size_t pos, len;
1439 PosLenToImpl(nStart, nLen, &pos, &len);
1440 return FromImpl(m_impl.substr(pos, len));
1441 }
1442
1443 // generic attributes & operations
1444 // as standard strlen()
1445 size_t Len() const { return length(); }
1446 // string contains any characters?
1447 bool IsEmpty() const { return empty(); }
1448 // empty string is "false", so !str will return true
1449 bool operator!() const { return empty(); }
1450 // truncate the string to given length
1451 wxString& Truncate(size_t uiLen);
1452 // empty string contents
1453 void Empty() { clear(); }
1454 // empty the string and free memory
1455 void Clear() { clear(); }
1456
1457 // contents test
1458 // Is an ascii value
1459 bool IsAscii() const;
1460 // Is a number
1461 bool IsNumber() const;
1462 // Is a word
1463 bool IsWord() const;
1464
1465 // data access (all indexes are 0 based)
1466 // read access
1467 wxUniChar at(size_t n) const
1468 { return wxStringOperations::DecodeChar(m_impl.begin() + PosToImpl(n)); }
1469 wxUniChar GetChar(size_t n) const
1470 { return at(n); }
1471 // read/write access
1472 wxUniCharRef at(size_t n)
1473 { return *GetIterForNthChar(n); }
1474 wxUniCharRef GetWritableChar(size_t n)
1475 { return at(n); }
1476 // write access
1477 void SetChar(size_t n, wxUniChar ch)
1478 { at(n) = ch; }
1479
1480 // get last character
1481 wxUniChar Last() const
1482 {
1483 wxASSERT_MSG( !empty(), _T("wxString: index out of bounds") );
1484 return *rbegin();
1485 }
1486
1487 // get writable last character
1488 wxUniCharRef Last()
1489 {
1490 wxASSERT_MSG( !empty(), _T("wxString: index out of bounds") );
1491 return *rbegin();
1492 }
1493
1494 /*
1495 Note that we we must define all of the overloads below to avoid
1496 ambiguity when using str[0].
1497 */
1498 wxUniChar operator[](int n) const
1499 { return at(n); }
1500 wxUniChar operator[](long n) const
1501 { return at(n); }
1502 wxUniChar operator[](size_t n) const
1503 { return at(n); }
1504 #ifndef wxSIZE_T_IS_UINT
1505 wxUniChar operator[](unsigned int n) const
1506 { return at(n); }
1507 #endif // size_t != unsigned int
1508
1509 // operator versions of GetWriteableChar()
1510 wxUniCharRef operator[](int n)
1511 { return at(n); }
1512 wxUniCharRef operator[](long n)
1513 { return at(n); }
1514 wxUniCharRef operator[](size_t n)
1515 { return at(n); }
1516 #ifndef wxSIZE_T_IS_UINT
1517 wxUniCharRef operator[](unsigned int n)
1518 { return at(n); }
1519 #endif // size_t != unsigned int
1520
1521 // explicit conversion to C string (use this with printf()!)
1522 wxCStrData c_str() const { return wxCStrData(this); }
1523 wxCStrData data() const { return c_str(); }
1524
1525 // implicit conversion to C string
1526 operator wxCStrData() const { return c_str(); }
1527
1528 // the first two operators conflict with operators for conversion to
1529 // std::string and they must be disabled in STL build; the next one only
1530 // makes sense if conversions to char* are also defined and not defining it
1531 // in STL build also helps us to get more clear error messages for the code
1532 // which relies on implicit conversion to char* in STL build
1533 #if !wxUSE_STL
1534 operator const char*() const { return c_str(); }
1535 operator const wchar_t*() const { return c_str(); }
1536
1537 // implicit conversion to untyped pointer for compatibility with previous
1538 // wxWidgets versions: this is the same as conversion to const char * so it
1539 // may fail!
1540 operator const void*() const { return c_str(); }
1541 #endif // wxUSE_STL
1542
1543 // identical to c_str(), for MFC compatibility
1544 const wxCStrData GetData() const { return c_str(); }
1545
1546 // explicit conversion to C string in internal representation (char*,
1547 // wchar_t*, UTF-8-encoded char*, depending on the build):
1548 const wxStringCharType *wx_str() const { return m_impl.c_str(); }
1549
1550 // conversion to *non-const* multibyte or widestring buffer; modifying
1551 // returned buffer won't affect the string, these methods are only useful
1552 // for passing values to const-incorrect functions
1553 wxWritableCharBuffer char_str(const wxMBConv& conv = wxConvLibc) const
1554 { return mb_str(conv); }
1555 wxWritableWCharBuffer wchar_str() const { return wc_str(); }
1556
1557 // conversion to the buffer of the given type T (= char or wchar_t) and
1558 // also optionally return the buffer length
1559 //
1560 // this is mostly/only useful for the template functions
1561 //
1562 // FIXME-VC6: the second argument only exists for VC6 which doesn't support
1563 // explicit template function selection, do not use it unless
1564 // you must support VC6!
1565 template <typename T>
1566 wxCharTypeBuffer<T> tchar_str(size_t *len = NULL,
1567 T * WXUNUSED(dummy) = NULL) const
1568 {
1569 #if wxUSE_UNICODE
1570 // we need a helper dispatcher depending on type
1571 return wxPrivate::wxStringAsBufHelper<T>::Get(*this, len);
1572 #else // ANSI
1573 // T can only be char in ANSI build
1574 if ( len )
1575 *len = length();
1576
1577 return wxCharTypeBuffer<T>::CreateNonOwned(wx_str());
1578 #endif // Unicode build kind
1579 }
1580
1581 // conversion to/from plain (i.e. 7 bit) ASCII: this is useful for
1582 // converting numbers or strings which are certain not to contain special
1583 // chars (typically system functions, X atoms, environment variables etc.)
1584 //
1585 // the behaviour of these functions with the strings containing anything
1586 // else than 7 bit ASCII characters is undefined, use at your own risk.
1587 #if wxUSE_UNICODE
1588 static wxString FromAscii(const char *ascii, size_t len);
1589 static wxString FromAscii(const char *ascii);
1590 static wxString FromAscii(char ascii);
1591 const wxCharBuffer ToAscii() const;
1592 #else // ANSI
1593 static wxString FromAscii(const char *ascii) { return wxString( ascii ); }
1594 static wxString FromAscii(const char *ascii, size_t len)
1595 { return wxString( ascii, len ); }
1596 static wxString FromAscii(char ascii) { return wxString( ascii ); }
1597 const char *ToAscii() const { return c_str(); }
1598 #endif // Unicode/!Unicode
1599
1600 // also provide unsigned char overloads as signed/unsigned doesn't matter
1601 // for 7 bit ASCII characters
1602 static wxString FromAscii(const unsigned char *ascii)
1603 { return FromAscii((const char *)ascii); }
1604 static wxString FromAscii(const unsigned char *ascii, size_t len)
1605 { return FromAscii((const char *)ascii, len); }
1606
1607 // conversion to/from UTF-8:
1608 #if wxUSE_UNICODE_UTF8
1609 static wxString FromUTF8Unchecked(const char *utf8)
1610 {
1611 if ( !utf8 )
1612 return wxEmptyString;
1613
1614 wxASSERT( wxStringOperations::IsValidUtf8String(utf8) );
1615 return FromImpl(wxStringImpl(utf8));
1616 }
1617 static wxString FromUTF8Unchecked(const char *utf8, size_t len)
1618 {
1619 if ( !utf8 )
1620 return wxEmptyString;
1621 if ( len == npos )
1622 return FromUTF8Unchecked(utf8);
1623
1624 wxASSERT( wxStringOperations::IsValidUtf8String(utf8, len) );
1625 return FromImpl(wxStringImpl(utf8, len));
1626 }
1627
1628 static wxString FromUTF8(const char *utf8)
1629 {
1630 if ( !utf8 || !wxStringOperations::IsValidUtf8String(utf8) )
1631 return "";
1632
1633 return FromImpl(wxStringImpl(utf8));
1634 }
1635 static wxString FromUTF8(const char *utf8, size_t len)
1636 {
1637 if ( len == npos )
1638 return FromUTF8(utf8);
1639
1640 if ( !utf8 || !wxStringOperations::IsValidUtf8String(utf8, len) )
1641 return "";
1642
1643 return FromImpl(wxStringImpl(utf8, len));
1644 }
1645
1646 const char* utf8_str() const { return wx_str(); }
1647 const char* ToUTF8() const { return wx_str(); }
1648
1649 // this function exists in UTF-8 build only and returns the length of the
1650 // internal UTF-8 representation
1651 size_t utf8_length() const { return m_impl.length(); }
1652 #elif wxUSE_UNICODE_WCHAR
1653 static wxString FromUTF8(const char *utf8, size_t len = npos)
1654 { return wxString(utf8, wxMBConvUTF8(), len); }
1655 static wxString FromUTF8Unchecked(const char *utf8, size_t len = npos)
1656 {
1657 const wxString s(utf8, wxMBConvUTF8(), len);
1658 wxASSERT_MSG( !utf8 || !*utf8 || !s.empty(),
1659 "string must be valid UTF-8" );
1660 return s;
1661 }
1662 const wxCharBuffer utf8_str() const { return mb_str(wxMBConvUTF8()); }
1663 const wxCharBuffer ToUTF8() const { return utf8_str(); }
1664 #else // ANSI
1665 static wxString FromUTF8(const char *utf8)
1666 { return wxString(wxMBConvUTF8().cMB2WC(utf8)); }
1667 static wxString FromUTF8(const char *utf8, size_t len)
1668 {
1669 size_t wlen;
1670 wxWCharBuffer buf(wxMBConvUTF8().cMB2WC(utf8, len == npos ? wxNO_LEN : len, &wlen));
1671 return wxString(buf.data(), wlen);
1672 }
1673 static wxString FromUTF8Unchecked(const char *utf8, size_t len = npos)
1674 {
1675 size_t wlen;
1676 wxWCharBuffer buf(wxMBConvUTF8().cMB2WC(utf8,
1677 len == npos ? wxNO_LEN : len,
1678 &wlen));
1679 wxASSERT_MSG( !utf8 || !*utf8 || wlen,
1680 "string must be valid UTF-8" );
1681
1682 return wxString(buf.data(), wlen);
1683 }
1684 const wxCharBuffer utf8_str() const
1685 { return wxMBConvUTF8().cWC2MB(wc_str()); }
1686 const wxCharBuffer ToUTF8() const { return utf8_str(); }
1687 #endif
1688
1689 // functions for storing binary data in wxString:
1690 #if wxUSE_UNICODE
1691 static wxString From8BitData(const char *data, size_t len)
1692 { return wxString(data, wxConvISO8859_1, len); }
1693 // version for NUL-terminated data:
1694 static wxString From8BitData(const char *data)
1695 { return wxString(data, wxConvISO8859_1); }
1696 const wxCharBuffer To8BitData() const { return mb_str(wxConvISO8859_1); }
1697 #else // ANSI
1698 static wxString From8BitData(const char *data, size_t len)
1699 { return wxString(data, len); }
1700 // version for NUL-terminated data:
1701 static wxString From8BitData(const char *data)
1702 { return wxString(data); }
1703 const char *To8BitData() const { return c_str(); }
1704 #endif // Unicode/ANSI
1705
1706 // conversions with (possible) format conversions: have to return a
1707 // buffer with temporary data
1708 //
1709 // the functions defined (in either Unicode or ANSI) mode are mb_str() to
1710 // return an ANSI (multibyte) string, wc_str() to return a wide string and
1711 // fn_str() to return a string which should be used with the OS APIs
1712 // accepting the file names. The return value is always the same, but the
1713 // type differs because a function may either return pointer to the buffer
1714 // directly or have to use intermediate buffer for translation.
1715 #if wxUSE_UNICODE
1716
1717 #if wxUSE_UTF8_LOCALE_ONLY
1718 const char* mb_str() const { return wx_str(); }
1719 const wxCharBuffer mb_str(const wxMBConv& conv) const;
1720 #else
1721 const wxCharBuffer mb_str(const wxMBConv& conv = wxConvLibc) const;
1722 #endif
1723
1724 const wxWX2MBbuf mbc_str() const { return mb_str(*wxConvCurrent); }
1725
1726 #if wxUSE_UNICODE_WCHAR
1727 const wchar_t* wc_str() const { return wx_str(); }
1728 #elif wxUSE_UNICODE_UTF8
1729 const wxWCharBuffer wc_str() const;
1730 #endif
1731 // for compatibility with !wxUSE_UNICODE version
1732 const wxWX2WCbuf wc_str(const wxMBConv& WXUNUSED(conv)) const
1733 { return wc_str(); }
1734
1735 #if wxMBFILES
1736 const wxCharBuffer fn_str() const { return mb_str(wxConvFile); }
1737 #else // !wxMBFILES
1738 const wxWX2WCbuf fn_str() const { return wc_str(); }
1739 #endif // wxMBFILES/!wxMBFILES
1740
1741 #else // ANSI
1742 const wxChar* mb_str() const { return wx_str(); }
1743
1744 // for compatibility with wxUSE_UNICODE version
1745 const char* mb_str(const wxMBConv& WXUNUSED(conv)) const { return wx_str(); }
1746
1747 const wxWX2MBbuf mbc_str() const { return mb_str(); }
1748
1749 #if wxUSE_WCHAR_T
1750 const wxWCharBuffer wc_str(const wxMBConv& conv = wxConvLibc) const;
1751 #endif // wxUSE_WCHAR_T
1752 const wxCharBuffer fn_str() const { return wxConvFile.cWC2WX( wc_str( wxConvLibc ) ); }
1753 #endif // Unicode/ANSI
1754
1755 #if wxUSE_UNICODE_UTF8
1756 const wxWCharBuffer t_str() const { return wc_str(); }
1757 #elif wxUSE_UNICODE_WCHAR
1758 const wchar_t* t_str() const { return wx_str(); }
1759 #else
1760 const char* t_str() const { return wx_str(); }
1761 #endif
1762
1763
1764 // overloaded assignment
1765 // from another wxString
1766 wxString& operator=(const wxString& stringSrc)
1767 {
1768 if ( this != &stringSrc )
1769 {
1770 wxSTRING_INVALIDATE_CACHE();
1771
1772 m_impl = stringSrc.m_impl;
1773 }
1774
1775 return *this;
1776 }
1777
1778 wxString& operator=(const wxCStrData& cstr)
1779 { return *this = cstr.AsString(); }
1780 // from a character
1781 wxString& operator=(wxUniChar ch)
1782 {
1783 wxSTRING_INVALIDATE_CACHE();
1784
1785 #if wxUSE_UNICODE_UTF8
1786 if ( !ch.IsAscii() )
1787 m_impl = wxStringOperations::EncodeChar(ch);
1788 else
1789 #endif // wxUSE_UNICODE_UTF8
1790 m_impl = (wxStringCharType)ch;
1791 return *this;
1792 }
1793
1794 wxString& operator=(wxUniCharRef ch)
1795 { return operator=((wxUniChar)ch); }
1796 wxString& operator=(char ch)
1797 { return operator=(wxUniChar(ch)); }
1798 wxString& operator=(unsigned char ch)
1799 { return operator=(wxUniChar(ch)); }
1800 wxString& operator=(wchar_t ch)
1801 { return operator=(wxUniChar(ch)); }
1802 // from a C string - STL probably will crash on NULL,
1803 // so we need to compensate in that case
1804 #if wxUSE_STL_BASED_WXSTRING
1805 wxString& operator=(const char *psz)
1806 {
1807 wxSTRING_INVALIDATE_CACHE();
1808
1809 if ( psz )
1810 m_impl = ImplStr(psz);
1811 else
1812 clear();
1813
1814 return *this;
1815 }
1816
1817 wxString& operator=(const wchar_t *pwz)
1818 {
1819 wxSTRING_INVALIDATE_CACHE();
1820
1821 if ( pwz )
1822 m_impl = ImplStr(pwz);
1823 else
1824 clear();
1825
1826 return *this;
1827 }
1828 #else // !wxUSE_STL_BASED_WXSTRING
1829 wxString& operator=(const char *psz)
1830 {
1831 wxSTRING_INVALIDATE_CACHE();
1832
1833 m_impl = ImplStr(psz);
1834
1835 return *this;
1836 }
1837
1838 wxString& operator=(const wchar_t *pwz)
1839 {
1840 wxSTRING_INVALIDATE_CACHE();
1841
1842 m_impl = ImplStr(pwz);
1843
1844 return *this;
1845 }
1846 #endif // wxUSE_STL_BASED_WXSTRING/!wxUSE_STL_BASED_WXSTRING
1847
1848 wxString& operator=(const unsigned char *psz)
1849 { return operator=((const char*)psz); }
1850
1851 // from wxWCharBuffer
1852 wxString& operator=(const wxWCharBuffer& s)
1853 { return operator=(s.data()); } // FIXME-UTF8: fix for embedded NULs
1854 // from wxCharBuffer
1855 wxString& operator=(const wxCharBuffer& s)
1856 { return operator=(s.data()); } // FIXME-UTF8: fix for embedded NULs
1857
1858 // string concatenation
1859 // in place concatenation
1860 /*
1861 Concatenate and return the result. Note that the left to right
1862 associativity of << allows to write things like "str << str1 << str2
1863 << ..." (unlike with +=)
1864 */
1865 // string += string
1866 wxString& operator<<(const wxString& s)
1867 {
1868 #if WXWIN_COMPATIBILITY_2_8 && !wxUSE_STL_BASED_WXSTRING && !wxUSE_UNICODE_UTF8
1869 wxASSERT_MSG( s.IsValid(),
1870 _T("did you forget to call UngetWriteBuf()?") );
1871 #endif
1872
1873 append(s);
1874 return *this;
1875 }
1876 // string += C string
1877 wxString& operator<<(const char *psz)
1878 { append(psz); return *this; }
1879 wxString& operator<<(const wchar_t *pwz)
1880 { append(pwz); return *this; }
1881 wxString& operator<<(const wxCStrData& psz)
1882 { append(psz.AsString()); return *this; }
1883 // string += char
1884 wxString& operator<<(wxUniChar ch) { append(1, ch); return *this; }
1885 wxString& operator<<(wxUniCharRef ch) { append(1, ch); return *this; }
1886 wxString& operator<<(char ch) { append(1, ch); return *this; }
1887 wxString& operator<<(unsigned char ch) { append(1, ch); return *this; }
1888 wxString& operator<<(wchar_t ch) { append(1, ch); return *this; }
1889
1890 // string += buffer (i.e. from wxGetString)
1891 wxString& operator<<(const wxWCharBuffer& s)
1892 { return operator<<((const wchar_t *)s); }
1893 wxString& operator<<(const wxCharBuffer& s)
1894 { return operator<<((const char *)s); }
1895
1896 // string += C string
1897 wxString& Append(const wxString& s)
1898 {
1899 // test for empty() to share the string if possible
1900 if ( empty() )
1901 *this = s;
1902 else
1903 append(s);
1904 return *this;
1905 }
1906 wxString& Append(const char* psz)
1907 { append(psz); return *this; }
1908 wxString& Append(const wchar_t* pwz)
1909 { append(pwz); return *this; }
1910 wxString& Append(const wxCStrData& psz)
1911 { append(psz); return *this; }
1912 wxString& Append(const wxCharBuffer& psz)
1913 { append(psz); return *this; }
1914 wxString& Append(const wxWCharBuffer& psz)
1915 { append(psz); return *this; }
1916 wxString& Append(const char* psz, size_t nLen)
1917 { append(psz, nLen); return *this; }
1918 wxString& Append(const wchar_t* pwz, size_t nLen)
1919 { append(pwz, nLen); return *this; }
1920 wxString& Append(const wxCStrData& psz, size_t nLen)
1921 { append(psz, nLen); return *this; }
1922 wxString& Append(const wxCharBuffer& psz, size_t nLen)
1923 { append(psz, nLen); return *this; }
1924 wxString& Append(const wxWCharBuffer& psz, size_t nLen)
1925 { append(psz, nLen); return *this; }
1926 // append count copies of given character
1927 wxString& Append(wxUniChar ch, size_t count = 1u)
1928 { append(count, ch); return *this; }
1929 wxString& Append(wxUniCharRef ch, size_t count = 1u)
1930 { append(count, ch); return *this; }
1931 wxString& Append(char ch, size_t count = 1u)
1932 { append(count, ch); return *this; }
1933 wxString& Append(unsigned char ch, size_t count = 1u)
1934 { append(count, ch); return *this; }
1935 wxString& Append(wchar_t ch, size_t count = 1u)
1936 { append(count, ch); return *this; }
1937
1938 // prepend a string, return the string itself
1939 wxString& Prepend(const wxString& str)
1940 { *this = str + *this; return *this; }
1941
1942 // non-destructive concatenation
1943 // two strings
1944 friend wxString WXDLLIMPEXP_BASE operator+(const wxString& string1,
1945 const wxString& string2);
1946 // string with a single char
1947 friend wxString WXDLLIMPEXP_BASE operator+(const wxString& string, wxUniChar ch);
1948 // char with a string
1949 friend wxString WXDLLIMPEXP_BASE operator+(wxUniChar ch, const wxString& string);
1950 // string with C string
1951 friend wxString WXDLLIMPEXP_BASE operator+(const wxString& string,
1952 const char *psz);
1953 friend wxString WXDLLIMPEXP_BASE operator+(const wxString& string,
1954 const wchar_t *pwz);
1955 // C string with string
1956 friend wxString WXDLLIMPEXP_BASE operator+(const char *psz,
1957 const wxString& string);
1958 friend wxString WXDLLIMPEXP_BASE operator+(const wchar_t *pwz,
1959 const wxString& string);
1960
1961 // stream-like functions
1962 // insert an int into string
1963 wxString& operator<<(int i)
1964 { return (*this) << Format(_T("%d"), i); }
1965 // insert an unsigned int into string
1966 wxString& operator<<(unsigned int ui)
1967 { return (*this) << Format(_T("%u"), ui); }
1968 // insert a long into string
1969 wxString& operator<<(long l)
1970 { return (*this) << Format(_T("%ld"), l); }
1971 // insert an unsigned long into string
1972 wxString& operator<<(unsigned long ul)
1973 { return (*this) << Format(_T("%lu"), ul); }
1974 #if defined wxLongLong_t && !defined wxLongLongIsLong
1975 // insert a long long if they exist and aren't longs
1976 wxString& operator<<(wxLongLong_t ll)
1977 {
1978 const wxChar *fmt = _T("%") wxLongLongFmtSpec _T("d");
1979 return (*this) << Format(fmt, ll);
1980 }
1981 // insert an unsigned long long
1982 wxString& operator<<(wxULongLong_t ull)
1983 {
1984 const wxChar *fmt = _T("%") wxLongLongFmtSpec _T("u");
1985 return (*this) << Format(fmt , ull);
1986 }
1987 #endif // wxLongLong_t && !wxLongLongIsLong
1988 // insert a float into string
1989 wxString& operator<<(float f)
1990 { return (*this) << Format(_T("%f"), f); }
1991 // insert a double into string
1992 wxString& operator<<(double d)
1993 { return (*this) << Format(_T("%g"), d); }
1994
1995 // string comparison
1996 // case-sensitive comparison (returns a value < 0, = 0 or > 0)
1997 int Cmp(const char *psz) const
1998 { return compare(psz); }
1999 int Cmp(const wchar_t *pwz) const
2000 { return compare(pwz); }
2001 int Cmp(const wxString& s) const
2002 { return compare(s); }
2003 int Cmp(const wxCStrData& s) const
2004 { return compare(s); }
2005 int Cmp(const wxCharBuffer& s) const
2006 { return compare(s); }
2007 int Cmp(const wxWCharBuffer& s) const
2008 { return compare(s); }
2009 // same as Cmp() but not case-sensitive
2010 int CmpNoCase(const wxString& s) const;
2011
2012 // test for the string equality, either considering case or not
2013 // (if compareWithCase then the case matters)
2014 bool IsSameAs(const wxString& str, bool compareWithCase = true) const
2015 {
2016 #if !wxUSE_UNICODE_UTF8
2017 // in UTF-8 build, length() is O(n) and doing this would be _slower_
2018 if ( length() != str.length() )
2019 return false;
2020 #endif
2021 return (compareWithCase ? Cmp(str) : CmpNoCase(str)) == 0;
2022 }
2023 bool IsSameAs(const char *str, bool compareWithCase = true) const
2024 { return (compareWithCase ? Cmp(str) : CmpNoCase(str)) == 0; }
2025 bool IsSameAs(const wchar_t *str, bool compareWithCase = true) const
2026 { return (compareWithCase ? Cmp(str) : CmpNoCase(str)) == 0; }
2027
2028 bool IsSameAs(const wxCStrData& str, bool compareWithCase = true) const
2029 { return IsSameAs(str.AsString(), compareWithCase); }
2030 bool IsSameAs(const wxCharBuffer& str, bool compareWithCase = true) const
2031 { return IsSameAs(str.data(), compareWithCase); }
2032 bool IsSameAs(const wxWCharBuffer& str, bool compareWithCase = true) const
2033 { return IsSameAs(str.data(), compareWithCase); }
2034 // comparison with a single character: returns true if equal
2035 bool IsSameAs(wxUniChar c, bool compareWithCase = true) const;
2036 // FIXME-UTF8: remove these overloads
2037 bool IsSameAs(wxUniCharRef c, bool compareWithCase = true) const
2038 { return IsSameAs(wxUniChar(c), compareWithCase); }
2039 bool IsSameAs(char c, bool compareWithCase = true) const
2040 { return IsSameAs(wxUniChar(c), compareWithCase); }
2041 bool IsSameAs(unsigned char c, bool compareWithCase = true) const
2042 { return IsSameAs(wxUniChar(c), compareWithCase); }
2043 bool IsSameAs(wchar_t c, bool compareWithCase = true) const
2044 { return IsSameAs(wxUniChar(c), compareWithCase); }
2045 bool IsSameAs(int c, bool compareWithCase = true) const
2046 { return IsSameAs(wxUniChar(c), compareWithCase); }
2047
2048 // simple sub-string extraction
2049 // return substring starting at nFirst of length nCount (or till the end
2050 // if nCount = default value)
2051 wxString Mid(size_t nFirst, size_t nCount = npos) const;
2052
2053 // operator version of Mid()
2054 wxString operator()(size_t start, size_t len) const
2055 { return Mid(start, len); }
2056
2057 // check if the string starts with the given prefix and return the rest
2058 // of the string in the provided pointer if it is not NULL; otherwise
2059 // return false
2060 bool StartsWith(const wxString& prefix, wxString *rest = NULL) const;
2061 // check if the string ends with the given suffix and return the
2062 // beginning of the string before the suffix in the provided pointer if
2063 // it is not NULL; otherwise return false
2064 bool EndsWith(const wxString& suffix, wxString *rest = NULL) const;
2065
2066 // get first nCount characters
2067 wxString Left(size_t nCount) const;
2068 // get last nCount characters
2069 wxString Right(size_t nCount) const;
2070 // get all characters before the first occurance of ch
2071 // (returns the whole string if ch not found)
2072 wxString BeforeFirst(wxUniChar ch) const;
2073 // get all characters before the last occurence of ch
2074 // (returns empty string if ch not found)
2075 wxString BeforeLast(wxUniChar ch) const;
2076 // get all characters after the first occurence of ch
2077 // (returns empty string if ch not found)
2078 wxString AfterFirst(wxUniChar ch) const;
2079 // get all characters after the last occurence of ch
2080 // (returns the whole string if ch not found)
2081 wxString AfterLast(wxUniChar ch) const;
2082
2083 // for compatibility only, use more explicitly named functions above
2084 wxString Before(wxUniChar ch) const { return BeforeLast(ch); }
2085 wxString After(wxUniChar ch) const { return AfterFirst(ch); }
2086
2087 // case conversion
2088 // convert to upper case in place, return the string itself
2089 wxString& MakeUpper();
2090 // convert to upper case, return the copy of the string
2091 wxString Upper() const { return wxString(*this).MakeUpper(); }
2092 // convert to lower case in place, return the string itself
2093 wxString& MakeLower();
2094 // convert to lower case, return the copy of the string
2095 wxString Lower() const { return wxString(*this).MakeLower(); }
2096 // convert the first character to the upper case and the rest to the
2097 // lower one, return the modified string itself
2098 wxString& MakeCapitalized();
2099 // convert the first character to the upper case and the rest to the
2100 // lower one, return the copy of the string
2101 wxString Capitalize() const { return wxString(*this).MakeCapitalized(); }
2102
2103 // trimming/padding whitespace (either side) and truncating
2104 // remove spaces from left or from right (default) side
2105 wxString& Trim(bool bFromRight = true);
2106 // add nCount copies chPad in the beginning or at the end (default)
2107 wxString& Pad(size_t nCount, wxUniChar chPad = wxT(' '), bool bFromRight = true);
2108
2109 // searching and replacing
2110 // searching (return starting index, or -1 if not found)
2111 int Find(wxUniChar ch, bool bFromEnd = false) const; // like strchr/strrchr
2112 int Find(wxUniCharRef ch, bool bFromEnd = false) const
2113 { return Find(wxUniChar(ch), bFromEnd); }
2114 int Find(char ch, bool bFromEnd = false) const
2115 { return Find(wxUniChar(ch), bFromEnd); }
2116 int Find(unsigned char ch, bool bFromEnd = false) const
2117 { return Find(wxUniChar(ch), bFromEnd); }
2118 int Find(wchar_t ch, bool bFromEnd = false) const
2119 { return Find(wxUniChar(ch), bFromEnd); }
2120 // searching (return starting index, or -1 if not found)
2121 int Find(const wxString& sub) const // like strstr
2122 {
2123 size_type idx = find(sub);
2124 return (idx == npos) ? wxNOT_FOUND : (int)idx;
2125 }
2126 int Find(const char *sub) const // like strstr
2127 {
2128 size_type idx = find(sub);
2129 return (idx == npos) ? wxNOT_FOUND : (int)idx;
2130 }
2131 int Find(const wchar_t *sub) const // like strstr
2132 {
2133 size_type idx = find(sub);
2134 return (idx == npos) ? wxNOT_FOUND : (int)idx;
2135 }
2136
2137 int Find(const wxCStrData& sub) const
2138 { return Find(sub.AsString()); }
2139 int Find(const wxCharBuffer& sub) const
2140 { return Find(sub.data()); }
2141 int Find(const wxWCharBuffer& sub) const
2142 { return Find(sub.data()); }
2143
2144 // replace first (or all of bReplaceAll) occurences of substring with
2145 // another string, returns the number of replacements made
2146 size_t Replace(const wxString& strOld,
2147 const wxString& strNew,
2148 bool bReplaceAll = true);
2149
2150 // check if the string contents matches a mask containing '*' and '?'
2151 bool Matches(const wxString& mask) const;
2152
2153 // conversion to numbers: all functions return true only if the whole
2154 // string is a number and put the value of this number into the pointer
2155 // provided, the base is the numeric base in which the conversion should be
2156 // done and must be comprised between 2 and 36 or be 0 in which case the
2157 // standard C rules apply (leading '0' => octal, "0x" => hex)
2158 // convert to a signed integer
2159 bool ToLong(long *val, int base = 10) const;
2160 // convert to an unsigned integer
2161 bool ToULong(unsigned long *val, int base = 10) const;
2162 // convert to wxLongLong
2163 #if defined(wxLongLong_t)
2164 bool ToLongLong(wxLongLong_t *val, int base = 10) const;
2165 // convert to wxULongLong
2166 bool ToULongLong(wxULongLong_t *val, int base = 10) const;
2167 #endif // wxLongLong_t
2168 // convert to a double
2169 bool ToDouble(double *val) const;
2170
2171
2172 #ifndef wxNEEDS_WXSTRING_PRINTF_MIXIN
2173 // formatted input/output
2174 // as sprintf(), returns the number of characters written or < 0 on error
2175 // (take 'this' into account in attribute parameter count)
2176 // int Printf(const wxString& format, ...);
2177 WX_DEFINE_VARARG_FUNC(int, Printf, 1, (const wxFormatString&),
2178 DoPrintfWchar, DoPrintfUtf8)
2179 #ifdef __WATCOMC__
2180 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
2181 WX_VARARG_WATCOM_WORKAROUND(int, Printf, 1, (const wxString&),
2182 (wxFormatString(f1)));
2183 WX_VARARG_WATCOM_WORKAROUND(int, Printf, 1, (const wxCStrData&),
2184 (wxFormatString(f1)));
2185 WX_VARARG_WATCOM_WORKAROUND(int, Printf, 1, (const char*),
2186 (wxFormatString(f1)));
2187 WX_VARARG_WATCOM_WORKAROUND(int, Printf, 1, (const wchar_t*),
2188 (wxFormatString(f1)));
2189 #endif
2190 #endif // !wxNEEDS_WXSTRING_PRINTF_MIXIN
2191 // as vprintf(), returns the number of characters written or < 0 on error
2192 int PrintfV(const wxString& format, va_list argptr);
2193
2194 #ifndef wxNEEDS_WXSTRING_PRINTF_MIXIN
2195 // returns the string containing the result of Printf() to it
2196 // static wxString Format(const wxString& format, ...) ATTRIBUTE_PRINTF_1;
2197 WX_DEFINE_VARARG_FUNC(static wxString, Format, 1, (const wxFormatString&),
2198 DoFormatWchar, DoFormatUtf8)
2199 #ifdef __WATCOMC__
2200 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
2201 WX_VARARG_WATCOM_WORKAROUND(static wxString, Format, 1, (const wxString&),
2202 (wxFormatString(f1)));
2203 WX_VARARG_WATCOM_WORKAROUND(static wxString, Format, 1, (const wxCStrData&),
2204 (wxFormatString(f1)));
2205 WX_VARARG_WATCOM_WORKAROUND(static wxString, Format, 1, (const char*),
2206 (wxFormatString(f1)));
2207 WX_VARARG_WATCOM_WORKAROUND(static wxString, Format, 1, (const wchar_t*),
2208 (wxFormatString(f1)));
2209 #endif
2210 #endif
2211 // the same as above, but takes a va_list
2212 static wxString FormatV(const wxString& format, va_list argptr);
2213
2214 // raw access to string memory
2215 // ensure that string has space for at least nLen characters
2216 // only works if the data of this string is not shared
2217 bool Alloc(size_t nLen) { reserve(nLen); return capacity() >= nLen; }
2218 // minimize the string's memory
2219 // only works if the data of this string is not shared
2220 bool Shrink();
2221 #if WXWIN_COMPATIBILITY_2_8 && !wxUSE_STL_BASED_WXSTRING && !wxUSE_UNICODE_UTF8
2222 // These are deprecated, use wxStringBuffer or wxStringBufferLength instead
2223 //
2224 // get writable buffer of at least nLen bytes. Unget() *must* be called
2225 // a.s.a.p. to put string back in a reasonable state!
2226 wxDEPRECATED( wxStringCharType *GetWriteBuf(size_t nLen) );
2227 // call this immediately after GetWriteBuf() has been used
2228 wxDEPRECATED( void UngetWriteBuf() );
2229 wxDEPRECATED( void UngetWriteBuf(size_t nLen) );
2230 #endif // WXWIN_COMPATIBILITY_2_8 && !wxUSE_STL_BASED_WXSTRING && wxUSE_UNICODE_UTF8
2231
2232 // wxWidgets version 1 compatibility functions
2233
2234 // use Mid()
2235 wxString SubString(size_t from, size_t to) const
2236 { return Mid(from, (to - from + 1)); }
2237 // values for second parameter of CompareTo function
2238 enum caseCompare {exact, ignoreCase};
2239 // values for first parameter of Strip function
2240 enum stripType {leading = 0x1, trailing = 0x2, both = 0x3};
2241
2242 #ifndef wxNEEDS_WXSTRING_PRINTF_MIXIN
2243 // use Printf()
2244 // (take 'this' into account in attribute parameter count)
2245 // int sprintf(const wxString& format, ...) ATTRIBUTE_PRINTF_2;
2246 WX_DEFINE_VARARG_FUNC(int, sprintf, 1, (const wxFormatString&),
2247 DoPrintfWchar, DoPrintfUtf8)
2248 #ifdef __WATCOMC__
2249 // workaround for http://bugzilla.openwatcom.org/show_bug.cgi?id=351
2250 WX_VARARG_WATCOM_WORKAROUND(int, sprintf, 1, (const wxString&),
2251 (wxFormatString(f1)));
2252 WX_VARARG_WATCOM_WORKAROUND(int, sprintf, 1, (const wxCStrData&),
2253 (wxFormatString(f1)));
2254 WX_VARARG_WATCOM_WORKAROUND(int, sprintf, 1, (const char*),
2255 (wxFormatString(f1)));
2256 WX_VARARG_WATCOM_WORKAROUND(int, sprintf, 1, (const wchar_t*),
2257 (wxFormatString(f1)));
2258 #endif
2259 #endif // wxNEEDS_WXSTRING_PRINTF_MIXIN
2260
2261 // use Cmp()
2262 int CompareTo(const wxChar* psz, caseCompare cmp = exact) const
2263 { return cmp == exact ? Cmp(psz) : CmpNoCase(psz); }
2264
2265 // use length()
2266 size_t Length() const { return length(); }
2267 // Count the number of characters
2268 int Freq(wxUniChar ch) const;
2269 // use MakeLower
2270 void LowerCase() { MakeLower(); }
2271 // use MakeUpper
2272 void UpperCase() { MakeUpper(); }
2273 // use Trim except that it doesn't change this string
2274 wxString Strip(stripType w = trailing) const;
2275
2276 // use Find (more general variants not yet supported)
2277 size_t Index(const wxChar* psz) const { return Find(psz); }
2278 size_t Index(wxUniChar ch) const { return Find(ch); }
2279 // use Truncate
2280 wxString& Remove(size_t pos) { return Truncate(pos); }
2281 wxString& RemoveLast(size_t n = 1) { return Truncate(length() - n); }
2282
2283 wxString& Remove(size_t nStart, size_t nLen)
2284 { return (wxString&)erase( nStart, nLen ); }
2285
2286 // use Find()
2287 int First( wxUniChar ch ) const { return Find(ch); }
2288 int First( wxUniCharRef ch ) const { return Find(ch); }
2289 int First( char ch ) const { return Find(ch); }
2290 int First( unsigned char ch ) const { return Find(ch); }
2291 int First( wchar_t ch ) const { return Find(ch); }
2292 int First( const wxString& str ) const { return Find(str); }
2293 int Last( wxUniChar ch ) const { return Find(ch, true); }
2294 bool Contains(const wxString& str) const { return Find(str) != wxNOT_FOUND; }
2295
2296 // use empty()
2297 bool IsNull() const { return empty(); }
2298
2299 // std::string compatibility functions
2300
2301 // take nLen chars starting at nPos
2302 wxString(const wxString& str, size_t nPos, size_t nLen)
2303 { assign(str, nPos, nLen); }
2304 // take all characters from first to last
2305 wxString(const_iterator first, const_iterator last)
2306 : m_impl(first.impl(), last.impl()) { }
2307 #if WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2308 // the 2 overloads below are for compatibility with the existing code using
2309 // pointers instead of iterators
2310 wxString(const char *first, const char *last)
2311 {
2312 SubstrBufFromMB str(ImplStr(first, last - first));
2313 m_impl.assign(str.data, str.len);
2314 }
2315 wxString(const wchar_t *first, const wchar_t *last)
2316 {
2317 SubstrBufFromWC str(ImplStr(first, last - first));
2318 m_impl.assign(str.data, str.len);
2319 }
2320 // and this one is needed to compile code adding offsets to c_str() result
2321 wxString(const wxCStrData& first, const wxCStrData& last)
2322 : m_impl(CreateConstIterator(first).impl(),
2323 CreateConstIterator(last).impl())
2324 {
2325 wxASSERT_MSG( first.m_str == last.m_str,
2326 _T("pointers must be into the same string") );
2327 }
2328 #endif // WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2329
2330 // lib.string.modifiers
2331 // append elements str[pos], ..., str[pos+n]
2332 wxString& append(const wxString& str, size_t pos, size_t n)
2333 {
2334 wxSTRING_UPDATE_CACHED_LENGTH(n);
2335
2336 size_t from, len;
2337 str.PosLenToImpl(pos, n, &from, &len);
2338 m_impl.append(str.m_impl, from, len);
2339 return *this;
2340 }
2341 // append a string
2342 wxString& append(const wxString& str)
2343 {
2344 wxSTRING_UPDATE_CACHED_LENGTH(str.length());
2345
2346 m_impl.append(str.m_impl);
2347 return *this;
2348 }
2349
2350 // append first n (or all if n == npos) characters of sz
2351 wxString& append(const char *sz)
2352 {
2353 wxSTRING_INVALIDATE_CACHED_LENGTH();
2354
2355 m_impl.append(ImplStr(sz));
2356 return *this;
2357 }
2358
2359 wxString& append(const wchar_t *sz)
2360 {
2361 wxSTRING_INVALIDATE_CACHED_LENGTH();
2362
2363 m_impl.append(ImplStr(sz));
2364 return *this;
2365 }
2366
2367 wxString& append(const char *sz, size_t n)
2368 {
2369 wxSTRING_INVALIDATE_CACHED_LENGTH();
2370
2371 SubstrBufFromMB str(ImplStr(sz, n));
2372 m_impl.append(str.data, str.len);
2373 return *this;
2374 }
2375 wxString& append(const wchar_t *sz, size_t n)
2376 {
2377 wxSTRING_UPDATE_CACHED_LENGTH(n);
2378
2379 SubstrBufFromWC str(ImplStr(sz, n));
2380 m_impl.append(str.data, str.len);
2381 return *this;
2382 }
2383
2384 wxString& append(const wxCStrData& str)
2385 { return append(str.AsString()); }
2386 wxString& append(const wxCharBuffer& str)
2387 { return append(str.data()); }
2388 wxString& append(const wxWCharBuffer& str)
2389 { return append(str.data()); }
2390 wxString& append(const wxCStrData& str, size_t n)
2391 { return append(str.AsString(), 0, n); }
2392 wxString& append(const wxCharBuffer& str, size_t n)
2393 { return append(str.data(), n); }
2394 wxString& append(const wxWCharBuffer& str, size_t n)
2395 { return append(str.data(), n); }
2396
2397 // append n copies of ch
2398 wxString& append(size_t n, wxUniChar ch)
2399 {
2400 #if wxUSE_UNICODE_UTF8
2401 if ( !ch.IsAscii() )
2402 {
2403 wxSTRING_INVALIDATE_CACHED_LENGTH();
2404
2405 m_impl.append(wxStringOperations::EncodeNChars(n, ch));
2406 }
2407 else // ASCII
2408 #endif
2409 {
2410 wxSTRING_UPDATE_CACHED_LENGTH(n);
2411
2412 m_impl.append(n, (wxStringCharType)ch);
2413 }
2414
2415 return *this;
2416 }
2417
2418 wxString& append(size_t n, wxUniCharRef ch)
2419 { return append(n, wxUniChar(ch)); }
2420 wxString& append(size_t n, char ch)
2421 { return append(n, wxUniChar(ch)); }
2422 wxString& append(size_t n, unsigned char ch)
2423 { return append(n, wxUniChar(ch)); }
2424 wxString& append(size_t n, wchar_t ch)
2425 { return append(n, wxUniChar(ch)); }
2426
2427 // append from first to last
2428 wxString& append(const_iterator first, const_iterator last)
2429 {
2430 wxSTRING_INVALIDATE_CACHED_LENGTH();
2431
2432 m_impl.append(first.impl(), last.impl());
2433 return *this;
2434 }
2435 #if WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2436 wxString& append(const char *first, const char *last)
2437 { return append(first, last - first); }
2438 wxString& append(const wchar_t *first, const wchar_t *last)
2439 { return append(first, last - first); }
2440 wxString& append(const wxCStrData& first, const wxCStrData& last)
2441 { return append(CreateConstIterator(first), CreateConstIterator(last)); }
2442 #endif // WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2443
2444 // same as `this_string = str'
2445 wxString& assign(const wxString& str)
2446 {
2447 wxSTRING_SET_CACHED_LENGTH(str.length());
2448
2449 m_impl = str.m_impl;
2450
2451 return *this;
2452 }
2453
2454 wxString& assign(const wxString& str, size_t len)
2455 {
2456 wxSTRING_SET_CACHED_LENGTH(len);
2457
2458 m_impl.assign(str.m_impl, 0, str.LenToImpl(len));
2459
2460 return *this;
2461 }
2462
2463 // same as ` = str[pos..pos + n]
2464 wxString& assign(const wxString& str, size_t pos, size_t n)
2465 {
2466 size_t from, len;
2467 str.PosLenToImpl(pos, n, &from, &len);
2468 m_impl.assign(str.m_impl, from, len);
2469
2470 // it's important to call this after PosLenToImpl() above in case str is
2471 // the same string as this one
2472 wxSTRING_SET_CACHED_LENGTH(n);
2473
2474 return *this;
2475 }
2476
2477 // same as `= first n (or all if n == npos) characters of sz'
2478 wxString& assign(const char *sz)
2479 {
2480 wxSTRING_INVALIDATE_CACHE();
2481
2482 m_impl.assign(ImplStr(sz));
2483
2484 return *this;
2485 }
2486
2487 wxString& assign(const wchar_t *sz)
2488 {
2489 wxSTRING_INVALIDATE_CACHE();
2490
2491 m_impl.assign(ImplStr(sz));
2492
2493 return *this;
2494 }
2495
2496 wxString& assign(const char *sz, size_t n)
2497 {
2498 wxSTRING_SET_CACHED_LENGTH(n);
2499
2500 SubstrBufFromMB str(ImplStr(sz, n));
2501 m_impl.assign(str.data, str.len);
2502
2503 return *this;
2504 }
2505
2506 wxString& assign(const wchar_t *sz, size_t n)
2507 {
2508 wxSTRING_SET_CACHED_LENGTH(n);
2509
2510 SubstrBufFromWC str(ImplStr(sz, n));
2511 m_impl.assign(str.data, str.len);
2512
2513 return *this;
2514 }
2515
2516 wxString& assign(const wxCStrData& str)
2517 { return assign(str.AsString()); }
2518 wxString& assign(const wxCharBuffer& str)
2519 { return assign(str.data()); }
2520 wxString& assign(const wxWCharBuffer& str)
2521 { return assign(str.data()); }
2522 wxString& assign(const wxCStrData& str, size_t len)
2523 { return assign(str.AsString(), len); }
2524 wxString& assign(const wxCharBuffer& str, size_t len)
2525 { return assign(str.data(), len); }
2526 wxString& assign(const wxWCharBuffer& str, size_t len)
2527 { return assign(str.data(), len); }
2528
2529 // same as `= n copies of ch'
2530 wxString& assign(size_t n, wxUniChar ch)
2531 {
2532 wxSTRING_SET_CACHED_LENGTH(n);
2533
2534 #if wxUSE_UNICODE_UTF8
2535 if ( !ch.IsAscii() )
2536 m_impl.assign(wxStringOperations::EncodeNChars(n, ch));
2537 else
2538 #endif
2539 m_impl.assign(n, (wxStringCharType)ch);
2540
2541 return *this;
2542 }
2543
2544 wxString& assign(size_t n, wxUniCharRef ch)
2545 { return assign(n, wxUniChar(ch)); }
2546 wxString& assign(size_t n, char ch)
2547 { return assign(n, wxUniChar(ch)); }
2548 wxString& assign(size_t n, unsigned char ch)
2549 { return assign(n, wxUniChar(ch)); }
2550 wxString& assign(size_t n, wchar_t ch)
2551 { return assign(n, wxUniChar(ch)); }
2552
2553 // assign from first to last
2554 wxString& assign(const_iterator first, const_iterator last)
2555 {
2556 wxSTRING_INVALIDATE_CACHE();
2557
2558 m_impl.assign(first.impl(), last.impl());
2559
2560 return *this;
2561 }
2562 #if WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2563 wxString& assign(const char *first, const char *last)
2564 { return assign(first, last - first); }
2565 wxString& assign(const wchar_t *first, const wchar_t *last)
2566 { return assign(first, last - first); }
2567 wxString& assign(const wxCStrData& first, const wxCStrData& last)
2568 { return assign(CreateConstIterator(first), CreateConstIterator(last)); }
2569 #endif // WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2570
2571 // string comparison
2572 int compare(const wxString& str) const;
2573 int compare(const char* sz) const;
2574 int compare(const wchar_t* sz) const;
2575 int compare(const wxCStrData& str) const
2576 { return compare(str.AsString()); }
2577 int compare(const wxCharBuffer& str) const
2578 { return compare(str.data()); }
2579 int compare(const wxWCharBuffer& str) const
2580 { return compare(str.data()); }
2581 // comparison with a substring
2582 int compare(size_t nStart, size_t nLen, const wxString& str) const;
2583 // comparison of 2 substrings
2584 int compare(size_t nStart, size_t nLen,
2585 const wxString& str, size_t nStart2, size_t nLen2) const;
2586 // substring comparison with first nCount characters of sz
2587 int compare(size_t nStart, size_t nLen,
2588 const char* sz, size_t nCount = npos) const;
2589 int compare(size_t nStart, size_t nLen,
2590 const wchar_t* sz, size_t nCount = npos) const;
2591
2592 // insert another string
2593 wxString& insert(size_t nPos, const wxString& str)
2594 { insert(GetIterForNthChar(nPos), str.begin(), str.end()); return *this; }
2595 // insert n chars of str starting at nStart (in str)
2596 wxString& insert(size_t nPos, const wxString& str, size_t nStart, size_t n)
2597 {
2598 wxSTRING_UPDATE_CACHED_LENGTH(n);
2599
2600 size_t from, len;
2601 str.PosLenToImpl(nStart, n, &from, &len);
2602 m_impl.insert(PosToImpl(nPos), str.m_impl, from, len);
2603
2604 return *this;
2605 }
2606
2607 // insert first n (or all if n == npos) characters of sz
2608 wxString& insert(size_t nPos, const char *sz)
2609 {
2610 wxSTRING_INVALIDATE_CACHE();
2611
2612 m_impl.insert(PosToImpl(nPos), ImplStr(sz));
2613
2614 return *this;
2615 }
2616
2617 wxString& insert(size_t nPos, const wchar_t *sz)
2618 {
2619 wxSTRING_INVALIDATE_CACHE();
2620
2621 m_impl.insert(PosToImpl(nPos), ImplStr(sz)); return *this;
2622 }
2623
2624 wxString& insert(size_t nPos, const char *sz, size_t n)
2625 {
2626 wxSTRING_UPDATE_CACHED_LENGTH(n);
2627
2628 SubstrBufFromMB str(ImplStr(sz, n));
2629 m_impl.insert(PosToImpl(nPos), str.data, str.len);
2630
2631 return *this;
2632 }
2633
2634 wxString& insert(size_t nPos, const wchar_t *sz, size_t n)
2635 {
2636 wxSTRING_UPDATE_CACHED_LENGTH(n);
2637
2638 SubstrBufFromWC str(ImplStr(sz, n));
2639 m_impl.insert(PosToImpl(nPos), str.data, str.len);
2640
2641 return *this;
2642 }
2643
2644 // insert n copies of ch
2645 wxString& insert(size_t nPos, size_t n, wxUniChar ch)
2646 {
2647 wxSTRING_UPDATE_CACHED_LENGTH(n);
2648
2649 #if wxUSE_UNICODE_UTF8
2650 if ( !ch.IsAscii() )
2651 m_impl.insert(PosToImpl(nPos), wxStringOperations::EncodeNChars(n, ch));
2652 else
2653 #endif
2654 m_impl.insert(PosToImpl(nPos), n, (wxStringCharType)ch);
2655 return *this;
2656 }
2657
2658 iterator insert(iterator it, wxUniChar ch)
2659 {
2660 wxSTRING_UPDATE_CACHED_LENGTH(1);
2661
2662 #if wxUSE_UNICODE_UTF8
2663 if ( !ch.IsAscii() )
2664 {
2665 size_t pos = IterToImplPos(it);
2666 m_impl.insert(pos, wxStringOperations::EncodeChar(ch));
2667 return iterator(this, m_impl.begin() + pos);
2668 }
2669 else
2670 #endif
2671 return iterator(this, m_impl.insert(it.impl(), (wxStringCharType)ch));
2672 }
2673
2674 void insert(iterator it, const_iterator first, const_iterator last)
2675 {
2676 wxSTRING_INVALIDATE_CACHE();
2677
2678 m_impl.insert(it.impl(), first.impl(), last.impl());
2679 }
2680
2681 #if WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2682 void insert(iterator it, const char *first, const char *last)
2683 { insert(it - begin(), first, last - first); }
2684 void insert(iterator it, const wchar_t *first, const wchar_t *last)
2685 { insert(it - begin(), first, last - first); }
2686 void insert(iterator it, const wxCStrData& first, const wxCStrData& last)
2687 { insert(it, CreateConstIterator(first), CreateConstIterator(last)); }
2688 #endif // WXWIN_COMPATIBILITY_STRING_PTR_AS_ITER
2689
2690 void insert(iterator it, size_type n, wxUniChar ch)
2691 {
2692 wxSTRING_UPDATE_CACHED_LENGTH(n);
2693
2694 #if wxUSE_UNICODE_UTF8
2695 if ( !ch.IsAscii() )
2696 m_impl.insert(IterToImplPos(it), wxStringOperations::EncodeNChars(n, ch));
2697 else
2698 #endif
2699 m_impl.insert(it.impl(), n, (wxStringCharType)ch);
2700 }
2701
2702 // delete characters from nStart to nStart + nLen
2703 wxString& erase(size_type pos = 0, size_type n = npos)
2704 {
2705 wxSTRING_INVALIDATE_CACHE();
2706
2707 size_t from, len;
2708 PosLenToImpl(pos, n, &from, &len);
2709 m_impl.erase(from, len);
2710
2711 return *this;
2712 }
2713
2714 // delete characters from first up to last
2715 iterator erase(iterator first, iterator last)
2716 {
2717 wxSTRING_INVALIDATE_CACHE();
2718
2719 return iterator(this, m_impl.erase(first.impl(), last.impl()));
2720 }
2721
2722 iterator erase(iterator first)
2723 {
2724 wxSTRING_UPDATE_CACHED_LENGTH(-1);
2725
2726 return iterator(this, m_impl.erase(first.impl()));
2727 }
2728
2729 #ifdef wxSTRING_BASE_HASNT_CLEAR
2730 void clear() { erase(); }
2731 #else
2732 void clear()
2733 {
2734 wxSTRING_SET_CACHED_LENGTH(0);
2735
2736 m_impl.clear();
2737 }
2738 #endif
2739
2740 // replaces the substring of length nLen starting at nStart
2741 wxString& replace(size_t nStart, size_t nLen, const char* sz)
2742 {
2743 wxSTRING_INVALIDATE_CACHE();
2744
2745 size_t from, len;
2746 PosLenToImpl(nStart, nLen, &from, &len);
2747 m_impl.replace(from, len, ImplStr(sz));
2748
2749 return *this;
2750 }
2751
2752 wxString& replace(size_t nStart, size_t nLen, const wchar_t* sz)
2753 {
2754 wxSTRING_INVALIDATE_CACHE();
2755
2756 size_t from, len;
2757 PosLenToImpl(nStart, nLen, &from, &len);
2758 m_impl.replace(from, len, ImplStr(sz));
2759
2760 return *this;
2761 }
2762
2763 // replaces the substring of length nLen starting at nStart
2764 wxString& replace(size_t nStart, size_t nLen, const wxString& str)
2765 {
2766 wxSTRING_INVALIDATE_CACHE();
2767
2768 size_t from, len;
2769 PosLenToImpl(nStart, nLen, &from, &len);
2770 m_impl.replace(from, len, str.m_impl);
2771
2772 return *this;
2773 }
2774
2775 // replaces the substring with nCount copies of ch
2776 wxString& replace(size_t nStart, size_t nLen, size_t nCount, wxUniChar ch)
2777 {
2778 wxSTRING_INVALIDATE_CACHE();
2779
2780 size_t from, len;
2781 PosLenToImpl(nStart, nLen, &from, &len);
2782 #if wxUSE_UNICODE_UTF8
2783 if ( !ch.IsAscii() )
2784 m_impl.replace(from, len, wxStringOperations::EncodeNChars(nCount, ch));
2785 else
2786 #endif
2787 m_impl.replace(from, len, nCount, (wxStringCharType)ch);
2788
2789 return *this;
2790 }
2791
2792 // replaces a substring with another substring
2793 wxString& replace(size_t nStart, size_t nLen,
2794 const wxString& str, size_t nStart2, size_t nLen2)
2795 {
2796 wxSTRING_INVALIDATE_CACHE();
2797
2798 size_t from, len;
2799 PosLenToImpl(nStart, nLen, &from, &len);
2800
2801 size_t from2, len2;
2802 str.PosLenToImpl(nStart2, nLen2, &from2, &len2);
2803
2804 m_impl.replace(from, len, str.m_impl, from2, len2);
2805
2806 return *this;
2807 }
2808
2809 // replaces the substring with first nCount chars of sz
2810 wxString& replace(size_t nStart, size_t nLen,
2811 const char* sz, size_t nCount)
2812 {
2813 wxSTRING_INVALIDATE_CACHE();
2814
2815 size_t from, len;
2816 PosLenToImpl(nStart, nLen, &from, &len);
2817
2818 SubstrBufFromMB str(ImplStr(sz, nCount));
2819
2820 m_impl.replace(from, len, str.data, str.len);
2821
2822 return *this;
2823 }
2824
2825 wxString& replace(size_t nStart, size_t nLen,
2826 const wchar_t* sz, size_t nCount)
2827 {
2828 wxSTRING_INVALIDATE_CACHE();
2829
2830 size_t from, len;
2831 PosLenToImpl(nStart, nLen, &from, &len);
2832
2833 SubstrBufFromWC str(ImplStr(sz, nCount));
2834
2835 m_impl.replace(from, len, str.data, str.len);
2836
2837 return *this;
2838 }
2839
2840 wxString& replace(size_t nStart, size_t nLen,
2841 const wxString& s, size_t nCount)
2842 {
2843 wxSTRING_INVALIDATE_CACHE();
2844
2845 size_t from, len;
2846 PosLenToImpl(nStart, nLen, &from, &len);
2847 m_impl.replace(from, len, s.m_impl.c_str(), s.LenToImpl(nCount));
2848
2849 return *this;
2850 }
2851
2852 wxString& replace(iterator first, iterator last, const char* s)
2853 {
2854 wxSTRING_INVALIDATE_CACHE();
2855
2856 m_impl.replace(first.impl(), last.impl(), ImplStr(s));
2857
2858 return *this;
2859 }
2860
2861 wxString& replace(iterator first, iterator last, const wchar_t* s)
2862 {
2863 wxSTRING_INVALIDATE_CACHE();
2864
2865 m_impl.replace(first.impl(), last.impl(), ImplStr(s));
2866
2867 return *this;
2868 }
2869
2870 wxString& replace(iterator first, iterator last, const char* s, size_type n)
2871 {
2872 wxSTRING_INVALIDATE_CACHE();
2873
2874 SubstrBufFromMB str(ImplStr(s, n));
2875 m_impl.replace(first.impl(), last.impl(), str.data, str.len);
2876
2877 return *this;
2878 }
2879
2880 wxString& replace(iterator first, iterator last, const wchar_t* s, size_type n)
2881 {
2882 wxSTRING_INVALIDATE_CACHE();
2883
2884 SubstrBufFromWC str(ImplStr(s, n));
2885 m_impl.replace(first.impl(), last.impl(), str.data, str.len);
2886
2887 return *this;
2888 }
2889
2890 wxString& replace(iterator first, iterator last, const wxString& s)
2891 {
2892 wxSTRING_INVALIDATE_CACHE();
2893
2894 m_impl.replace(first.impl(), last.impl(), s.m_impl);
2895
2896 return *this;
2897 }
2898
2899 wxString& replace(iterator first, iterator last, size_type n, wxUniChar ch)
2900 {
2901 wxSTRING_INVALIDATE_CACHE();
2902
2903 #if wxUSE_UNICODE_UTF8
2904 if ( !ch.IsAscii() )
2905 m_impl.replace(first.impl(), last.impl(),
2906 wxStringOperations::EncodeNChars(n, ch));
2907 else
2908 #endif
2909 m_impl.replace(first.impl(), last.impl(), n, (wxStringCharType)ch);
2910
2911 return *this;
2912 }
2913
2914 wxString& replace(iterator first, iterator last,
2915 const_iterator first1, const_iterator last1)
2916 {
2917 wxSTRING_INVALIDATE_CACHE();
2918
2919 m_impl.replace(first.impl(), last.impl(), first1.impl(), last1.impl());
2920
2921 return *this;
2922 }
2923
2924 wxString& replace(iterator first, iterator last,
2925 const char *first1, const char *last1)
2926 { replace(first, last, first1, last1 - first1); return *this; }
2927 wxString& replace(iterator first, iterator last,
2928 const wchar_t *first1, const wchar_t *last1)
2929 { replace(first, last, first1, last1 - first1); return *this; }
2930
2931 // swap two strings
2932 void swap(wxString& str)
2933 {
2934 #if wxUSE_STRING_POS_CACHE
2935 // we modify not only this string but also the other one directly so we
2936 // need to invalidate cache for both of them (we could also try to
2937 // exchange their cache entries but it seems unlikely to be worth it)
2938 InvalidateCache();
2939 str.InvalidateCache();
2940 #endif // wxUSE_STRING_POS_CACHE
2941
2942 m_impl.swap(str.m_impl);
2943 }
2944
2945 // find a substring
2946 size_t find(const wxString& str, size_t nStart = 0) const
2947 { return PosFromImpl(m_impl.find(str.m_impl, PosToImpl(nStart))); }
2948
2949 // find first n characters of sz
2950 size_t find(const char* sz, size_t nStart = 0, size_t n = npos) const
2951 {
2952 SubstrBufFromMB str(ImplStr(sz, n));
2953 return PosFromImpl(m_impl.find(str.data, PosToImpl(nStart), str.len));
2954 }
2955 size_t find(const wchar_t* sz, size_t nStart = 0, size_t n = npos) const
2956 {
2957 SubstrBufFromWC str(ImplStr(sz, n));
2958 return PosFromImpl(m_impl.find(str.data, PosToImpl(nStart), str.len));
2959 }
2960 size_t find(const wxCharBuffer& s, size_t nStart = 0, size_t n = npos) const
2961 { return find(s.data(), nStart, n); }
2962 size_t find(const wxWCharBuffer& s, size_t nStart = 0, size_t n = npos) const
2963 { return find(s.data(), nStart, n); }
2964 size_t find(const wxCStrData& s, size_t nStart = 0, size_t n = npos) const
2965 { return find(s.AsWChar(), nStart, n); }
2966
2967 // find the first occurence of character ch after nStart
2968 size_t find(wxUniChar ch, size_t nStart = 0) const
2969 {
2970 #if wxUSE_UNICODE_UTF8
2971 if ( !ch.IsAscii() )
2972 return PosFromImpl(m_impl.find(wxStringOperations::EncodeChar(ch),
2973 PosToImpl(nStart)));
2974 else
2975 #endif
2976 return PosFromImpl(m_impl.find((wxStringCharType)ch,
2977 PosToImpl(nStart)));
2978
2979 }
2980 size_t find(wxUniCharRef ch, size_t nStart = 0) const
2981 { return find(wxUniChar(ch), nStart); }
2982 size_t find(char ch, size_t nStart = 0) const
2983 { return find(wxUniChar(ch), nStart); }
2984 size_t find(unsigned char ch, size_t nStart = 0) const
2985 { return find(wxUniChar(ch), nStart); }
2986 size_t find(wchar_t ch, size_t nStart = 0) const
2987 { return find(wxUniChar(ch), nStart); }
2988
2989 // rfind() family is exactly like find() but works right to left
2990
2991 // as find, but from the end
2992 size_t rfind(const wxString& str, size_t nStart = npos) const
2993 { return PosFromImpl(m_impl.rfind(str.m_impl, PosToImpl(nStart))); }
2994
2995 // as find, but from the end
2996 size_t rfind(const char* sz, size_t nStart = npos, size_t n = npos) const
2997 {
2998 SubstrBufFromMB str(ImplStr(sz, n));
2999 return PosFromImpl(m_impl.rfind(str.data, PosToImpl(nStart), str.len));
3000 }
3001 size_t rfind(const wchar_t* sz, size_t nStart = npos, size_t n = npos) const
3002 {
3003 SubstrBufFromWC str(ImplStr(sz, n));
3004 return PosFromImpl(m_impl.rfind(str.data, PosToImpl(nStart), str.len));
3005 }
3006 size_t rfind(const wxCharBuffer& s, size_t nStart = npos, size_t n = npos) const
3007 { return rfind(s.data(), nStart, n); }
3008 size_t rfind(const wxWCharBuffer& s, size_t nStart = npos, size_t n = npos) const
3009 { return rfind(s.data(), nStart, n); }
3010 size_t rfind(const wxCStrData& s, size_t nStart = npos, size_t n = npos) const
3011 { return rfind(s.AsWChar(), nStart, n); }
3012 // as find, but from the end
3013 size_t rfind(wxUniChar ch, size_t nStart = npos) const
3014 {
3015 #if wxUSE_UNICODE_UTF8
3016 if ( !ch.IsAscii() )
3017 return PosFromImpl(m_impl.rfind(wxStringOperations::EncodeChar(ch),
3018 PosToImpl(nStart)));
3019 else
3020 #endif
3021 return PosFromImpl(m_impl.rfind((wxStringCharType)ch,
3022 PosToImpl(nStart)));
3023 }
3024 size_t rfind(wxUniCharRef ch, size_t nStart = npos) const
3025 { return rfind(wxUniChar(ch), nStart); }
3026 size_t rfind(char ch, size_t nStart = npos) const
3027 { return rfind(wxUniChar(ch), nStart); }
3028 size_t rfind(unsigned char ch, size_t nStart = npos) const
3029 { return rfind(wxUniChar(ch), nStart); }
3030 size_t rfind(wchar_t ch, size_t nStart = npos) const
3031 { return rfind(wxUniChar(ch), nStart); }
3032
3033 // find first/last occurence of any character (not) in the set:
3034 #if wxUSE_STL_BASED_WXSTRING && !wxUSE_UNICODE_UTF8
3035 // FIXME-UTF8: this is not entirely correct, because it doesn't work if
3036 // sizeof(wchar_t)==2 and surrogates are present in the string;
3037 // should we care? Probably not.
3038 size_t find_first_of(const wxString& str, size_t nStart = 0) const
3039 { return m_impl.find_first_of(str.m_impl, nStart); }
3040 size_t find_first_of(const char* sz, size_t nStart = 0) const
3041 { return m_impl.find_first_of(ImplStr(sz), nStart); }
3042 size_t find_first_of(const wchar_t* sz, size_t nStart = 0) const
3043 { return m_impl.find_first_of(ImplStr(sz), nStart); }
3044 size_t find_first_of(const char* sz, size_t nStart, size_t n) const
3045 { return m_impl.find_first_of(ImplStr(sz), nStart, n); }
3046 size_t find_first_of(const wchar_t* sz, size_t nStart, size_t n) const
3047 { return m_impl.find_first_of(ImplStr(sz), nStart, n); }
3048 size_t find_first_of(wxUniChar c, size_t nStart = 0) const
3049 { return m_impl.find_first_of((wxChar)c, nStart); }
3050
3051 size_t find_last_of(const wxString& str, size_t nStart = npos) const
3052 { return m_impl.find_last_of(str.m_impl, nStart); }
3053 size_t find_last_of(const char* sz, size_t nStart = npos) const
3054 { return m_impl.find_last_of(ImplStr(sz), nStart); }
3055 size_t find_last_of(const wchar_t* sz, size_t nStart = npos) const
3056 { return m_impl.find_last_of(ImplStr(sz), nStart); }
3057 size_t find_last_of(const char* sz, size_t nStart, size_t n) const
3058 { return m_impl.find_last_of(ImplStr(sz), nStart, n); }
3059 size_t find_last_of(const wchar_t* sz, size_t nStart, size_t n) const
3060 { return m_impl.find_last_of(ImplStr(sz), nStart, n); }
3061 size_t find_last_of(wxUniChar c, size_t nStart = npos) const
3062 { return m_impl.find_last_of((wxChar)c, nStart); }
3063
3064 size_t find_first_not_of(const wxString& str, size_t nStart = 0) const
3065 { return m_impl.find_first_not_of(str.m_impl, nStart); }
3066 size_t find_first_not_of(const char* sz, size_t nStart = 0) const
3067 { return m_impl.find_first_not_of(ImplStr(sz), nStart); }
3068 size_t find_first_not_of(const wchar_t* sz, size_t nStart = 0) const
3069 { return m_impl.find_first_not_of(ImplStr(sz), nStart); }
3070 size_t find_first_not_of(const char* sz, size_t nStart, size_t n) const
3071 { return m_impl.find_first_not_of(ImplStr(sz), nStart, n); }
3072 size_t find_first_not_of(const wchar_t* sz, size_t nStart, size_t n) const
3073 { return m_impl.find_first_not_of(ImplStr(sz), nStart, n); }
3074 size_t find_first_not_of(wxUniChar c, size_t nStart = 0) const
3075 { return m_impl.find_first_not_of((wxChar)c, nStart); }
3076
3077 size_t find_last_not_of(const wxString& str, size_t nStart = npos) const
3078 { return m_impl.find_last_not_of(str.m_impl, nStart); }
3079 size_t find_last_not_of(const char* sz, size_t nStart = npos) const
3080 { return m_impl.find_last_not_of(ImplStr(sz), nStart); }
3081 size_t find_last_not_of(const wchar_t* sz, size_t nStart = npos) const
3082 { return m_impl.find_last_not_of(ImplStr(sz), nStart); }
3083 size_t find_last_not_of(const char* sz, size_t nStart, size_t n) const
3084 { return m_impl.find_last_not_of(ImplStr(sz), nStart, n); }
3085 size_t find_last_not_of(const wchar_t* sz, size_t nStart, size_t n) const
3086 { return m_impl.find_last_not_of(ImplStr(sz), nStart, n); }
3087 size_t find_last_not_of(wxUniChar c, size_t nStart = npos) const
3088 { return m_impl.find_last_not_of((wxChar)c, nStart); }
3089 #else
3090 // we can't use std::string implementation in UTF-8 build, because the
3091 // character sets would be interpreted wrongly:
3092
3093 // as strpbrk() but starts at nStart, returns npos if not found
3094 size_t find_first_of(const wxString& str, size_t nStart = 0) const
3095 #if wxUSE_UNICODE // FIXME-UTF8: temporary
3096 { return find_first_of(str.wc_str(), nStart); }
3097 #else
3098 { return find_first_of(str.mb_str(), nStart); }
3099 #endif
3100 // same as above
3101 size_t find_first_of(const char* sz, size_t nStart = 0) const;
3102 size_t find_first_of(const wchar_t* sz, size_t nStart = 0) const;
3103 size_t find_first_of(const char* sz, size_t nStart, size_t n) const;
3104 size_t find_first_of(const wchar_t* sz, size_t nStart, size_t n) const;
3105 // same as find(char, size_t)
3106 size_t find_first_of(wxUniChar c, size_t nStart = 0) const
3107 { return find(c, nStart); }
3108 // find the last (starting from nStart) char from str in this string
3109 size_t find_last_of (const wxString& str, size_t nStart = npos) const
3110 #if wxUSE_UNICODE // FIXME-UTF8: temporary
3111 { return find_last_of(str.wc_str(), nStart); }
3112 #else
3113 { return find_last_of(str.mb_str(), nStart); }
3114 #endif
3115 // same as above
3116 size_t find_last_of (const char* sz, size_t nStart = npos) const;
3117 size_t find_last_of (const wchar_t* sz, size_t nStart = npos) const;
3118 size_t find_last_of(const char* sz, size_t nStart, size_t n) const;
3119 size_t find_last_of(const wchar_t* sz, size_t nStart, size_t n) const;
3120 // same as above
3121 size_t find_last_of(wxUniChar c, size_t nStart = npos) const
3122 { return rfind(c, nStart); }
3123
3124 // find first/last occurence of any character not in the set
3125
3126 // as strspn() (starting from nStart), returns npos on failure
3127 size_t find_first_not_of(const wxString& str, size_t nStart = 0) const
3128 #if wxUSE_UNICODE // FIXME-UTF8: temporary
3129 { return find_first_not_of(str.wc_str(), nStart); }
3130 #else
3131 { return find_first_not_of(str.mb_str(), nStart); }
3132 #endif
3133 // same as above
3134 size_t find_first_not_of(const char* sz, size_t nStart = 0) const;
3135 size_t find_first_not_of(const wchar_t* sz, size_t nStart = 0) const;
3136 size_t find_first_not_of(const char* sz, size_t nStart, size_t n) const;
3137 size_t find_first_not_of(const wchar_t* sz, size_t nStart, size_t n) const;
3138 // same as above
3139 size_t find_first_not_of(wxUniChar ch, size_t nStart = 0) const;
3140 // as strcspn()
3141 size_t find_last_not_of(const wxString& str, size_t nStart = npos) const
3142 #if wxUSE_UNICODE // FIXME-UTF8: temporary
3143 { return find_last_not_of(str.wc_str(), nStart); }
3144 #else
3145 { return find_last_not_of(str.mb_str(), nStart); }
3146 #endif
3147 // same as above
3148 size_t find_last_not_of(const char* sz, size_t nStart = npos) const;
3149 size_t find_last_not_of(const wchar_t* sz, size_t nStart = npos) const;
3150 size_t find_last_not_of(const char* sz, size_t nStart, size_t n) const;
3151 size_t find_last_not_of(const wchar_t* sz, size_t nStart, size_t n) const;
3152 // same as above
3153 size_t find_last_not_of(wxUniChar ch, size_t nStart = npos) const;
3154 #endif // wxUSE_STL_BASED_WXSTRING && !wxUSE_UNICODE_UTF8 or not
3155
3156 // provide char/wchar_t/wxUniCharRef overloads for char-finding functions
3157 // above to resolve ambiguities:
3158 size_t find_first_of(wxUniCharRef ch, size_t nStart = 0) const
3159 { return find_first_of(wxUniChar(ch), nStart); }
3160 size_t find_first_of(char ch, size_t nStart = 0) const
3161 { return find_first_of(wxUniChar(ch), nStart); }
3162 size_t find_first_of(unsigned char ch, size_t nStart = 0) const
3163 { return find_first_of(wxUniChar(ch), nStart); }
3164 size_t find_first_of(wchar_t ch, size_t nStart = 0) const
3165 { return find_first_of(wxUniChar(ch), nStart); }
3166 size_t find_last_of(wxUniCharRef ch, size_t nStart = npos) const
3167 { return find_last_of(wxUniChar(ch), nStart); }
3168 size_t find_last_of(char ch, size_t nStart = npos) const
3169 { return find_last_of(wxUniChar(ch), nStart); }
3170 size_t find_last_of(unsigned char ch, size_t nStart = npos) const
3171 { return find_last_of(wxUniChar(ch), nStart); }
3172 size_t find_last_of(wchar_t ch, size_t nStart = npos) const
3173 { return find_last_of(wxUniChar(ch), nStart); }
3174 size_t find_first_not_of(wxUniCharRef ch, size_t nStart = 0) const
3175 { return find_first_not_of(wxUniChar(ch), nStart); }
3176 size_t find_first_not_of(char ch, size_t nStart = 0) const
3177 { return find_first_not_of(wxUniChar(ch), nStart); }
3178 size_t find_first_not_of(unsigned char ch, size_t nStart = 0) const
3179 { return find_first_not_of(wxUniChar(ch), nStart); }
3180 size_t find_first_not_of(wchar_t ch, size_t nStart = 0) const
3181 { return find_first_not_of(wxUniChar(ch), nStart); }
3182 size_t find_last_not_of(wxUniCharRef ch, size_t nStart = npos) const
3183 { return find_last_not_of(wxUniChar(ch), nStart); }
3184 size_t find_last_not_of(char ch, size_t nStart = npos) const
3185 { return find_last_not_of(wxUniChar(ch), nStart); }
3186 size_t find_last_not_of(unsigned char ch, size_t nStart = npos) const
3187 { return find_last_not_of(wxUniChar(ch), nStart); }
3188 size_t find_last_not_of(wchar_t ch, size_t nStart = npos) const
3189 { return find_last_not_of(wxUniChar(ch), nStart); }
3190
3191 // and additional overloads for the versions taking strings:
3192 size_t find_first_of(const wxCStrData& sz, size_t nStart = 0) const
3193 { return find_first_of(sz.AsString(), nStart); }
3194 size_t find_first_of(const wxCharBuffer& sz, size_t nStart = 0) const
3195 { return find_first_of(sz.data(), nStart); }
3196 size_t find_first_of(const wxWCharBuffer& sz, size_t nStart = 0) const
3197 { return find_first_of(sz.data(), nStart); }
3198 size_t find_first_of(const wxCStrData& sz, size_t nStart, size_t n) const
3199 { return find_first_of(sz.AsWChar(), nStart, n); }
3200 size_t find_first_of(const wxCharBuffer& sz, size_t nStart, size_t n) const
3201 { return find_first_of(sz.data(), nStart, n); }
3202 size_t find_first_of(const wxWCharBuffer& sz, size_t nStart, size_t n) const
3203 { return find_first_of(sz.data(), nStart, n); }
3204
3205 size_t find_last_of(const wxCStrData& sz, size_t nStart = 0) const
3206 { return find_last_of(sz.AsString(), nStart); }
3207 size_t find_last_of(const wxCharBuffer& sz, size_t nStart = 0) const
3208 { return find_last_of(sz.data(), nStart); }
3209 size_t find_last_of(const wxWCharBuffer& sz, size_t nStart = 0) const
3210 { return find_last_of(sz.data(), nStart); }
3211 size_t find_last_of(const wxCStrData& sz, size_t nStart, size_t n) const
3212 { return find_last_of(sz.AsWChar(), nStart, n); }
3213 size_t find_last_of(const wxCharBuffer& sz, size_t nStart, size_t n) const
3214 { return find_last_of(sz.data(), nStart, n); }
3215 size_t find_last_of(const wxWCharBuffer& sz, size_t nStart, size_t n) const
3216 { return find_last_of(sz.data(), nStart, n); }
3217
3218 size_t find_first_not_of(const wxCStrData& sz, size_t nStart = 0) const
3219 { return find_first_not_of(sz.AsString(), nStart); }
3220 size_t find_first_not_of(const wxCharBuffer& sz, size_t nStart = 0) const
3221 { return find_first_not_of(sz.data(), nStart); }
3222 size_t find_first_not_of(const wxWCharBuffer& sz, size_t nStart = 0) const
3223 { return find_first_not_of(sz.data(), nStart); }
3224 size_t find_first_not_of(const wxCStrData& sz, size_t nStart, size_t n) const
3225 { return find_first_not_of(sz.AsWChar(), nStart, n); }
3226 size_t find_first_not_of(const wxCharBuffer& sz, size_t nStart, size_t n) const
3227 { return find_first_not_of(sz.data(), nStart, n); }
3228 size_t find_first_not_of(const wxWCharBuffer& sz, size_t nStart, size_t n) const
3229 { return find_first_not_of(sz.data(), nStart, n); }
3230
3231 size_t find_last_not_of(const wxCStrData& sz, size_t nStart = 0) const
3232 { return find_last_not_of(sz.AsString(), nStart); }
3233 size_t find_last_not_of(const wxCharBuffer& sz, size_t nStart = 0) const
3234 { return find_last_not_of(sz.data(), nStart); }
3235 size_t find_last_not_of(const wxWCharBuffer& sz, size_t nStart = 0) const
3236 { return find_last_not_of(sz.data(), nStart); }
3237 size_t find_last_not_of(const wxCStrData& sz, size_t nStart, size_t n) const
3238 { return find_last_not_of(sz.AsWChar(), nStart, n); }
3239 size_t find_last_not_of(const wxCharBuffer& sz, size_t nStart, size_t n) const
3240 { return find_last_not_of(sz.data(), nStart, n); }
3241 size_t find_last_not_of(const wxWCharBuffer& sz, size_t nStart, size_t n) const
3242 { return find_last_not_of(sz.data(), nStart, n); }
3243
3244 // string += string
3245 wxString& operator+=(const wxString& s)
3246 {
3247 wxSTRING_INVALIDATE_CACHED_LENGTH();
3248
3249 m_impl += s.m_impl;
3250 return *this;
3251 }
3252 // string += C string
3253 wxString& operator+=(const char *psz)
3254 {
3255 wxSTRING_INVALIDATE_CACHED_LENGTH();
3256
3257 m_impl += ImplStr(psz);
3258 return *this;
3259 }
3260 wxString& operator+=(const wchar_t *pwz)
3261 {
3262 wxSTRING_INVALIDATE_CACHED_LENGTH();
3263
3264 m_impl += ImplStr(pwz);
3265 return *this;
3266 }
3267 wxString& operator+=(const wxCStrData& s)
3268 {
3269 wxSTRING_INVALIDATE_CACHED_LENGTH();
3270
3271 m_impl += s.AsString().m_impl;
3272 return *this;
3273 }
3274 wxString& operator+=(const wxCharBuffer& s)
3275 { return operator+=(s.data()); }
3276 wxString& operator+=(const wxWCharBuffer& s)
3277 { return operator+=(s.data()); }
3278 // string += char
3279 wxString& operator+=(wxUniChar ch)
3280 {
3281 wxSTRING_UPDATE_CACHED_LENGTH(1);
3282
3283 #if wxUSE_UNICODE_UTF8
3284 if ( !ch.IsAscii() )
3285 m_impl += wxStringOperations::EncodeChar(ch);
3286 else
3287 #endif
3288 m_impl += (wxStringCharType)ch;
3289 return *this;
3290 }
3291 wxString& operator+=(wxUniCharRef ch) { return *this += wxUniChar(ch); }
3292 wxString& operator+=(int ch) { return *this += wxUniChar(ch); }
3293 wxString& operator+=(char ch) { return *this += wxUniChar(ch); }
3294 wxString& operator+=(unsigned char ch) { return *this += wxUniChar(ch); }
3295 wxString& operator+=(wchar_t ch) { return *this += wxUniChar(ch); }
3296
3297 private:
3298 #if !wxUSE_STL_BASED_WXSTRING
3299 // helpers for wxStringBuffer and wxStringBufferLength
3300 wxStringCharType *DoGetWriteBuf(size_t nLen)
3301 {
3302 return m_impl.DoGetWriteBuf(nLen);
3303 }
3304
3305 void DoUngetWriteBuf()
3306 {
3307 wxSTRING_INVALIDATE_CACHE();
3308
3309 m_impl.DoUngetWriteBuf();
3310 }
3311
3312 void DoUngetWriteBuf(size_t nLen)
3313 {
3314 wxSTRING_SET_CACHED_LENGTH(nLen);
3315
3316 m_impl.DoUngetWriteBuf(nLen);
3317 }
3318 #endif // !wxUSE_STL_BASED_WXSTRING
3319
3320 #ifndef wxNEEDS_WXSTRING_PRINTF_MIXIN
3321 #if !wxUSE_UTF8_LOCALE_ONLY
3322 int DoPrintfWchar(const wxChar *format, ...);
3323 static wxString DoFormatWchar(const wxChar *format, ...);
3324 #endif
3325 #if wxUSE_UNICODE_UTF8
3326 int DoPrintfUtf8(const char *format, ...);
3327 static wxString DoFormatUtf8(const char *format, ...);
3328 #endif
3329 #endif
3330
3331 #if !wxUSE_STL_BASED_WXSTRING
3332 // check string's data validity
3333 bool IsValid() const { return m_impl.GetStringData()->IsValid(); }
3334 #endif
3335
3336 private:
3337 wxStringImpl m_impl;
3338
3339 // buffers for compatibility conversion from (char*)c_str() and
3340 // (wchar_t*)c_str():
3341 // FIXME-UTF8: bechmark various approaches to keeping compatibility buffers
3342 template<typename T>
3343 struct ConvertedBuffer
3344 {
3345 ConvertedBuffer() : m_buf(NULL) {}
3346 ~ConvertedBuffer()
3347 { free(m_buf); }
3348
3349 operator T*() const { return m_buf; }
3350
3351 ConvertedBuffer& operator=(T *str)
3352 {
3353 free(m_buf);
3354 m_buf = str;
3355 return *this;
3356 }
3357
3358 T *m_buf;
3359 };
3360 #if wxUSE_UNICODE && !wxUSE_UTF8_LOCALE_ONLY
3361 ConvertedBuffer<char> m_convertedToChar;
3362 #endif
3363 #if !wxUSE_UNICODE_WCHAR
3364 ConvertedBuffer<wchar_t> m_convertedToWChar;
3365 #endif
3366
3367 #if wxUSE_UNICODE_UTF8
3368 // FIXME-UTF8: (try to) move this elsewhere (TLS) or solve differently
3369 // assigning to character pointer to by wxString::interator may
3370 // change the underlying wxStringImpl iterator, so we have to
3371 // keep track of all iterators and update them as necessary:
3372 struct wxStringIteratorNodeHead
3373 {
3374 wxStringIteratorNodeHead() : ptr(NULL) {}
3375 wxStringIteratorNode *ptr;
3376
3377 // copying is disallowed as it would result in more than one pointer into
3378 // the same linked list
3379 DECLARE_NO_COPY_CLASS(wxStringIteratorNodeHead)
3380 };
3381
3382 wxStringIteratorNodeHead m_iterators;
3383
3384 friend class WXDLLIMPEXP_FWD_BASE wxStringIteratorNode;
3385 friend class WXDLLIMPEXP_FWD_BASE wxUniCharRef;
3386 #endif // wxUSE_UNICODE_UTF8
3387
3388 friend class WXDLLIMPEXP_FWD_BASE wxCStrData;
3389 friend class wxStringInternalBuffer;
3390 friend class wxStringInternalBufferLength;
3391 };
3392
3393 #ifdef wxNEEDS_WXSTRING_PRINTF_MIXIN
3394 #pragma warning (default:4275)
3395 #endif
3396
3397 // string iterator operators that satisfy STL Random Access Iterator
3398 // requirements:
3399 inline wxString::iterator operator+(ptrdiff_t n, wxString::iterator i)
3400 { return i + n; }
3401 inline wxString::const_iterator operator+(ptrdiff_t n, wxString::const_iterator i)
3402 { return i + n; }
3403 inline wxString::reverse_iterator operator+(ptrdiff_t n, wxString::reverse_iterator i)
3404 { return i + n; }
3405 inline wxString::const_reverse_iterator operator+(ptrdiff_t n, wxString::const_reverse_iterator i)
3406 { return i + n; }
3407
3408 // notice that even though for many compilers the friend declarations above are
3409 // enough, from the point of view of C++ standard we must have the declarations
3410 // here as friend ones are not injected in the enclosing namespace and without
3411 // them the code fails to compile with conforming compilers such as xlC or g++4
3412 wxString WXDLLIMPEXP_BASE operator+(const wxString& string1, const wxString& string2);
3413 wxString WXDLLIMPEXP_BASE operator+(const wxString& string, const char *psz);
3414 wxString WXDLLIMPEXP_BASE operator+(const wxString& string, const wchar_t *pwz);
3415 wxString WXDLLIMPEXP_BASE operator+(const char *psz, const wxString& string);
3416 wxString WXDLLIMPEXP_BASE operator+(const wchar_t *pwz, const wxString& string);
3417
3418 wxString WXDLLIMPEXP_BASE operator+(const wxString& string, wxUniChar ch);
3419 wxString WXDLLIMPEXP_BASE operator+(wxUniChar ch, const wxString& string);
3420
3421 inline wxString operator+(const wxString& string, wxUniCharRef ch)
3422 { return string + (wxUniChar)ch; }
3423 inline wxString operator+(const wxString& string, char ch)
3424 { return string + wxUniChar(ch); }
3425 inline wxString operator+(const wxString& string, wchar_t ch)
3426 { return string + wxUniChar(ch); }
3427 inline wxString operator+(wxUniCharRef ch, const wxString& string)
3428 { return (wxUniChar)ch + string; }
3429 inline wxString operator+(char ch, const wxString& string)
3430 { return wxUniChar(ch) + string; }
3431 inline wxString operator+(wchar_t ch, const wxString& string)
3432 { return wxUniChar(ch) + string; }
3433
3434
3435 #define wxGetEmptyString() wxString()
3436
3437 // ----------------------------------------------------------------------------
3438 // helper functions which couldn't be defined inline
3439 // ----------------------------------------------------------------------------
3440
3441 namespace wxPrivate
3442 {
3443
3444 #if wxUSE_UNICODE_WCHAR
3445
3446 template <>
3447 struct wxStringAsBufHelper<char>
3448 {
3449 static wxCharBuffer Get(const wxString& s, size_t *len)
3450 {
3451 wxCharBuffer buf(s.mb_str());
3452 if ( len )
3453 *len = buf ? strlen(buf) : 0;
3454 return buf;
3455 }
3456 };
3457
3458 template <>
3459 struct wxStringAsBufHelper<wchar_t>
3460 {
3461 static wxWCharBuffer Get(const wxString& s, size_t *len)
3462 {
3463 if ( len )
3464 *len = s.length();
3465 return wxWCharBuffer::CreateNonOwned(s.wx_str());
3466 }
3467 };
3468
3469 #elif wxUSE_UNICODE_UTF8
3470
3471 template <>
3472 struct wxStringAsBufHelper<char>
3473 {
3474 static wxCharBuffer Get(const wxString& s, size_t *len)
3475 {
3476 if ( len )
3477 *len = s.utf8_length();
3478 return wxCharBuffer::CreateNonOwned(s.wx_str());
3479 }
3480 };
3481
3482 template <>
3483 struct wxStringAsBufHelper<wchar_t>
3484 {
3485 static wxWCharBuffer Get(const wxString& s, size_t *len)
3486 {
3487 wxWCharBuffer wbuf(s.wc_str());
3488 if ( len )
3489 *len = wxWcslen(wbuf);
3490 return wbuf;
3491 }
3492 };
3493
3494 #endif // Unicode build kind
3495
3496 } // namespace wxPrivate
3497
3498 // ----------------------------------------------------------------------------
3499 // wxStringBuffer: a tiny class allowing to get a writable pointer into string
3500 // ----------------------------------------------------------------------------
3501
3502 #if !wxUSE_STL_BASED_WXSTRING
3503 // string buffer for direct access to string data in their native
3504 // representation:
3505 class wxStringInternalBuffer
3506 {
3507 public:
3508 typedef wxStringCharType CharType;
3509
3510 wxStringInternalBuffer(wxString& str, size_t lenWanted = 1024)
3511 : m_str(str), m_buf(NULL)
3512 { m_buf = m_str.DoGetWriteBuf(lenWanted); }
3513
3514 ~wxStringInternalBuffer() { m_str.DoUngetWriteBuf(); }
3515
3516 operator wxStringCharType*() const { return m_buf; }
3517
3518 private:
3519 wxString& m_str;
3520 wxStringCharType *m_buf;
3521
3522 DECLARE_NO_COPY_CLASS(wxStringInternalBuffer)
3523 };
3524
3525 class wxStringInternalBufferLength
3526 {
3527 public:
3528 typedef wxStringCharType CharType;
3529
3530 wxStringInternalBufferLength(wxString& str, size_t lenWanted = 1024)
3531 : m_str(str), m_buf(NULL), m_len(0), m_lenSet(false)
3532 {
3533 m_buf = m_str.DoGetWriteBuf(lenWanted);
3534 wxASSERT(m_buf != NULL);
3535 }
3536
3537 ~wxStringInternalBufferLength()
3538 {
3539 wxASSERT(m_lenSet);
3540 m_str.DoUngetWriteBuf(m_len);
3541 }
3542
3543 operator wxStringCharType*() const { return m_buf; }
3544 void SetLength(size_t length) { m_len = length; m_lenSet = true; }
3545
3546 private:
3547 wxString& m_str;
3548 wxStringCharType *m_buf;
3549 size_t m_len;
3550 bool m_lenSet;
3551
3552 DECLARE_NO_COPY_CLASS(wxStringInternalBufferLength)
3553 };
3554
3555 #endif // !wxUSE_STL_BASED_WXSTRING
3556
3557 template<typename T>
3558 class WXDLLIMPEXP_BASE wxStringTypeBufferBase
3559 {
3560 public:
3561 typedef T CharType;
3562
3563 wxStringTypeBufferBase(wxString& str, size_t lenWanted = 1024)
3564 : m_str(str), m_buf(lenWanted)
3565 {
3566 // for compatibility with old wxStringBuffer which provided direct
3567 // access to wxString internal buffer, initialize ourselves with the
3568 // string initial contents
3569
3570 // FIXME-VC6: remove the ugly (CharType *)NULL and use normal
3571 // tchar_str<CharType>
3572 size_t len;
3573 const wxCharTypeBuffer<CharType> buf(str.tchar_str(&len, (CharType *)NULL));
3574 if ( buf )
3575 {
3576 if ( len > lenWanted )
3577 {
3578 // in this case there is not enough space for terminating NUL,
3579 // ensure that we still put it there
3580 m_buf.data()[lenWanted] = 0;
3581 len = lenWanted - 1;
3582 }
3583
3584 memcpy(m_buf.data(), buf, (len + 1)*sizeof(CharType));
3585 }
3586 //else: conversion failed, this can happen when trying to get Unicode
3587 // string contents into a char string
3588 }
3589
3590 operator CharType*() { return m_buf.data(); }
3591
3592 protected:
3593 wxString& m_str;
3594 wxCharTypeBuffer<CharType> m_buf;
3595 };
3596
3597 template<typename T>
3598 class WXDLLIMPEXP_BASE wxStringTypeBufferLengthBase
3599 : public wxStringTypeBufferBase<T>
3600 {
3601 public:
3602 wxStringTypeBufferLengthBase(wxString& str, size_t lenWanted = 1024)
3603 : wxStringTypeBufferBase<T>(str, lenWanted),
3604 m_len(0),
3605 m_lenSet(false)
3606 { }
3607
3608 ~wxStringTypeBufferLengthBase()
3609 {
3610 wxASSERT_MSG( this->m_lenSet, "forgot to call SetLength()" );
3611 }
3612
3613 void SetLength(size_t length) { m_len = length; m_lenSet = true; }
3614
3615 protected:
3616 size_t m_len;
3617 bool m_lenSet;
3618 };
3619
3620 template<typename T>
3621 class wxStringTypeBuffer : public wxStringTypeBufferBase<T>
3622 {
3623 public:
3624 wxStringTypeBuffer(wxString& str, size_t lenWanted = 1024)
3625 : wxStringTypeBufferBase<T>(str, lenWanted)
3626 { }
3627
3628 ~wxStringTypeBuffer()
3629 {
3630 this->m_str.assign(this->m_buf.data());
3631 }
3632
3633 DECLARE_NO_COPY_CLASS(wxStringTypeBuffer)
3634 };
3635
3636 template<typename T>
3637 class wxStringTypeBufferLength : public wxStringTypeBufferLengthBase<T>
3638 {
3639 public:
3640 wxStringTypeBufferLength(wxString& str, size_t lenWanted = 1024)
3641 : wxStringTypeBufferLengthBase<T>(str, lenWanted)
3642 { }
3643
3644 ~wxStringTypeBufferLength()
3645 {
3646 this->m_str.assign(this->m_buf.data(), this->m_len);
3647 }
3648
3649 DECLARE_NO_COPY_CLASS(wxStringTypeBufferLength)
3650 };
3651
3652 #if wxUSE_STL_BASED_WXSTRING
3653
3654 WXDLLIMPEXP_TEMPLATE_INSTANCE_BASE( wxStringTypeBufferBase<wxStringCharType> )
3655
3656 class wxStringInternalBuffer : public wxStringTypeBufferBase<wxStringCharType>
3657 {
3658 public:
3659 wxStringInternalBuffer(wxString& str, size_t lenWanted = 1024)
3660 : wxStringTypeBufferBase<wxStringCharType>(str, lenWanted) {}
3661 ~wxStringInternalBuffer()
3662 { m_str.m_impl.assign(m_buf.data()); }
3663
3664 DECLARE_NO_COPY_CLASS(wxStringInternalBuffer)
3665 };
3666
3667 WXDLLIMPEXP_TEMPLATE_INSTANCE_BASE(
3668 wxStringTypeBufferLengthBase<wxStringCharType> )
3669
3670 class wxStringInternalBufferLength
3671 : public wxStringTypeBufferLengthBase<wxStringCharType>
3672 {
3673 public:
3674 wxStringInternalBufferLength(wxString& str, size_t lenWanted = 1024)
3675 : wxStringTypeBufferLengthBase<wxStringCharType>(str, lenWanted) {}
3676
3677 ~wxStringInternalBufferLength()
3678 {
3679 m_str.m_impl.assign(m_buf.data(), m_len);
3680 }
3681
3682 DECLARE_NO_COPY_CLASS(wxStringInternalBufferLength)
3683 };
3684
3685 #endif // wxUSE_STL_BASED_WXSTRING
3686
3687
3688 #if wxUSE_STL_BASED_WXSTRING || wxUSE_UNICODE_UTF8
3689 typedef wxStringTypeBuffer<wxChar> wxStringBuffer;
3690 typedef wxStringTypeBufferLength<wxChar> wxStringBufferLength;
3691 #else // if !wxUSE_STL_BASED_WXSTRING && !wxUSE_UNICODE_UTF8
3692 typedef wxStringInternalBuffer wxStringBuffer;
3693 typedef wxStringInternalBufferLength wxStringBufferLength;
3694 #endif // !wxUSE_STL_BASED_WXSTRING && !wxUSE_UNICODE_UTF8
3695
3696 #if wxUSE_UNICODE_UTF8
3697 typedef wxStringInternalBuffer wxUTF8StringBuffer;
3698 typedef wxStringInternalBufferLength wxUTF8StringBufferLength;
3699 #elif wxUSE_UNICODE_WCHAR
3700
3701 WXDLLIMPEXP_TEMPLATE_INSTANCE_BASE( wxStringTypeBufferBase<char> )
3702
3703 class WXDLLIMPEXP_BASE wxUTF8StringBuffer : public wxStringTypeBufferBase<char>
3704 {
3705 public:
3706 wxUTF8StringBuffer(wxString& str, size_t lenWanted = 1024)
3707 : wxStringTypeBufferBase<char>(str, lenWanted) {}
3708 ~wxUTF8StringBuffer();
3709
3710 DECLARE_NO_COPY_CLASS(wxUTF8StringBuffer)
3711 };
3712
3713 WXDLLIMPEXP_TEMPLATE_INSTANCE_BASE( wxStringTypeBufferLengthBase<char> )
3714
3715 class WXDLLIMPEXP_BASE wxUTF8StringBufferLength
3716 : public wxStringTypeBufferLengthBase<char>
3717 {
3718 public:
3719 wxUTF8StringBufferLength(wxString& str, size_t lenWanted = 1024)
3720 : wxStringTypeBufferLengthBase<char>(str, lenWanted) {}
3721 ~wxUTF8StringBufferLength();
3722
3723 DECLARE_NO_COPY_CLASS(wxUTF8StringBufferLength)
3724 };
3725 #endif // wxUSE_UNICODE_UTF8/wxUSE_UNICODE_WCHAR
3726
3727
3728 // ---------------------------------------------------------------------------
3729 // wxString comparison functions: operator versions are always case sensitive
3730 // ---------------------------------------------------------------------------
3731
3732 #define wxCMP_WXCHAR_STRING(p, s, op) 0 op s.Cmp(p)
3733
3734 wxDEFINE_ALL_COMPARISONS(const wxChar *, const wxString&, wxCMP_WXCHAR_STRING)
3735
3736 #undef wxCMP_WXCHAR_STRING
3737
3738 inline bool operator==(const wxString& s1, const wxString& s2)
3739 { return s1.IsSameAs(s2); }
3740 inline bool operator!=(const wxString& s1, const wxString& s2)
3741 { return !s1.IsSameAs(s2); }
3742 inline bool operator< (const wxString& s1, const wxString& s2)
3743 { return s1.Cmp(s2) < 0; }
3744 inline bool operator> (const wxString& s1, const wxString& s2)
3745 { return s1.Cmp(s2) > 0; }
3746 inline bool operator<=(const wxString& s1, const wxString& s2)
3747 { return s1.Cmp(s2) <= 0; }
3748 inline bool operator>=(const wxString& s1, const wxString& s2)
3749 { return s1.Cmp(s2) >= 0; }
3750
3751 inline bool operator==(const wxString& s1, const wxCStrData& s2)
3752 { return s1 == s2.AsString(); }
3753 inline bool operator==(const wxCStrData& s1, const wxString& s2)
3754 { return s1.AsString() == s2; }
3755 inline bool operator!=(const wxString& s1, const wxCStrData& s2)
3756 { return s1 != s2.AsString(); }
3757 inline bool operator!=(const wxCStrData& s1, const wxString& s2)
3758 { return s1.AsString() != s2; }
3759
3760 inline bool operator==(const wxString& s1, const wxWCharBuffer& s2)
3761 { return (s1.Cmp((const wchar_t *)s2) == 0); }
3762 inline bool operator==(const wxWCharBuffer& s1, const wxString& s2)
3763 { return (s2.Cmp((const wchar_t *)s1) == 0); }
3764 inline bool operator!=(const wxString& s1, const wxWCharBuffer& s2)
3765 { return (s1.Cmp((const wchar_t *)s2) != 0); }
3766 inline bool operator!=(const wxWCharBuffer& s1, const wxString& s2)
3767 { return (s2.Cmp((const wchar_t *)s1) != 0); }
3768
3769 inline bool operator==(const wxString& s1, const wxCharBuffer& s2)
3770 { return (s1.Cmp((const char *)s2) == 0); }
3771 inline bool operator==(const wxCharBuffer& s1, const wxString& s2)
3772 { return (s2.Cmp((const char *)s1) == 0); }
3773 inline bool operator!=(const wxString& s1, const wxCharBuffer& s2)
3774 { return (s1.Cmp((const char *)s2) != 0); }
3775 inline bool operator!=(const wxCharBuffer& s1, const wxString& s2)
3776 { return (s2.Cmp((const char *)s1) != 0); }
3777
3778 inline wxString operator+(const wxString& string, const wxWCharBuffer& buf)
3779 { return string + (const wchar_t *)buf; }
3780 inline wxString operator+(const wxWCharBuffer& buf, const wxString& string)
3781 { return (const wchar_t *)buf + string; }
3782
3783 inline wxString operator+(const wxString& string, const wxCharBuffer& buf)
3784 { return string + (const char *)buf; }
3785 inline wxString operator+(const wxCharBuffer& buf, const wxString& string)
3786 { return (const char *)buf + string; }
3787
3788 // comparison with char
3789 inline bool operator==(const wxUniChar& c, const wxString& s) { return s.IsSameAs(c); }
3790 inline bool operator==(const wxUniCharRef& c, const wxString& s) { return s.IsSameAs(c); }
3791 inline bool operator==(char c, const wxString& s) { return s.IsSameAs(c); }
3792 inline bool operator==(wchar_t c, const wxString& s) { return s.IsSameAs(c); }
3793 inline bool operator==(int c, const wxString& s) { return s.IsSameAs(c); }
3794 inline bool operator==(const wxString& s, const wxUniChar& c) { return s.IsSameAs(c); }
3795 inline bool operator==(const wxString& s, const wxUniCharRef& c) { return s.IsSameAs(c); }
3796 inline bool operator==(const wxString& s, char c) { return s.IsSameAs(c); }
3797 inline bool operator==(const wxString& s, wchar_t c) { return s.IsSameAs(c); }
3798 inline bool operator!=(const wxUniChar& c, const wxString& s) { return !s.IsSameAs(c); }
3799 inline bool operator!=(const wxUniCharRef& c, const wxString& s) { return !s.IsSameAs(c); }
3800 inline bool operator!=(char c, const wxString& s) { return !s.IsSameAs(c); }
3801 inline bool operator!=(wchar_t c, const wxString& s) { return !s.IsSameAs(c); }
3802 inline bool operator!=(int c, const wxString& s) { return !s.IsSameAs(c); }
3803 inline bool operator!=(const wxString& s, const wxUniChar& c) { return !s.IsSameAs(c); }
3804 inline bool operator!=(const wxString& s, const wxUniCharRef& c) { return !s.IsSameAs(c); }
3805 inline bool operator!=(const wxString& s, char c) { return !s.IsSameAs(c); }
3806 inline bool operator!=(const wxString& s, wchar_t c) { return !s.IsSameAs(c); }
3807
3808 // comparison with C string in Unicode build
3809 #if wxUSE_UNICODE
3810
3811 #define wxCMP_CHAR_STRING(p, s, op) wxString(p) op s
3812
3813 wxDEFINE_ALL_COMPARISONS(const char *, const wxString&, wxCMP_CHAR_STRING)
3814
3815 #undef wxCMP_CHAR_STRING
3816
3817 #endif // wxUSE_UNICODE
3818
3819 // we also need to provide the operators for comparison with wxCStrData to
3820 // resolve ambiguity between operator(const wxChar *,const wxString &) and
3821 // operator(const wxChar *, const wxChar *) for "p == s.c_str()"
3822 //
3823 // notice that these are (shallow) pointer comparisons, not (deep) string ones
3824 #define wxCMP_CHAR_CSTRDATA(p, s, op) p op s.AsChar()
3825 #define wxCMP_WCHAR_CSTRDATA(p, s, op) p op s.AsWChar()
3826
3827 wxDEFINE_ALL_COMPARISONS(const wchar_t *, const wxCStrData&, wxCMP_WCHAR_CSTRDATA)
3828 wxDEFINE_ALL_COMPARISONS(const char *, const wxCStrData&, wxCMP_CHAR_CSTRDATA)
3829
3830 #undef wxCMP_CHAR_CSTRDATA
3831 #undef wxCMP_WCHAR_CSTRDATA
3832
3833 // ---------------------------------------------------------------------------
3834 // Implementation only from here until the end of file
3835 // ---------------------------------------------------------------------------
3836
3837 #if wxUSE_STD_IOSTREAM
3838
3839 #include "wx/iosfwrap.h"
3840
3841 WXDLLIMPEXP_BASE wxSTD ostream& operator<<(wxSTD ostream&, const wxString&);
3842 WXDLLIMPEXP_BASE wxSTD ostream& operator<<(wxSTD ostream&, const wxCStrData&);
3843 WXDLLIMPEXP_BASE wxSTD ostream& operator<<(wxSTD ostream&, const wxCharBuffer&);
3844 #ifndef __BORLANDC__
3845 WXDLLIMPEXP_BASE wxSTD ostream& operator<<(wxSTD ostream&, const wxWCharBuffer&);
3846 #endif
3847
3848 #if wxUSE_UNICODE && defined(HAVE_WOSTREAM)
3849
3850 WXDLLIMPEXP_BASE wxSTD wostream& operator<<(wxSTD wostream&, const wxString&);
3851 WXDLLIMPEXP_BASE wxSTD wostream& operator<<(wxSTD wostream&, const wxCStrData&);
3852 WXDLLIMPEXP_BASE wxSTD wostream& operator<<(wxSTD wostream&, const wxWCharBuffer&);
3853
3854 #endif // wxUSE_UNICODE && defined(HAVE_WOSTREAM)
3855
3856 #endif // wxUSE_STD_IOSTREAM
3857
3858 // ---------------------------------------------------------------------------
3859 // wxCStrData implementation
3860 // ---------------------------------------------------------------------------
3861
3862 inline wxCStrData::wxCStrData(char *buf)
3863 : m_str(new wxString(buf)), m_offset(0), m_owned(true) {}
3864 inline wxCStrData::wxCStrData(wchar_t *buf)
3865 : m_str(new wxString(buf)), m_offset(0), m_owned(true) {}
3866
3867 inline wxCStrData::wxCStrData(const wxCStrData& data)
3868 : m_str(data.m_owned ? new wxString(*data.m_str) : data.m_str),
3869 m_offset(data.m_offset),
3870 m_owned(data.m_owned)
3871 {
3872 }
3873
3874 inline wxCStrData::~wxCStrData()
3875 {
3876 if ( m_owned )
3877 delete wx_const_cast(wxString*, m_str); // cast to silence warnings
3878 }
3879
3880 // simple cases for AsChar() and AsWChar(), the complicated ones are
3881 // in string.cpp
3882 #if wxUSE_UNICODE_WCHAR
3883 inline const wchar_t* wxCStrData::AsWChar() const
3884 {
3885 return m_str->wx_str() + m_offset;
3886 }
3887 #endif // wxUSE_UNICODE_WCHAR
3888
3889 #if !wxUSE_UNICODE
3890 inline const char* wxCStrData::AsChar() const
3891 {
3892 return m_str->wx_str() + m_offset;
3893 }
3894 #endif // !wxUSE_UNICODE
3895
3896 #if wxUSE_UTF8_LOCALE_ONLY
3897 inline const char* wxCStrData::AsChar() const
3898 {
3899 return wxStringOperations::AddToIter(m_str->wx_str(), m_offset);
3900 }
3901 #endif // wxUSE_UTF8_LOCALE_ONLY
3902
3903 inline const wxCharBuffer wxCStrData::AsCharBuf() const
3904 {
3905 #if !wxUSE_UNICODE
3906 return wxCharBuffer::CreateNonOwned(AsChar());
3907 #else
3908 return AsString().mb_str();
3909 #endif
3910 }
3911
3912 inline const wxWCharBuffer wxCStrData::AsWCharBuf() const
3913 {
3914 #if wxUSE_UNICODE_WCHAR
3915 return wxWCharBuffer::CreateNonOwned(AsWChar());
3916 #else
3917 return AsString().wc_str();
3918 #endif
3919 }
3920
3921 inline wxString wxCStrData::AsString() const
3922 {
3923 if ( m_offset == 0 )
3924 return *m_str;
3925 else
3926 return m_str->Mid(m_offset);
3927 }
3928
3929 inline const wxStringCharType *wxCStrData::AsInternal() const
3930 {
3931 #if wxUSE_UNICODE_UTF8
3932 return wxStringOperations::AddToIter(m_str->wx_str(), m_offset);
3933 #else
3934 return m_str->wx_str() + m_offset;
3935 #endif
3936 }
3937
3938 inline wxUniChar wxCStrData::operator*() const
3939 {
3940 if ( m_str->empty() )
3941 return wxUniChar(_T('\0'));
3942 else
3943 return (*m_str)[m_offset];
3944 }
3945
3946 inline wxUniChar wxCStrData::operator[](size_t n) const
3947 {
3948 // NB: we intentionally use operator[] and not at() here because the former
3949 // works for the terminating NUL while the latter does not
3950 return (*m_str)[m_offset + n];
3951 }
3952
3953 // ----------------------------------------------------------------------------
3954 // more wxCStrData operators
3955 // ----------------------------------------------------------------------------
3956
3957 // we need to define those to allow "size_t pos = p - s.c_str()" where p is
3958 // some pointer into the string
3959 inline size_t operator-(const char *p, const wxCStrData& cs)
3960 {
3961 return p - cs.AsChar();
3962 }
3963
3964 inline size_t operator-(const wchar_t *p, const wxCStrData& cs)
3965 {
3966 return p - cs.AsWChar();
3967 }
3968
3969 // ----------------------------------------------------------------------------
3970 // implementation of wx[W]CharBuffer inline methods using wxCStrData
3971 // ----------------------------------------------------------------------------
3972
3973 // FIXME-UTF8: move this to buffer.h
3974 inline wxCharBuffer::wxCharBuffer(const wxCStrData& cstr)
3975 : wxCharTypeBufferBase(cstr.AsCharBuf())
3976 {
3977 }
3978
3979 inline wxWCharBuffer::wxWCharBuffer(const wxCStrData& cstr)
3980 : wxCharTypeBufferBase(cstr.AsWCharBuf())
3981 {
3982 }
3983
3984 #if wxUSE_UNICODE_UTF8
3985 // ----------------------------------------------------------------------------
3986 // implementation of wxStringIteratorNode inline methods
3987 // ----------------------------------------------------------------------------
3988
3989 void wxStringIteratorNode::DoSet(const wxString *str,
3990 wxStringImpl::const_iterator *citer,
3991 wxStringImpl::iterator *iter)
3992 {
3993 m_prev = NULL;
3994 m_iter = iter;
3995 m_citer = citer;
3996 m_str = str;
3997 if ( str )
3998 {
3999 m_next = str->m_iterators.ptr;
4000 wx_const_cast(wxString*, m_str)->m_iterators.ptr = this;
4001 if ( m_next )
4002 m_next->m_prev = this;
4003 }
4004 else
4005 {
4006 m_next = NULL;
4007 }
4008 }
4009
4010 void wxStringIteratorNode::clear()
4011 {
4012 if ( m_next )
4013 m_next->m_prev = m_prev;
4014 if ( m_prev )
4015 m_prev->m_next = m_next;
4016 else if ( m_str ) // first in the list
4017 wx_const_cast(wxString*, m_str)->m_iterators.ptr = m_next;
4018
4019 m_next = m_prev = NULL;
4020 m_citer = NULL;
4021 m_iter = NULL;
4022 m_str = NULL;
4023 }
4024 #endif // wxUSE_UNICODE_UTF8
4025
4026 #if WXWIN_COMPATIBILITY_2_8
4027 // lot of code out there doesn't explicitly include wx/crt.h, but uses
4028 // CRT wrappers that are now declared in wx/wxcrt.h and wx/wxcrtvararg.h,
4029 // so let's include this header now that wxString is defined and it's safe
4030 // to do it:
4031 #include "wx/crt.h"
4032 #endif
4033
4034 #endif // _WX_WXSTRING_H_