/* * Copyright (C) 2008 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ #define LOG_TAG "Resources" #include "JNIHelp.h" #include "AndroidSystemNatives.h" #include "cutils/log.h" #include "unicode/numfmt.h" #include "unicode/locid.h" #include "unicode/ucal.h" #include "unicode/gregocal.h" #include "unicode/ucurr.h" #include "unicode/calendar.h" #include "unicode/datefmt.h" #include "unicode/dtfmtsym.h" #include "unicode/decimfmt.h" #include "unicode/dcfmtsym.h" #include "unicode/uclean.h" #include "unicode/smpdtfmt.h" #include "unicode/strenum.h" #include "unicode/ustring.h" #include "unicode/timezone.h" #include "ErrorCode.h" #include <stdlib.h> #include <string.h> #include <time.h> #include <sys/time.h> static jclass string_class; class ScopedResourceBundle { public: ScopedResourceBundle(UResourceBundle* bundle) : mBundle(bundle) { } ~ScopedResourceBundle() { if (mBundle != NULL) { ures_close(mBundle); } } UResourceBundle* get() { return mBundle; } private: UResourceBundle* mBundle; }; static Locale getLocale(JNIEnv* env, jstring locale) { const char* name = env->GetStringUTFChars(locale, NULL); Locale result = Locale::createFromName(name); env->ReleaseStringUTFChars(locale, name); return result; } static jint getCurrencyFractionDigitsNative(JNIEnv* env, jclass clazz, jstring currencyCode) { UErrorCode status = U_ZERO_ERROR; NumberFormat* fmt = NumberFormat::createCurrencyInstance(status); if (U_FAILURE(status)) { return -1; } const jchar* cCode = env->GetStringChars(currencyCode, NULL); fmt->setCurrency(cCode, status); env->ReleaseStringChars(currencyCode, cCode); if (U_FAILURE(status)) { return -1; } // for CurrencyFormats the minimum and maximum fraction digits are the same. int result = fmt->getMinimumFractionDigits(); delete fmt; return result; } static jstring getCurrencyCodeNative(JNIEnv* env, jclass clazz, jstring key) { UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle supplData(ures_openDirect(NULL, "supplementalData", &status)); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle currencyMap(ures_getByKey(supplData.get(), "CurrencyMap", NULL, &status)); if (U_FAILURE(status)) { return NULL; } const char* keyChars = env->GetStringUTFChars(key, NULL); ScopedResourceBundle currency(ures_getByKey(currencyMap.get(), keyChars, NULL, &status)); env->ReleaseStringUTFChars(key, keyChars); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle currencyElem(ures_getByIndex(currency.get(), 0, NULL, &status)); if (U_FAILURE(status)) { return env->NewStringUTF("None"); } // check if there is a 'to' date. If there is, the currency isn't used anymore. ScopedResourceBundle currencyTo(ures_getByKey(currencyElem.get(), "to", NULL, &status)); if (!U_FAILURE(status)) { // return and let the caller throw an exception return NULL; } // We need to reset 'status'. It works like errno in that ICU doesn't set it // to U_ZERO_ERROR on success: it only touches it on error, and the test // above means it now holds a failure code. status = U_ZERO_ERROR; ScopedResourceBundle currencyId(ures_getByKey(currencyElem.get(), "id", NULL, &status)); if (U_FAILURE(status)) { // No id defined for this country return env->NewStringUTF("None"); } int length; const jchar* id = ures_getString(currencyId.get(), &length, &status); if (U_FAILURE(status) || length == 0) { return env->NewStringUTF("None"); } return env->NewString(id, length); } static jstring getCurrencySymbolNative(JNIEnv* env, jclass clazz, jstring locale, jstring currencyCode) { // LOGI("ENTER getCurrencySymbolNative"); const char* locName = env->GetStringUTFChars(locale, NULL); UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle root(ures_open(NULL, locName, &status)); env->ReleaseStringUTFChars(locale, locName); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle currencies(ures_getByKey(root.get(), "Currencies", NULL, &status)); if (U_FAILURE(status)) { return NULL; } const char* currName = env->GetStringUTFChars(currencyCode, NULL); ScopedResourceBundle currencyElems(ures_getByKey(currencies.get(), currName, NULL, &status)); env->ReleaseStringUTFChars(currencyCode, currName); if (U_FAILURE(status)) { return NULL; } int currSymbL; const jchar* currSymbU = ures_getStringByIndex(currencyElems.get(), 0, &currSymbL, &status); if (U_FAILURE(status)) { return NULL; } return (currSymbL == 0) ? NULL : env->NewString(currSymbU, currSymbL); } static jstring getDisplayCountryNative(JNIEnv* env, jclass clazz, jstring targetLocale, jstring locale) { Locale loc = getLocale(env, locale); Locale targetLoc = getLocale(env, targetLocale); UnicodeString str; targetLoc.getDisplayCountry(loc, str); return env->NewString(str.getBuffer(), str.length()); } static jstring getDisplayLanguageNative(JNIEnv* env, jclass clazz, jstring targetLocale, jstring locale) { Locale loc = getLocale(env, locale); Locale targetLoc = getLocale(env, targetLocale); UnicodeString str; targetLoc.getDisplayLanguage(loc, str); return env->NewString(str.getBuffer(), str.length()); } static jstring getDisplayVariantNative(JNIEnv* env, jclass clazz, jstring targetLocale, jstring locale) { Locale loc = getLocale(env, locale); Locale targetLoc = getLocale(env, targetLocale); UnicodeString str; targetLoc.getDisplayVariant(loc, str); return env->NewString(str.getBuffer(), str.length()); } static jstring getISO3CountryNative(JNIEnv* env, jclass clazz, jstring locale) { Locale loc = getLocale(env, locale); return env->NewStringUTF(loc.getISO3Country()); } static jstring getISO3LanguageNative(JNIEnv* env, jclass clazz, jstring locale) { Locale loc = getLocale(env, locale); return env->NewStringUTF(loc.getISO3Language()); } static jobjectArray toStringArray(JNIEnv* env, const char* const* strings) { size_t count = 0; while (strings[count] != NULL) { ++count; } jobjectArray result = env->NewObjectArray(count, string_class, NULL); for (size_t i = 0; i < count; ++i) { jstring s = env->NewStringUTF(strings[i]); env->SetObjectArrayElement(result, i, s); env->DeleteLocalRef(s); } return result; } static jobjectArray getISOCountriesNative(JNIEnv* env, jclass clazz) { return toStringArray(env, Locale::getISOCountries()); } static jobjectArray getISOLanguagesNative(JNIEnv* env, jclass clazz) { return toStringArray(env, Locale::getISOLanguages()); } static jobjectArray getAvailableLocalesNative(JNIEnv* env, jclass clazz) { size_t count = uloc_countAvailable(); jobjectArray result = env->NewObjectArray(count, string_class, NULL); for (size_t i = 0; i < count; ++i) { jstring s = env->NewStringUTF(uloc_getAvailable(i)); env->SetObjectArrayElement(result, i, s); env->DeleteLocalRef(s); } return result; } static TimeZone* timeZoneFromId(JNIEnv* env, jstring id) { const jchar* chars = env->GetStringChars(id, NULL); const UnicodeString zoneID(reinterpret_cast<const UChar*>(chars), env->GetStringLength(id)); env->ReleaseStringChars(id, chars); return TimeZone::createTimeZone(zoneID); } static jstring formatDate(JNIEnv* env, const SimpleDateFormat& fmt, const UDate& when) { UnicodeString str; fmt.format(when, str); return env->NewString(str.getBuffer(), str.length()); } static void getTimeZonesNative(JNIEnv* env, jclass clazz, jobjectArray outerArray, jstring locale) { // get all timezone objects jobjectArray zoneIdArray = (jobjectArray) env->GetObjectArrayElement(outerArray, 0); int count = env->GetArrayLength(zoneIdArray); TimeZone* zones[count]; for(int i = 0; i < count; i++) { jstring id = (jstring) env->GetObjectArrayElement(zoneIdArray, i); zones[i] = timeZoneFromId(env, id); env->DeleteLocalRef(id); } Locale loc = getLocale(env, locale); UErrorCode status = U_ZERO_ERROR; UnicodeString longPattern("zzzz",""); SimpleDateFormat longFormat(longPattern, loc, status); UnicodeString shortPattern("z",""); SimpleDateFormat shortFormat(shortPattern, loc, status); jobjectArray longStdTimeArray = (jobjectArray) env->GetObjectArrayElement(outerArray, 1); jobjectArray shortStdTimeArray = (jobjectArray) env->GetObjectArrayElement(outerArray, 2); jobjectArray longDlTimeArray = (jobjectArray) env->GetObjectArrayElement(outerArray, 3); jobjectArray shortDlTimeArray = (jobjectArray) env->GetObjectArrayElement(outerArray, 4); // 15th January 2008 UDate date1 = 1203105600000.0; // 15th July 2008 UDate date2 = 1218826800000.0; for (int i = 0; i < count; ++i) { TimeZone* tz = zones[i]; longFormat.setTimeZone(*tz); shortFormat.setTimeZone(*tz); int32_t daylightOffset; int32_t rawOffset; tz->getOffset(date1, false, rawOffset, daylightOffset, status); UDate standardDate; UDate daylightSavingDate; if (daylightOffset != 0) { // The Timezone is reporting that we are in daylight time // for the winter date. The dates are for the wrong hemisphere, // swap them. standardDate = date2; daylightSavingDate = date1; } else { standardDate = date1; daylightSavingDate = date2; } jstring content = formatDate(env, shortFormat, daylightSavingDate); env->SetObjectArrayElement(shortDlTimeArray, i, content); env->DeleteLocalRef(content); content = formatDate(env, shortFormat, standardDate); env->SetObjectArrayElement(shortStdTimeArray, i, content); env->DeleteLocalRef(content); content = formatDate(env, longFormat, daylightSavingDate); env->SetObjectArrayElement(longDlTimeArray, i, content); env->DeleteLocalRef(content); content = formatDate(env, longFormat, standardDate); env->SetObjectArrayElement(longStdTimeArray, i, content); env->DeleteLocalRef(content); delete tz; } } static jstring getDisplayTimeZoneNative(JNIEnv* env, jclass clazz, jstring zoneId, jboolean isDST, jint style, jstring localeId) { TimeZone* zone = timeZoneFromId(env, zoneId); Locale locale = getLocale(env, localeId); // Try to get the display name of the TimeZone according to the Locale UnicodeString displayName; zone->getDisplayName((UBool)isDST, (style == 0 ? TimeZone::SHORT : TimeZone::LONG), locale, displayName); jstring result = env->NewString(displayName.getBuffer(), displayName.length()); delete zone; return result; } static bool getDayIntVector(JNIEnv* env, UResourceBundle* gregorian, int* values) { // get the First day of week and the minimal days in first week numbers UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "DateTimeElements", NULL, &status)); if (U_FAILURE(status)) { return false; } int intVectSize; const int* result = ures_getIntVector(gregorianElems.get(), &intVectSize, &status); if (U_FAILURE(status) || intVectSize != 2) { return false; } values[0] = result[0]; values[1] = result[1]; return true; } static jobjectArray getAmPmMarkers(JNIEnv* env, UResourceBundle* gregorian) { UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "AmPmMarkers", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ures_resetIterator(gregorianElems.get()); int lengthAm, lengthPm; const jchar* am = ures_getStringByIndex(gregorianElems.get(), 0, &lengthAm, &status); const jchar* pm = ures_getStringByIndex(gregorianElems.get(), 1, &lengthPm, &status); if (U_FAILURE(status)) { return NULL; } jobjectArray amPmMarkers = env->NewObjectArray(2, string_class, NULL); jstring amU = env->NewString(am, lengthAm); env->SetObjectArrayElement(amPmMarkers, 0, amU); env->DeleteLocalRef(amU); jstring pmU = env->NewString(pm, lengthPm); env->SetObjectArrayElement(amPmMarkers, 1, pmU); env->DeleteLocalRef(pmU); return amPmMarkers; } static jobjectArray getEras(JNIEnv* env, UResourceBundle* gregorian) { UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "eras", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle eraElems(ures_getByKey(gregorianElems.get(), "abbreviated", NULL, &status)); if (U_FAILURE(status)) { return NULL; } int eraCount = ures_getSize(eraElems.get()); jobjectArray eras = env->NewObjectArray(eraCount, string_class, NULL); ures_resetIterator(eraElems.get()); for (int i = 0; i < eraCount; ++i) { int eraLength; const jchar* era = ures_getStringByIndex(eraElems.get(), i, &eraLength, &status); if (U_FAILURE(status)) { return NULL; } jstring eraU = env->NewString(era, eraLength); env->SetObjectArrayElement(eras, i, eraU); env->DeleteLocalRef(eraU); } return eras; } static jobjectArray getMonthNames(JNIEnv* env, UResourceBundle* gregorian, bool longNames) { UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "monthNames", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle monthNameElems(ures_getByKey(gregorianElems.get(), "format", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle monthNameElemsFormat(ures_getByKey(monthNameElems.get(), longNames ? "wide" : "abbreviated", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ures_resetIterator(monthNameElemsFormat.get()); int monthCount = ures_getSize(monthNameElemsFormat.get()); // the array length is +1 because the harmony locales had an empty string at the end of their month name array jobjectArray months = env->NewObjectArray(monthCount + 1, string_class, NULL); for (int i = 0; i < monthCount; ++i) { int monthNameLength; const jchar* month = ures_getStringByIndex(monthNameElemsFormat.get(), i, &monthNameLength, &status); if (U_FAILURE(status)) { return NULL; } jstring monthU = env->NewString(month, monthNameLength); env->SetObjectArrayElement(months, i, monthU); env->DeleteLocalRef(monthU); } jstring monthU = env->NewStringUTF(""); env->SetObjectArrayElement(months, monthCount, monthU); env->DeleteLocalRef(monthU); return months; } static jobjectArray getLongMonthNames(JNIEnv* env, UResourceBundle* gregorian) { return getMonthNames(env, gregorian, true); } static jobjectArray getShortMonthNames(JNIEnv* env, UResourceBundle* gregorian) { return getMonthNames(env, gregorian, false); } static jobjectArray getWeekdayNames(JNIEnv* env, UResourceBundle* gregorian, bool longNames) { UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle gregorianElems(ures_getByKey(gregorian, "dayNames", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle dayNameElems(ures_getByKey(gregorianElems.get(), "format", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ScopedResourceBundle dayNameElemsFormat(ures_getByKey(dayNameElems.get(), longNames ? "wide" : "abbreviated", NULL, &status)); if (U_FAILURE(status)) { return NULL; } ures_resetIterator(dayNameElemsFormat.get()); int dayCount = ures_getSize(dayNameElemsFormat.get()); jobjectArray weekdays = env->NewObjectArray(dayCount + 1, string_class, NULL); // first entry in the weekdays array is an empty string env->SetObjectArrayElement(weekdays, 0, env->NewStringUTF("")); for(int i = 0; i < dayCount; i++) { int dayNameLength; const jchar* day = ures_getStringByIndex(dayNameElemsFormat.get(), i, &dayNameLength, &status); if(U_FAILURE(status)) { return NULL; } jstring dayU = env->NewString(day, dayNameLength); env->SetObjectArrayElement(weekdays, i + 1, dayU); env->DeleteLocalRef(dayU); } return weekdays; } static jobjectArray getLongWeekdayNames(JNIEnv* env, UResourceBundle* gregorian) { return getWeekdayNames(env, gregorian, true); } static jobjectArray getShortWeekdayNames(JNIEnv* env, UResourceBundle* gregorian) { return getWeekdayNames(env, gregorian, false); } static jstring getDecimalPatternChars(JNIEnv* env, UResourceBundle* rootElems) { UErrorCode status = U_ZERO_ERROR; int zeroL, digitL, decSepL, groupL, listL, percentL, permillL, expL, currSepL, minusL; const jchar* zero = ures_getStringByIndex(rootElems, 4, &zeroL, &status); const jchar* digit = ures_getStringByIndex(rootElems, 5, &digitL, &status); const jchar* decSep = ures_getStringByIndex(rootElems, 0, &decSepL, &status); const jchar* group = ures_getStringByIndex(rootElems, 1, &groupL, &status); const jchar* list = ures_getStringByIndex(rootElems, 2, &listL, &status); const jchar* percent = ures_getStringByIndex(rootElems, 3, &percentL, &status); const jchar* permill = ures_getStringByIndex(rootElems, 8, &permillL, &status); const jchar* exp = ures_getStringByIndex(rootElems, 7, &expL, &status); const jchar* currSep = ures_getStringByIndex(rootElems, 0, &currSepL, &status); const jchar* minus = ures_getStringByIndex(rootElems, 6, &minusL, &status); if (U_FAILURE(status)) { return NULL; } jchar patternChars[11]; patternChars[0] = 0; u_strncat(patternChars, zero, 1); u_strncat(patternChars, digit, 1); u_strncat(patternChars, decSep, 1); u_strncat(patternChars, group, 1); u_strncat(patternChars, list, 1); u_strncat(patternChars, percent, 1); u_strncat(patternChars, permill, 1); u_strncat(patternChars, exp, 1); u_strncat(patternChars, currSep, 1); u_strncat(patternChars, minus, 1); return env->NewString(patternChars, 10); } static jstring getIntCurrencyCode(JNIEnv* env, jstring locale) { const char* locStr = env->GetStringUTFChars(locale, NULL); // Extract the 2-character country name. if (strlen(locStr) < 5) { env->ReleaseStringUTFChars(locale, locStr); return NULL; } if (locStr[3] < 'A' || locStr[3] > 'Z' || locStr[4] < 'A' || locStr[4] > 'Z') { env->ReleaseStringUTFChars(locale, locStr); return NULL; } char country[3] = {0,0,0}; country[0] = locStr[3]; country[1] = locStr[4]; env->ReleaseStringUTFChars(locale, locStr); return getCurrencyCodeNative(env, NULL, env->NewStringUTF(country)); } static void setIntegerField(JNIEnv* env, jobject obj, const char* fieldName, int value) { // Convert our int to a java.lang.Integer. // TODO: switch to Integer.valueOf, add error checking. jclass integerClass = env->FindClass("java/lang/Integer"); jmethodID constructor = env->GetMethodID(integerClass, "<init>", "(I)V"); jobject integerValue = env->NewObject(integerClass, constructor, value); // Set the field. jclass localeDataClass = env->FindClass("com/ibm/icu4jni/util/LocaleData"); jfieldID fid = env->GetFieldID(localeDataClass, fieldName, "Ljava/lang/Integer;"); env->SetObjectField(obj, fid, integerValue); } static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, jstring value) { jclass localeDataClass = env->FindClass("com/ibm/icu4jni/util/LocaleData"); jfieldID fid = env->GetFieldID(localeDataClass, fieldName, "Ljava/lang/String;"); env->SetObjectField(obj, fid, value); } static void setStringArrayField(JNIEnv* env, jobject obj, const char* fieldName, jobjectArray value) { jclass localeDataClass = env->FindClass("com/ibm/icu4jni/util/LocaleData"); jfieldID fid = env->GetFieldID(localeDataClass, fieldName, "[Ljava/lang/String;"); env->SetObjectField(obj, fid, value); } static void setStringField(JNIEnv* env, jobject obj, const char* fieldName, UResourceBundle* bundle, int index) { UErrorCode status = U_ZERO_ERROR; int charCount; const UChar* chars = ures_getStringByIndex(bundle, index, &charCount, &status); if (U_SUCCESS(status)) { setStringField(env, obj, fieldName, env->NewString(chars, charCount)); } } static jboolean initLocaleDataImpl(JNIEnv* env, jclass clazz, jstring locale, jobject localeData) { const char* loc = env->GetStringUTFChars(locale, NULL); UErrorCode status = U_ZERO_ERROR; ScopedResourceBundle root(ures_openU(NULL, loc, &status)); env->ReleaseStringUTFChars(locale, loc); if (U_FAILURE(status)) { LOGE("Error getting ICU resource bundle: %s", u_errorName(status)); status = U_ZERO_ERROR; return JNI_FALSE; } ScopedResourceBundle calendar(ures_getByKey(root.get(), "calendar", NULL, &status)); if (U_FAILURE(status)) { LOGE("Error getting ICU calendar resource bundle: %s", u_errorName(status)); return JNI_FALSE; } ScopedResourceBundle gregorian(ures_getByKey(calendar.get(), "gregorian", NULL, &status)); if (U_FAILURE(status)) { LOGE("Error getting ICU gregorian resource bundle: %s", u_errorName(status)); return JNI_FALSE; } int firstDayVals[2]; if (getDayIntVector(env, gregorian.get(), firstDayVals)) { setIntegerField(env, localeData, "firstDayOfWeek", firstDayVals[0]); setIntegerField(env, localeData, "minimalDaysInFirstWeek", firstDayVals[1]); } setStringArrayField(env, localeData, "amPm", getAmPmMarkers(env, gregorian.get())); setStringArrayField(env, localeData, "eras", getEras(env, gregorian.get())); setStringArrayField(env, localeData, "longMonthNames", getLongMonthNames(env, gregorian.get())); setStringArrayField(env, localeData, "shortMonthNames", getShortMonthNames(env, gregorian.get())); setStringArrayField(env, localeData, "longWeekdayNames", getLongWeekdayNames(env, gregorian.get())); setStringArrayField(env, localeData, "shortWeekdayNames", getShortWeekdayNames(env, gregorian.get())); ScopedResourceBundle gregorianElems(ures_getByKey(gregorian.get(), "DateTimePatterns", NULL, &status)); if (U_SUCCESS(status)) { setStringField(env, localeData, "fullTimeFormat", gregorianElems.get(), 0); setStringField(env, localeData, "longTimeFormat", gregorianElems.get(), 1); setStringField(env, localeData, "mediumTimeFormat", gregorianElems.get(), 2); setStringField(env, localeData, "shortTimeFormat", gregorianElems.get(), 3); setStringField(env, localeData, "fullDateFormat", gregorianElems.get(), 4); setStringField(env, localeData, "longDateFormat", gregorianElems.get(), 5); setStringField(env, localeData, "mediumDateFormat", gregorianElems.get(), 6); setStringField(env, localeData, "shortDateFormat", gregorianElems.get(), 7); } status = U_ZERO_ERROR; ScopedResourceBundle numberElements(ures_getByKey(root.get(), "NumberElements", NULL, &status)); if (U_SUCCESS(status) && ures_getSize(numberElements.get()) >= 11) { setStringField(env, localeData, "decimalPatternChars", getDecimalPatternChars(env, numberElements.get())); setStringField(env, localeData, "infinity", numberElements.get(), 9); setStringField(env, localeData, "NaN", numberElements.get(), 10); } status = U_ZERO_ERROR; jstring internationalCurrencySymbol = getIntCurrencyCode(env, locale); jstring currencySymbol = NULL; if (internationalCurrencySymbol != NULL) { currencySymbol = getCurrencySymbolNative(env, clazz, locale, internationalCurrencySymbol); } else { internationalCurrencySymbol = env->NewStringUTF("XXX"); } if (currencySymbol == NULL) { // This is the UTF-8 encoding of U+00A4 (CURRENCY SIGN). currencySymbol = env->NewStringUTF("\xc2\xa4"); } setStringField(env, localeData, "currencySymbol", currencySymbol); setStringField(env, localeData, "internationalCurrencySymbol", internationalCurrencySymbol); ScopedResourceBundle numberPatterns(ures_getByKey(root.get(), "NumberPatterns", NULL, &status)); if (U_SUCCESS(status) && ures_getSize(numberPatterns.get()) >= 3) { setStringField(env, localeData, "numberPattern", numberPatterns.get(), 0); setStringField(env, localeData, "currencyPattern", numberPatterns.get(), 1); setStringField(env, localeData, "percentPattern", numberPatterns.get(), 2); } return JNI_TRUE; } static JNINativeMethod gMethods[] = { /* name, signature, funcPtr */ {"getCurrencyFractionDigitsNative", "(Ljava/lang/String;)I", (void*) getCurrencyFractionDigitsNative}, {"getCurrencyCodeNative", "(Ljava/lang/String;)Ljava/lang/String;", (void*) getCurrencyCodeNative}, {"getCurrencySymbolNative", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;", (void*) getCurrencySymbolNative}, {"getDisplayCountryNative", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;", (void*) getDisplayCountryNative}, {"getDisplayLanguageNative", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;", (void*) getDisplayLanguageNative}, {"getDisplayVariantNative", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/String;", (void*) getDisplayVariantNative}, {"getISO3CountryNative", "(Ljava/lang/String;)Ljava/lang/String;", (void*) getISO3CountryNative}, {"getISO3LanguageNative", "(Ljava/lang/String;)Ljava/lang/String;", (void*) getISO3LanguageNative}, {"getISOCountriesNative", "()[Ljava/lang/String;", (void*) getISOCountriesNative}, {"getISOLanguagesNative", "()[Ljava/lang/String;", (void*) getISOLanguagesNative}, {"getAvailableLocalesNative", "()[Ljava/lang/String;", (void*) getAvailableLocalesNative}, {"getTimeZonesNative", "([[Ljava/lang/String;Ljava/lang/String;)V", (void*) getTimeZonesNative}, {"getDisplayTimeZoneNative", "(Ljava/lang/String;ZILjava/lang/String;)Ljava/lang/String;", (void*) getDisplayTimeZoneNative}, {"initLocaleDataImpl", "(Ljava/lang/String;Lcom/ibm/icu4jni/util/LocaleData;)Z", (void*) initLocaleDataImpl}, }; int register_com_ibm_icu4jni_util_Resources(JNIEnv* env) { jclass stringclass = env->FindClass("java/lang/String"); if (stringclass == NULL) { return -1; } string_class = (jclass) env->NewGlobalRef(stringclass); return jniRegisterNativeMethods(env, "com/ibm/icu4jni/util/Resources", gMethods, NELEM(gMethods)); }