]>
git.saurik.com Git - wxWidgets.git/blob - src/common/stringops.cpp
c69a18774fb8ba7713a09554e43d831c3a31828f
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/stringops.cpp
3 // Purpose: implementation of wxString primitive operations
4 // Author: Vaclav Slavik
8 // Copyright: (c) 2007 REA Elektronik GmbH
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ===========================================================================
14 // ===========================================================================
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
24 #include "wx/stringops.h"
27 // ===========================================================================
29 // ===========================================================================
31 #if wxUSE_UNICODE_UTF8
33 // ---------------------------------------------------------------------------
34 // UTF-8 sequences lengths
35 // ---------------------------------------------------------------------------
37 const unsigned char wxStringOperationsUtf8::ms_utf8IterTable
[256] = {
38 // single-byte sequences (ASCII):
39 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 00..0F
40 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 10..1F
41 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 20..2F
42 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 30..3F
43 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 40..4F
44 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 50..5F
45 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 60..6F
46 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 70..7F
48 // these are invalid, we use step 1 to skip
49 // over them (should never happen):
50 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 80..8F
51 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // 90..9F
52 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // A0..AF
53 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, // B0..BF
56 // two-byte sequences:
57 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, // C2..CF
58 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, 2, // D0..DF
60 // three-byte sequences:
61 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, 3, // E0..EF
63 // four-byte sequences:
64 4, 4, 4, 4, 4, // F0..F4
66 // these are invalid again (5- or 6-byte
67 // sequences and sequences for code points
68 // above U+10FFFF, as restricted by RFC 3629):
69 1, 1, 1, 1, 1, 1, 1, 1, 1, 1, 1 // F5..FF
72 // ---------------------------------------------------------------------------
74 // ---------------------------------------------------------------------------
77 // Table 3.1B from Unicode spec: Legal UTF-8 Byte Sequences
79 // Code Points | 1st Byte | 2nd Byte | 3rd Byte | 4th Byte |
80 // -------------------+----------+----------+----------+----------+
81 // U+0000..U+007F | 00..7F | | | |
82 // U+0080..U+07FF | C2..DF | 80..BF | | |
83 // U+0800..U+0FFF | E0 | A0..BF | 80..BF | |
84 // U+1000..U+FFFF | E1..EF | 80..BF | 80..BF | |
85 // U+10000..U+3FFFF | F0 | 90..BF | 80..BF | 80..BF |
86 // U+40000..U+FFFFF | F1..F3 | 80..BF | 80..BF | 80..BF |
87 // U+100000..U+10FFFF | F4 | 80..8F | 80..BF | 80..BF |
88 // -------------------+----------+----------+----------+----------+
90 bool wxStringOperationsUtf8::IsValidUtf8String(const char *str
, size_t len
)
93 return true; // empty string is UTF8 string
95 const unsigned char *c
= (const unsigned char*)str
;
96 const unsigned char * const end
= (len
== wxStringImpl::npos
) ? NULL
: c
+ len
;
98 for ( ; c
!= end
&& *c
; ++c
)
100 unsigned char b
= *c
;
104 // if the string is not NULL-terminated, verify we have enough
105 // bytes in it left for current character's encoding:
106 if ( c
+ ms_utf8IterTable
[*c
] > end
)
110 if ( b
<= 0x7F ) // 00..7F
113 else if ( b
< 0xC2 ) // invalid lead bytes: 80..C1
116 // two-byte sequences:
117 else if ( b
<= 0xDF ) // C2..DF
120 if ( !(b
>= 0x80 && b
<= 0xBF ) )
124 // three-byte sequences:
125 else if ( b
== 0xE0 )
128 if ( !(b
>= 0xA0 && b
<= 0xBF ) )
131 if ( !(b
>= 0x80 && b
<= 0xBF ) )
134 else if ( b
== 0xED )
137 if ( !(b
>= 0x80 && b
<= 0x9F ) )
140 if ( !(b
>= 0x80 && b
<= 0xBF ) )
143 else if ( b
<= 0xEF ) // E1..EC EE..EF
145 for ( int i
= 0; i
< 2; ++i
)
148 if ( !(b
>= 0x80 && b
<= 0xBF ) )
153 // four-byte sequences:
154 else if ( b
== 0xF0 )
157 if ( !(b
>= 0x90 && b
<= 0xBF ) )
159 for ( int i
= 0; i
< 2; ++i
)
162 if ( !(b
>= 0x80 && b
<= 0xBF ) )
166 else if ( b
<= 0xF3 ) // F1..F3
168 for ( int i
= 0; i
< 3; ++i
)
171 if ( !(b
>= 0x80 && b
<= 0xBF ) )
175 else if ( b
== 0xF4 )
178 if ( !(b
>= 0x80 && b
<= 0x8F ) )
180 for ( int i
= 0; i
< 2; ++i
)
183 if ( !(b
>= 0x80 && b
<= 0xBF ) )
187 else // otherwise, it's invalid lead byte
194 // NB: this is in this file and not unichar.cpp to keep all UTF-8 encoding
195 // code in single place
196 wxUniChar::Utf8CharBuffer
wxUniChar::AsUTF8() const
198 Utf8CharBuffer buf
= { "" }; // init to avoid g++ 4.1 warning with -O2
199 char *out
= buf
.data
;
201 value_type code
= GetValue();
203 // Char. number range | UTF-8 octet sequence
204 // (hexadecimal) | (binary)
205 // ----------------------+---------------------------------------------
206 // 0000 0000 - 0000 007F | 0xxxxxxx
207 // 0000 0080 - 0000 07FF | 110xxxxx 10xxxxxx
208 // 0000 0800 - 0000 FFFF | 1110xxxx 10xxxxxx 10xxxxxx
209 // 0001 0000 - 0010 FFFF | 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx
211 // Code point value is stored in bits marked with 'x', lowest-order bit
212 // of the value on the right side in the diagram above.
220 else if ( code
<= 0x07FF )
223 // NB: this line takes 6 least significant bits, encodes them as
224 // 10xxxxxx and discards them so that the next byte can be encoded:
225 out
[1] = 0x80 | (code
& 0x3F); code
>>= 6;
226 out
[0] = 0xC0 | code
;
228 else if ( code
< 0xFFFF )
231 out
[2] = 0x80 | (code
& 0x3F); code
>>= 6;
232 out
[1] = 0x80 | (code
& 0x3F); code
>>= 6;
233 out
[0] = 0xE0 | code
;
235 else if ( code
<= 0x10FFFF )
238 out
[3] = 0x80 | (code
& 0x3F); code
>>= 6;
239 out
[2] = 0x80 | (code
& 0x3F); code
>>= 6;
240 out
[1] = 0x80 | (code
& 0x3F); code
>>= 6;
241 out
[0] = 0xF0 | code
;
245 wxFAIL_MSG( _T("trying to encode undefined Unicode character") );
253 wxStringOperationsUtf8::DecodeNonAsciiChar(wxStringImpl::const_iterator i
)
255 wxASSERT( IsValidUtf8LeadByte(*i
) );
257 wxUniChar::value_type code
= 0;
258 size_t len
= GetUtf8CharLength(*i
);
259 wxASSERT_MSG( len
<= 4, _T("invalid UTF-8 sequence length") );
261 // Char. number range | UTF-8 octet sequence
262 // (hexadecimal) | (binary)
263 // ----------------------+---------------------------------------------
264 // 0000 0000 - 0000 007F | 0xxxxxxx
265 // 0000 0080 - 0000 07FF | 110xxxxx 10xxxxxx
266 // 0000 0800 - 0000 FFFF | 1110xxxx 10xxxxxx 10xxxxxx
267 // 0001 0000 - 0010 FFFF | 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx
269 // Code point value is stored in bits marked with 'x', lowest-order bit
270 // of the value on the right side in the diagram above.
273 // mask to extract lead byte's value ('x' bits above), by sequence's length:
274 static const unsigned char s_leadValueMask
[4] = { 0x7F, 0x1F, 0x0F, 0x07 };
276 // mask and value of lead byte's most significant bits, by length:
277 static const unsigned char s_leadMarkerMask
[4] = { 0x80, 0xE0, 0xF0, 0xF8 };
278 static const unsigned char s_leadMarkerVal
[4] = { 0x00, 0xC0, 0xE0, 0xF0 };
281 // extract the lead byte's value bits:
282 wxASSERT_MSG( ((unsigned char)*i
& s_leadMarkerMask
[len
-1]) ==
283 s_leadMarkerVal
[len
-1],
284 _T("invalid UTF-8 lead byte") );
285 code
= (unsigned char)*i
& s_leadValueMask
[len
-1];
287 // all remaining bytes, if any, are handled in the same way regardless of
288 // sequence's length:
289 for ( ++i
; len
> 1; --len
, ++i
)
291 wxASSERT_MSG( ((unsigned char)*i
& 0xC0) == 0x80,
292 _T("invalid UTF-8 byte") );
295 code
|= (unsigned char)*i
& 0x3F;
298 return wxUniChar(code
);
301 wxCharBuffer
wxStringOperationsUtf8::EncodeNChars(size_t n
, const wxUniChar
& ch
)
303 Utf8CharBuffer
once(EncodeChar(ch
));
304 // the IncIter() table can be used to determine the length of ch's encoding:
305 size_t len
= ms_utf8IterTable
[(unsigned char)once
.data
[0]];
307 wxCharBuffer
buf(n
* len
);
308 char *ptr
= buf
.data();
309 for ( size_t i
= 0; i
< n
; i
++, ptr
+= len
)
311 memcpy(ptr
, once
.data
, len
);
317 #endif // wxUSE_UNICODE_UTF8