2006-04-26 10:16:19 +00:00
|
|
|
/*
|
2008-02-28 19:17:33 +00:00
|
|
|
* Copyright 2004-2008, Haiku, Inc.
|
2006-04-26 10:16:19 +00:00
|
|
|
* Distributed under the terms of the MIT License.
|
|
|
|
*/
|
|
|
|
#ifndef _UTF8_FUNCTIONS_H
|
|
|
|
#define _UTF8_FUNCTIONS_H
|
2004-01-15 07:15:37 +00:00
|
|
|
|
2006-04-25 20:12:06 +00:00
|
|
|
|
|
|
|
#include <SupportDefs.h>
|
|
|
|
|
2004-01-15 07:15:37 +00:00
|
|
|
|
|
|
|
static inline bool
|
|
|
|
IsInsideGlyph(uchar ch)
|
|
|
|
{
|
2006-08-29 22:20:18 +00:00
|
|
|
return (ch & 0xc0) == 0x80;
|
2004-01-15 07:15:37 +00:00
|
|
|
}
|
|
|
|
|
2006-12-25 21:58:00 +00:00
|
|
|
|
2004-01-15 07:15:37 +00:00
|
|
|
static inline uint32
|
2005-06-15 17:09:00 +00:00
|
|
|
UTF8NextCharLenUnsafe(const char *text)
|
2004-01-15 07:15:37 +00:00
|
|
|
{
|
|
|
|
const char *ptr = text;
|
2005-06-15 17:09:00 +00:00
|
|
|
|
2004-01-15 07:15:37 +00:00
|
|
|
do {
|
|
|
|
ptr++;
|
|
|
|
} while (IsInsideGlyph(*ptr));
|
|
|
|
|
|
|
|
return ptr - text;
|
|
|
|
}
|
|
|
|
|
2006-12-25 21:58:00 +00:00
|
|
|
|
2005-06-15 17:09:00 +00:00
|
|
|
static inline uint32
|
|
|
|
UTF8NextCharLen(const char *text)
|
|
|
|
{
|
|
|
|
if (text == NULL || *text == 0)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
return UTF8NextCharLenUnsafe(text);
|
|
|
|
}
|
|
|
|
|
2006-12-25 21:58:00 +00:00
|
|
|
|
2005-05-20 23:51:33 +00:00
|
|
|
static inline uint32
|
|
|
|
UTF8PreviousCharLen(const char *text, const char *limit)
|
|
|
|
{
|
|
|
|
const char *ptr = text;
|
|
|
|
|
|
|
|
if (ptr == NULL || limit == NULL)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
do {
|
|
|
|
if (ptr == limit)
|
|
|
|
break;
|
|
|
|
ptr--;
|
|
|
|
} while (IsInsideGlyph(*ptr));
|
|
|
|
|
|
|
|
return text - ptr;
|
|
|
|
}
|
|
|
|
|
2005-05-13 17:30:59 +00:00
|
|
|
|
2006-12-25 21:58:00 +00:00
|
|
|
/*! UTF8CountBytes gets the length (in bytes) of a UTF8 string. Up to
|
|
|
|
numChars characters are read. If numChars is a negative value it is ignored
|
2006-12-25 22:02:01 +00:00
|
|
|
and the string is read up to the terminating 0.
|
2006-12-25 21:58:00 +00:00
|
|
|
*/
|
2005-06-15 17:09:00 +00:00
|
|
|
static inline uint32
|
2006-12-25 21:58:00 +00:00
|
|
|
UTF8CountBytes(const char *bytes, int32 numChars)
|
2005-06-15 17:09:00 +00:00
|
|
|
{
|
2008-02-28 19:17:33 +00:00
|
|
|
if (bytes == NULL)
|
2006-12-25 21:58:00 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (numChars < 0)
|
|
|
|
numChars = INT_MAX;
|
|
|
|
|
|
|
|
const char *base = bytes;
|
2008-02-28 19:17:33 +00:00
|
|
|
while (bytes[0] != '\0') {
|
|
|
|
if ((bytes[0] & 0xc0) != 0x80) {
|
|
|
|
if (--numChars < 0)
|
|
|
|
break;
|
2005-06-15 17:09:00 +00:00
|
|
|
}
|
2008-02-28 19:17:33 +00:00
|
|
|
bytes++;
|
2005-06-15 17:09:00 +00:00
|
|
|
}
|
2006-12-25 21:58:00 +00:00
|
|
|
|
2008-02-28 19:17:33 +00:00
|
|
|
return bytes - base;
|
2005-05-13 17:30:59 +00:00
|
|
|
}
|
|
|
|
|
2006-12-25 21:58:00 +00:00
|
|
|
|
|
|
|
/*! UTF8CountChars gets the length (in characters) of a UTF8 string. Up to
|
|
|
|
numBytes bytes are read. If numBytes is a negative value it is ignored
|
2006-12-25 22:02:01 +00:00
|
|
|
and the string is read up to the terminating 0.
|
2006-12-25 21:58:00 +00:00
|
|
|
*/
|
2004-01-15 07:15:37 +00:00
|
|
|
static inline uint32
|
2006-12-25 21:58:00 +00:00
|
|
|
UTF8CountChars(const char *bytes, int32 numBytes)
|
2004-01-15 07:15:37 +00:00
|
|
|
{
|
2008-02-28 19:17:33 +00:00
|
|
|
if (bytes == NULL)
|
2006-12-25 21:58:00 +00:00
|
|
|
return 0;
|
|
|
|
|
|
|
|
uint32 length = 0;
|
2008-02-28 19:17:33 +00:00
|
|
|
const char *last;
|
2006-12-25 21:58:00 +00:00
|
|
|
if (numBytes < 0)
|
|
|
|
last = (const char *)UINT_MAX;
|
2008-02-28 19:17:33 +00:00
|
|
|
else
|
|
|
|
last = bytes + numBytes - 1;
|
2006-12-25 21:58:00 +00:00
|
|
|
|
2008-02-28 19:17:33 +00:00
|
|
|
while (bytes[0] && bytes <= last) {
|
|
|
|
if ((bytes++[0] & 0xc0) != 0x80)
|
|
|
|
length++;
|
2005-05-20 23:51:33 +00:00
|
|
|
}
|
|
|
|
|
2006-12-25 21:58:00 +00:00
|
|
|
return length;
|
2004-01-15 07:15:37 +00:00
|
|
|
}
|
|
|
|
|
2005-05-20 23:51:33 +00:00
|
|
|
|
2006-08-29 22:20:18 +00:00
|
|
|
/*! UTF8ToCharCode converts the input that includes potential multibyte chars
|
2006-02-05 23:36:59 +00:00
|
|
|
to UTF-32 char codes that can be used by FreeType. The string pointer is
|
|
|
|
then advanced to the next character in the string. In case the terminating
|
2008-02-28 23:04:47 +00:00
|
|
|
0 is reached, the string pointer is not advanced anymore and nulls are
|
2006-02-05 23:36:59 +00:00
|
|
|
returned. This makes it safe to overruns and enables streamed processing
|
2006-08-29 22:20:18 +00:00
|
|
|
of UTF8 strings.
|
|
|
|
*/
|
2006-02-05 23:36:59 +00:00
|
|
|
static inline uint32
|
|
|
|
UTF8ToCharCode(const char **bytes)
|
|
|
|
{
|
2008-02-28 23:04:47 +00:00
|
|
|
#define UTF8_SUBSTITUTE_CHARACTER 0xfffd
|
|
|
|
|
|
|
|
uint32 result;
|
|
|
|
if (((*bytes)[0] & 0x80) == 0) {
|
|
|
|
// a single byte character
|
|
|
|
result = (*bytes)[0];
|
|
|
|
if (result != '\0') {
|
|
|
|
// do not advance beyond the terminating '\0'
|
|
|
|
(*bytes)++;
|
2006-02-05 23:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2008-02-28 23:04:47 +00:00
|
|
|
if (((*bytes)[0] & 0xc0) == 0x80) {
|
|
|
|
// not a proper multibyte start
|
|
|
|
(*bytes)++;
|
|
|
|
return UTF8_SUBSTITUTE_CHARACTER;
|
2006-02-05 23:36:59 +00:00
|
|
|
}
|
|
|
|
|
2008-02-28 23:04:47 +00:00
|
|
|
// start of a multibyte character
|
|
|
|
uint8 mask = 0x80;
|
|
|
|
result = (uint32)((*bytes)[0] & 0xff);
|
2006-02-05 23:36:59 +00:00
|
|
|
(*bytes)++;
|
2008-02-28 23:04:47 +00:00
|
|
|
|
|
|
|
while (result & mask) {
|
|
|
|
if (mask == 0x02) {
|
|
|
|
// seven byte char - invalid
|
|
|
|
return UTF8_SUBSTITUTE_CHARACTER;
|
|
|
|
}
|
|
|
|
|
|
|
|
result &= ~mask;
|
|
|
|
mask >>= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
while (((*bytes)[0] & 0xc0) == 0x80) {
|
|
|
|
result <<= 6;
|
|
|
|
result += (*bytes)[0] & 0x3f;
|
|
|
|
(*bytes)++;
|
|
|
|
|
|
|
|
mask <<= 1;
|
|
|
|
if (mask == 0x40)
|
2008-10-14 13:16:44 +00:00
|
|
|
return result;
|
2008-02-28 23:04:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (mask == 0x40)
|
|
|
|
return result;
|
|
|
|
|
|
|
|
if ((*bytes)[0] == '\0') {
|
|
|
|
// string terminated within multibyte char
|
|
|
|
return 0x00;
|
|
|
|
}
|
|
|
|
|
|
|
|
// not enough bytes in multibyte char
|
|
|
|
return UTF8_SUBSTITUTE_CHARACTER;
|
|
|
|
|
|
|
|
#undef UTF8_SUBSTITUTE_CHARACTER
|
2006-02-05 23:36:59 +00:00
|
|
|
}
|
|
|
|
|
2006-04-26 10:16:19 +00:00
|
|
|
#endif // _UTF8_FUNCTIONS_H
|