150 lines
5.1 KiB
C++
150 lines
5.1 KiB
C++
// © 2016 and later: Unicode, Inc. and others.
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
/*
|
|
**********************************************************************
|
|
* Copyright (C) 1997-2014, International Business Machines
|
|
* Corporation and others. All Rights Reserved.
|
|
**********************************************************************
|
|
*
|
|
* File USCRIPT.C
|
|
*
|
|
* Modification History:
|
|
*
|
|
* Date Name Description
|
|
* 07/06/2001 Ram Creation.
|
|
******************************************************************************
|
|
*/
|
|
|
|
#include "unicode/uchar.h"
|
|
#include "unicode/uscript.h"
|
|
#include "unicode/uloc.h"
|
|
#include "bytesinkutil.h"
|
|
#include "charstr.h"
|
|
#include "cmemory.h"
|
|
#include "cstring.h"
|
|
#include "ulocimp.h"
|
|
|
|
static const UScriptCode JAPANESE[3] = { USCRIPT_KATAKANA, USCRIPT_HIRAGANA, USCRIPT_HAN };
|
|
static const UScriptCode KOREAN[2] = { USCRIPT_HANGUL, USCRIPT_HAN };
|
|
static const UScriptCode HAN_BOPO[2] = { USCRIPT_HAN, USCRIPT_BOPOMOFO };
|
|
|
|
static int32_t
|
|
setCodes(const UScriptCode *src, int32_t length,
|
|
UScriptCode *dest, int32_t capacity, UErrorCode *err) {
|
|
int32_t i;
|
|
if(U_FAILURE(*err)) { return 0; }
|
|
if(length > capacity) {
|
|
*err = U_BUFFER_OVERFLOW_ERROR;
|
|
return length;
|
|
}
|
|
for(i = 0; i < length; ++i) {
|
|
dest[i] = src[i];
|
|
}
|
|
return length;
|
|
}
|
|
|
|
static int32_t
|
|
setOneCode(UScriptCode script, UScriptCode *scripts, int32_t capacity, UErrorCode *err) {
|
|
if(U_FAILURE(*err)) { return 0; }
|
|
if(1 > capacity) {
|
|
*err = U_BUFFER_OVERFLOW_ERROR;
|
|
return 1;
|
|
}
|
|
scripts[0] = script;
|
|
return 1;
|
|
}
|
|
|
|
static int32_t
|
|
getCodesFromLocale(const char *locale,
|
|
UScriptCode *scripts, int32_t capacity, UErrorCode *err) {
|
|
UErrorCode internalErrorCode = U_ZERO_ERROR;
|
|
char lang[8] = {0};
|
|
char script[8] = {0};
|
|
int32_t scriptLength;
|
|
if(U_FAILURE(*err)) { return 0; }
|
|
// Multi-script languages, equivalent to the LocaleScript data
|
|
// that we used to load from locale resource bundles.
|
|
/*length = */ uloc_getLanguage(locale, lang, UPRV_LENGTHOF(lang), &internalErrorCode);
|
|
if(U_FAILURE(internalErrorCode) || internalErrorCode == U_STRING_NOT_TERMINATED_WARNING) {
|
|
return 0;
|
|
}
|
|
if(0 == uprv_strcmp(lang, "ja")) {
|
|
return setCodes(JAPANESE, UPRV_LENGTHOF(JAPANESE), scripts, capacity, err);
|
|
}
|
|
if(0 == uprv_strcmp(lang, "ko")) {
|
|
return setCodes(KOREAN, UPRV_LENGTHOF(KOREAN), scripts, capacity, err);
|
|
}
|
|
scriptLength = uloc_getScript(locale, script, UPRV_LENGTHOF(script), &internalErrorCode);
|
|
if(U_FAILURE(internalErrorCode) || internalErrorCode == U_STRING_NOT_TERMINATED_WARNING) {
|
|
return 0;
|
|
}
|
|
if(0 == uprv_strcmp(lang, "zh") && 0 == uprv_strcmp(script, "Hant")) {
|
|
return setCodes(HAN_BOPO, UPRV_LENGTHOF(HAN_BOPO), scripts, capacity, err);
|
|
}
|
|
// Explicit script code.
|
|
if(scriptLength != 0) {
|
|
UScriptCode scriptCode = (UScriptCode)u_getPropertyValueEnum(UCHAR_SCRIPT, script);
|
|
if(scriptCode != USCRIPT_INVALID_CODE) {
|
|
if(scriptCode == USCRIPT_SIMPLIFIED_HAN || scriptCode == USCRIPT_TRADITIONAL_HAN) {
|
|
scriptCode = USCRIPT_HAN;
|
|
}
|
|
return setOneCode(scriptCode, scripts, capacity, err);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
/* TODO: this is a bad API and should be deprecated, ticket #11141 */
|
|
U_CAPI int32_t U_EXPORT2
|
|
uscript_getCode(const char* nameOrAbbrOrLocale,
|
|
UScriptCode* fillIn,
|
|
int32_t capacity,
|
|
UErrorCode* err){
|
|
UBool triedCode;
|
|
UErrorCode internalErrorCode;
|
|
int32_t length;
|
|
|
|
if(U_FAILURE(*err)) {
|
|
return 0;
|
|
}
|
|
if(nameOrAbbrOrLocale==NULL ||
|
|
(fillIn == NULL ? capacity != 0 : capacity < 0)) {
|
|
*err = U_ILLEGAL_ARGUMENT_ERROR;
|
|
return 0;
|
|
}
|
|
|
|
triedCode = FALSE;
|
|
if(uprv_strchr(nameOrAbbrOrLocale, '-')==NULL && uprv_strchr(nameOrAbbrOrLocale, '_')==NULL ){
|
|
/* try long and abbreviated script names first */
|
|
UScriptCode code = (UScriptCode) u_getPropertyValueEnum(UCHAR_SCRIPT, nameOrAbbrOrLocale);
|
|
if(code!=USCRIPT_INVALID_CODE) {
|
|
return setOneCode(code, fillIn, capacity, err);
|
|
}
|
|
triedCode = TRUE;
|
|
}
|
|
internalErrorCode = U_ZERO_ERROR;
|
|
length = getCodesFromLocale(nameOrAbbrOrLocale, fillIn, capacity, err);
|
|
if(U_FAILURE(*err) || length != 0) {
|
|
return length;
|
|
}
|
|
icu::CharString likely;
|
|
{
|
|
icu::CharStringByteSink sink(&likely);
|
|
ulocimp_addLikelySubtags(nameOrAbbrOrLocale, sink, &internalErrorCode);
|
|
}
|
|
if(U_SUCCESS(internalErrorCode) && internalErrorCode != U_STRING_NOT_TERMINATED_WARNING) {
|
|
length = getCodesFromLocale(likely.data(), fillIn, capacity, err);
|
|
if(U_FAILURE(*err) || length != 0) {
|
|
return length;
|
|
}
|
|
}
|
|
if(!triedCode) {
|
|
/* still not found .. try long and abbreviated script names again */
|
|
UScriptCode code = (UScriptCode) u_getPropertyValueEnum(UCHAR_SCRIPT, nameOrAbbrOrLocale);
|
|
if(code!=USCRIPT_INVALID_CODE) {
|
|
return setOneCode(code, fillIn, capacity, err);
|
|
}
|
|
}
|
|
return 0;
|
|
}
|