mirror of
https://github.com/FirebirdSQL/firebird.git
synced 2025-01-24 23:23:03 +01:00
1432 lines
44 KiB
C++
1432 lines
44 KiB
C++
|
/**
|
||
|
*******************************************************************************
|
||
|
* Copyright (C) 2001-2004, International Business Machines Corporation and *
|
||
|
* others. All Rights Reserved. *
|
||
|
*******************************************************************************
|
||
|
*
|
||
|
*******************************************************************************
|
||
|
*/
|
||
|
|
||
|
#include "unicode/utypes.h"
|
||
|
|
||
|
#if !UCONFIG_NO_SERVICE
|
||
|
|
||
|
#include "icusvtst.h"
|
||
|
#include "iculserv.h"
|
||
|
#include <stdio.h>
|
||
|
|
||
|
|
||
|
class MyListener : public EventListener {
|
||
|
};
|
||
|
|
||
|
class WrongListener : public EventListener {
|
||
|
};
|
||
|
|
||
|
class ICUNSubclass : public ICUNotifier {
|
||
|
public:
|
||
|
UBool acceptsListener(const EventListener& l) const {
|
||
|
return TRUE;
|
||
|
// return l instanceof MyListener;
|
||
|
}
|
||
|
|
||
|
virtual void notifyListener(EventListener& l) const {
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class LKFSubclass : public LocaleKeyFactory {
|
||
|
Hashtable table;
|
||
|
|
||
|
public:
|
||
|
LKFSubclass(UBool visible)
|
||
|
: LocaleKeyFactory(visible ? VISIBLE : INVISIBLE, "LKFSubclass")
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
table.put("en_US", this, status);
|
||
|
}
|
||
|
|
||
|
protected:
|
||
|
virtual const Hashtable* getSupportedIDs(UErrorCode &/*status*/) const {
|
||
|
return &table;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
class Integer : public UObject {
|
||
|
public:
|
||
|
const int32_t _val;
|
||
|
|
||
|
Integer(int32_t val) : _val(val) {
|
||
|
}
|
||
|
|
||
|
Integer(const Integer& rhs) : UObject(rhs), _val(rhs._val) {
|
||
|
}
|
||
|
virtual ~Integer() {
|
||
|
}
|
||
|
|
||
|
virtual UBool operator==(const UObject& other) const
|
||
|
{
|
||
|
return other.getDynamicClassID() == getStaticClassID() &&
|
||
|
_val == ((Integer&)other)._val;
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
/**
|
||
|
* UObject boilerplate.
|
||
|
*/
|
||
|
virtual UClassID getDynamicClassID() const {
|
||
|
return getStaticClassID();
|
||
|
}
|
||
|
|
||
|
static UClassID getStaticClassID() {
|
||
|
return (UClassID)&fgClassID;
|
||
|
}
|
||
|
|
||
|
public:
|
||
|
virtual UnicodeString& debug(UnicodeString& result) const {
|
||
|
debugClass(result);
|
||
|
result.append(" val: ");
|
||
|
result.append(_val);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
virtual UnicodeString& debugClass(UnicodeString& result) const {
|
||
|
return result.append("Integer");
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
static const char fgClassID;
|
||
|
};
|
||
|
|
||
|
const char Integer::fgClassID = '\0';
|
||
|
|
||
|
// use locale keys
|
||
|
class TestIntegerService : public ICUService {
|
||
|
public:
|
||
|
ICUServiceKey* createKey(const UnicodeString* id, UErrorCode& status) const {
|
||
|
return LocaleKey::createWithCanonicalFallback(id, NULL, status); // no fallback locale
|
||
|
}
|
||
|
|
||
|
virtual ICUServiceFactory* createSimpleFactory(UObject* obj, const UnicodeString& id, UBool visible, UErrorCode& status)
|
||
|
{
|
||
|
if (U_SUCCESS(status) && obj && obj->getDynamicClassID() == Integer::getStaticClassID()) {
|
||
|
return new SimpleFactory((Integer*)obj, id, visible);
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
virtual UObject* cloneInstance(UObject* instance) const {
|
||
|
return instance ? new Integer(*(Integer*)instance) : NULL;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
ICUServiceTest::ICUServiceTest() {
|
||
|
}
|
||
|
|
||
|
ICUServiceTest::~ICUServiceTest() {
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::runIndexedTest(int32_t index, UBool exec, const char* &name,
|
||
|
char* /*par*/)
|
||
|
{
|
||
|
switch (index) {
|
||
|
TESTCASE(0,testAPI_One);
|
||
|
TESTCASE(1,testAPI_Two);
|
||
|
TESTCASE(2,testRBF);
|
||
|
TESTCASE(3,testNotification);
|
||
|
TESTCASE(4,testLocale);
|
||
|
TESTCASE(5,testWrapFactory);
|
||
|
TESTCASE(6,testCoverage);
|
||
|
default: name = ""; break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
UnicodeString append(UnicodeString& result, const UObject* obj)
|
||
|
{
|
||
|
char buffer[128];
|
||
|
if (obj == NULL) {
|
||
|
result.append("NULL");
|
||
|
} else {
|
||
|
UClassID id = obj->getDynamicClassID();
|
||
|
if (id == UnicodeString::getStaticClassID()) {
|
||
|
result.append(*(UnicodeString*)obj);
|
||
|
} else if (id == Locale::getStaticClassID()) {
|
||
|
result.append(((Locale*)obj)->getName());
|
||
|
} else if (id == Integer::getStaticClassID()) {
|
||
|
sprintf(buffer, "%d", (int)((Integer*)obj)->_val);
|
||
|
result.append(buffer);
|
||
|
} else {
|
||
|
sprintf(buffer, "%p", (const void*)obj);
|
||
|
result.append(buffer);
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
UnicodeString&
|
||
|
ICUServiceTest::lrmsg(UnicodeString& result, const UnicodeString& message, const UObject* lhs, const UObject* rhs) const
|
||
|
{
|
||
|
result.append(message);
|
||
|
result.append(" lhs: ");
|
||
|
append(result, lhs);
|
||
|
result.append(", rhs: ");
|
||
|
append(result, rhs);
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::confirmBoolean(const UnicodeString& message, UBool val)
|
||
|
{
|
||
|
if (val) {
|
||
|
logln(message);
|
||
|
} else {
|
||
|
errln(message);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#if 0
|
||
|
void
|
||
|
ICUServiceTest::confirmEqual(const UnicodeString& message, const UObject* lhs, const UObject* rhs)
|
||
|
{
|
||
|
UBool equ = (lhs == NULL)
|
||
|
? (rhs == NULL)
|
||
|
: (rhs != NULL && lhs->operator==(*rhs));
|
||
|
|
||
|
UnicodeString temp;
|
||
|
lrmsg(temp, message, lhs, rhs);
|
||
|
|
||
|
if (equ) {
|
||
|
logln(temp);
|
||
|
} else {
|
||
|
errln(temp);
|
||
|
}
|
||
|
}
|
||
|
#else
|
||
|
void
|
||
|
ICUServiceTest::confirmEqual(const UnicodeString& message, const Integer* lhs, const Integer* rhs)
|
||
|
{
|
||
|
UBool equ = (lhs == NULL)
|
||
|
? (rhs == NULL)
|
||
|
: (rhs != NULL && lhs->operator==(*rhs));
|
||
|
|
||
|
UnicodeString temp;
|
||
|
lrmsg(temp, message, lhs, rhs);
|
||
|
|
||
|
if (equ) {
|
||
|
logln(temp);
|
||
|
} else {
|
||
|
errln(temp);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::confirmEqual(const UnicodeString& message, const UnicodeString* lhs, const UnicodeString* rhs)
|
||
|
{
|
||
|
UBool equ = (lhs == NULL)
|
||
|
? (rhs == NULL)
|
||
|
: (rhs != NULL && lhs->operator==(*rhs));
|
||
|
|
||
|
UnicodeString temp;
|
||
|
lrmsg(temp, message, lhs, rhs);
|
||
|
|
||
|
if (equ) {
|
||
|
logln(temp);
|
||
|
} else {
|
||
|
errln(temp);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::confirmEqual(const UnicodeString& message, const Locale* lhs, const Locale* rhs)
|
||
|
{
|
||
|
UBool equ = (lhs == NULL)
|
||
|
? (rhs == NULL)
|
||
|
: (rhs != NULL && lhs->operator==(*rhs));
|
||
|
|
||
|
UnicodeString temp;
|
||
|
lrmsg(temp, message, lhs, rhs);
|
||
|
|
||
|
if (equ) {
|
||
|
logln(temp);
|
||
|
} else {
|
||
|
errln(temp);
|
||
|
}
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
// use these for now
|
||
|
void
|
||
|
ICUServiceTest::confirmStringsEqual(const UnicodeString& message, const UnicodeString& lhs, const UnicodeString& rhs)
|
||
|
{
|
||
|
UBool equ = lhs == rhs;
|
||
|
|
||
|
UnicodeString temp = message;
|
||
|
temp.append(" lhs: ");
|
||
|
temp.append(lhs);
|
||
|
temp.append(" rhs: ");
|
||
|
temp.append(rhs);
|
||
|
|
||
|
if (equ) {
|
||
|
logln(temp);
|
||
|
} else {
|
||
|
errln(temp);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::confirmIdentical(const UnicodeString& message, const UObject* lhs, const UObject *rhs)
|
||
|
{
|
||
|
UnicodeString temp;
|
||
|
lrmsg(temp, message, lhs, rhs);
|
||
|
if (lhs == rhs) {
|
||
|
logln(temp);
|
||
|
} else {
|
||
|
errln(temp);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::confirmIdentical(const UnicodeString& message, int32_t lhs, int32_t rhs)
|
||
|
{
|
||
|
if (lhs == rhs) {
|
||
|
logln(message + " lhs: " + lhs + " rhs: " + rhs);
|
||
|
} else {
|
||
|
errln(message + " lhs: " + lhs + " rhs: " + rhs);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::msgstr(const UnicodeString& message, UObject* obj, UBool err)
|
||
|
{
|
||
|
if (obj) {
|
||
|
UnicodeString* str = (UnicodeString*)obj;
|
||
|
logln(message + *str);
|
||
|
delete str;
|
||
|
} else if (err) {
|
||
|
errln("Error " + message + "string is NULL");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::testAPI_One()
|
||
|
{
|
||
|
// create a service using locale keys,
|
||
|
TestIntegerService service;
|
||
|
|
||
|
// register an object with one locale,
|
||
|
// search for an object with a more specific locale
|
||
|
// should return the original object
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* singleton0 = new Integer(0);
|
||
|
service.registerInstance(singleton0, "en_US", status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_FOO", status);
|
||
|
confirmEqual("1) en_US_FOO -> en_US", result, singleton0);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// register a new object with the more specific locale
|
||
|
// search for an object with that locale
|
||
|
// should return the new object
|
||
|
Integer* singleton1 = new Integer(1);
|
||
|
service.registerInstance(singleton1, "en_US_FOO", status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_FOO", status);
|
||
|
confirmEqual("2) en_US_FOO -> en_US_FOO", result, singleton1);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// search for an object that falls back to the first registered locale
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_BAR", status);
|
||
|
confirmEqual("3) en_US_BAR -> en_US", result, singleton0);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// get a list of the factories, should be two
|
||
|
{
|
||
|
confirmIdentical("4) factory size", service.countFactories(), 2);
|
||
|
}
|
||
|
|
||
|
// register a new object with yet another locale
|
||
|
Integer* singleton2 = new Integer(2);
|
||
|
service.registerInstance(singleton2, "en", status);
|
||
|
{
|
||
|
confirmIdentical("5) factory size", service.countFactories(), 3);
|
||
|
}
|
||
|
|
||
|
// search for an object with the new locale
|
||
|
// stack of factories is now en, en_US_FOO, en_US
|
||
|
// search for en_US should still find en_US object
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_BAR", status);
|
||
|
confirmEqual("6) en_US_BAR -> en_US", result, singleton0);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// register a new object with an old id, should hide earlier factory using this id, but leave it there
|
||
|
Integer* singleton3 = new Integer(3);
|
||
|
URegistryKey s3key = service.registerInstance(singleton3, "en_US", status);
|
||
|
{
|
||
|
confirmIdentical("9) factory size", service.countFactories(), 4);
|
||
|
}
|
||
|
|
||
|
// should get data from that new factory
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_BAR", status);
|
||
|
confirmEqual("10) en_US_BAR -> (3)", result, singleton3);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// remove new factory
|
||
|
// should have fewer factories again
|
||
|
// singleton3 dead!
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
service.unregister(s3key, status);
|
||
|
confirmIdentical("11) factory size", service.countFactories(), 3);
|
||
|
}
|
||
|
|
||
|
// should get original data again after remove factory
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_BAR", status);
|
||
|
confirmEqual("12) en_US_BAR -> (3)", result, singleton0);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// shouldn't find unregistered ids
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("foo", status);
|
||
|
confirmIdentical("13) foo -> null", result, NULL);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// should find non-canonical strings
|
||
|
{
|
||
|
UnicodeString resultID;
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("EN_us_fOo", &resultID, status);
|
||
|
confirmEqual("14a) find-non-canonical", result, singleton1);
|
||
|
confirmStringsEqual("14b) find non-canonical", resultID, "en_US_FOO");
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// should be able to register non-canonical strings and get them canonicalized
|
||
|
Integer* singleton4 = new Integer(4);
|
||
|
service.registerInstance(singleton4, "eN_ca_dUde", status);
|
||
|
{
|
||
|
UnicodeString resultID;
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("En_Ca_DuDe", &resultID, status);
|
||
|
confirmEqual("15a) find-non-canonical", result, singleton4);
|
||
|
confirmStringsEqual("15b) register non-canonical", resultID, "en_CA_DUDE");
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// should be able to register invisible factories, these will not
|
||
|
// be visible by default, but if you know the secret password you
|
||
|
// can still access these services...
|
||
|
Integer* singleton5 = new Integer(5);
|
||
|
service.registerInstance(singleton5, "en_US_BAR", FALSE, status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Integer* result = (Integer*)service.get("en_US_BAR", status);
|
||
|
confirmEqual("17) get invisible", result, singleton5);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// should not be able to locate invisible services
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, status);
|
||
|
service.getVisibleIDs(ids, status);
|
||
|
UnicodeString target = "en_US_BAR";
|
||
|
confirmBoolean("18) find invisible", !ids.contains(&target));
|
||
|
}
|
||
|
|
||
|
// clear factory and caches
|
||
|
service.reset();
|
||
|
confirmBoolean("19) is default", service.isDefault());
|
||
|
}
|
||
|
|
||
|
/*
|
||
|
******************************************************************
|
||
|
*/
|
||
|
|
||
|
class TestStringService : public ICUService {
|
||
|
public:
|
||
|
ICUServiceKey* createKey(const UnicodeString* id, UErrorCode& status) const {
|
||
|
return LocaleKey::createWithCanonicalFallback(id, NULL, status); // no fallback locale
|
||
|
}
|
||
|
|
||
|
virtual ICUServiceFactory* createSimpleFactory(UObject* obj, const UnicodeString& id, UBool visible, UErrorCode& /* status */)
|
||
|
{
|
||
|
if (obj && obj->getDynamicClassID() == UnicodeString::getStaticClassID()) {
|
||
|
return new SimpleFactory((UnicodeString*)obj, id, visible);
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
virtual UObject* cloneInstance(UObject* instance) const {
|
||
|
return instance ? new UnicodeString(*(UnicodeString*)instance) : NULL;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// this creates a string for any id, but doesn't report anything
|
||
|
class AnonymousStringFactory : public ICUServiceFactory
|
||
|
{
|
||
|
public:
|
||
|
virtual UObject* create(const ICUServiceKey& key, const ICUService* /* service */, UErrorCode& /* status */) const {
|
||
|
return new UnicodeString(key.getID());
|
||
|
}
|
||
|
|
||
|
virtual void updateVisibleIDs(Hashtable& /*result*/, UErrorCode& /*status*/) const {
|
||
|
// do nothing
|
||
|
}
|
||
|
|
||
|
virtual UnicodeString& getDisplayName(const UnicodeString& /*id*/, const Locale& /*locale*/, UnicodeString& result) const {
|
||
|
// do nothing
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
virtual UClassID getDynamicClassID() const {
|
||
|
return getStaticClassID();
|
||
|
}
|
||
|
|
||
|
static UClassID getStaticClassID() {
|
||
|
return (UClassID)&fgClassID;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
static const char fgClassID;
|
||
|
};
|
||
|
|
||
|
const char AnonymousStringFactory::fgClassID = '\0';
|
||
|
|
||
|
class TestMultipleKeyStringFactory : public ICUServiceFactory {
|
||
|
UErrorCode _status;
|
||
|
UVector _ids;
|
||
|
UnicodeString _factoryID;
|
||
|
|
||
|
public:
|
||
|
TestMultipleKeyStringFactory(const UnicodeString ids[], int32_t count, const UnicodeString& factoryID)
|
||
|
: _status(U_ZERO_ERROR)
|
||
|
, _ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, count, _status)
|
||
|
, _factoryID(factoryID + ": ")
|
||
|
{
|
||
|
for (int i = 0; i < count; ++i) {
|
||
|
_ids.addElement(new UnicodeString(ids[i]), _status);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
~TestMultipleKeyStringFactory() {
|
||
|
}
|
||
|
|
||
|
UObject* create(const ICUServiceKey& key, const ICUService* /* service */, UErrorCode& status) const {
|
||
|
if (U_FAILURE(status)) {
|
||
|
return NULL;
|
||
|
}
|
||
|
UnicodeString temp;
|
||
|
key.currentID(temp);
|
||
|
if (U_SUCCESS(_status)) {
|
||
|
if (_ids.contains(&temp)) {
|
||
|
return new UnicodeString(_factoryID + temp);
|
||
|
}
|
||
|
} else {
|
||
|
status = _status;
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
void updateVisibleIDs(Hashtable& result, UErrorCode& status) const {
|
||
|
if (U_SUCCESS(_status)) {
|
||
|
for (int32_t i = 0; i < _ids.size(); ++i) {
|
||
|
result.put(*(UnicodeString*)_ids[i], (void*)this, status);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
UnicodeString& getDisplayName(const UnicodeString& id, const Locale& locale, UnicodeString& result) const {
|
||
|
if (U_SUCCESS(_status) && _ids.contains((void*)&id)) {
|
||
|
char buffer[128];
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
int32_t len = id.extract(buffer, sizeof(buffer), NULL, status);
|
||
|
if (U_SUCCESS(status)) {
|
||
|
if (len == sizeof(buffer)) {
|
||
|
--len;
|
||
|
}
|
||
|
buffer[len] = 0;
|
||
|
Locale loc = Locale::createFromName(buffer);
|
||
|
loc.getDisplayName(locale, result);
|
||
|
return result;
|
||
|
}
|
||
|
}
|
||
|
result.setToBogus(); // shouldn't happen
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
virtual UClassID getDynamicClassID() const {
|
||
|
return getStaticClassID();
|
||
|
}
|
||
|
|
||
|
static UClassID getStaticClassID() {
|
||
|
return (UClassID)&fgClassID;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
static const char fgClassID;
|
||
|
};
|
||
|
|
||
|
const char TestMultipleKeyStringFactory::fgClassID = '\0';
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::testAPI_Two()
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
TestStringService service;
|
||
|
service.registerFactory(new AnonymousStringFactory(), status);
|
||
|
|
||
|
// anonymous factory will still handle the id
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
const UnicodeString en_US = "en_US";
|
||
|
UnicodeString* result = (UnicodeString*)service.get(en_US, status);
|
||
|
confirmEqual("21) locale", result, &en_US);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// still normalizes id
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
const UnicodeString en_US_BAR = "en_US_BAR";
|
||
|
UnicodeString resultID;
|
||
|
UnicodeString* result = (UnicodeString*)service.get("EN_us_bar", &resultID, status);
|
||
|
confirmEqual("22) locale", &resultID, &en_US_BAR);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// we can override for particular ids
|
||
|
UnicodeString* singleton0 = new UnicodeString("Zero");
|
||
|
service.registerInstance(singleton0, "en_US_BAR", status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* result = (UnicodeString*)service.get("en_US_BAR", status);
|
||
|
confirmEqual("23) override super", result, singleton0);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
// empty service should not recognize anything
|
||
|
service.reset();
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* result = (UnicodeString*)service.get("en_US", status);
|
||
|
confirmIdentical("24) empty", result, NULL);
|
||
|
}
|
||
|
|
||
|
// create a custom multiple key factory
|
||
|
{
|
||
|
UnicodeString xids[] = {
|
||
|
"en_US_VALLEY_GIRL",
|
||
|
"en_US_VALLEY_BOY",
|
||
|
"en_US_SURFER_GAL",
|
||
|
"en_US_SURFER_DUDE"
|
||
|
};
|
||
|
int32_t count = sizeof(xids)/sizeof(UnicodeString);
|
||
|
|
||
|
ICUServiceFactory* f = new TestMultipleKeyStringFactory(xids, count, "Later");
|
||
|
service.registerFactory(f, status);
|
||
|
}
|
||
|
|
||
|
// iterate over the visual ids returned by the multiple factory
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, 0, status);
|
||
|
service.getVisibleIDs(ids, status);
|
||
|
for (int i = 0; i < ids.size(); ++i) {
|
||
|
const UnicodeString* id = (const UnicodeString*)ids[i];
|
||
|
UnicodeString* result = (UnicodeString*)service.get(*id, status);
|
||
|
if (result) {
|
||
|
logln(" " + *id + " --> " + *result);
|
||
|
delete result;
|
||
|
} else {
|
||
|
errln("could not find " + *id);
|
||
|
}
|
||
|
}
|
||
|
// four visible ids
|
||
|
confirmIdentical("25) visible ids", ids.size(), 4);
|
||
|
}
|
||
|
|
||
|
// iterate over the display names
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector names(userv_deleteStringPair, NULL, status);
|
||
|
service.getDisplayNames(names, Locale::getGerman(), status);
|
||
|
for (int i = 0; i < names.size(); ++i) {
|
||
|
const StringPair* pair = (const StringPair*)names[i];
|
||
|
logln(" " + pair->displayName + " --> " + pair->id);
|
||
|
}
|
||
|
confirmIdentical("26) display names", names.size(), 4);
|
||
|
}
|
||
|
|
||
|
// no valid display name
|
||
|
{
|
||
|
UnicodeString name;
|
||
|
service.getDisplayName("en_US_VALLEY_GEEK", name);
|
||
|
confirmBoolean("27) get display name", name.isBogus());
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UnicodeString name;
|
||
|
service.getDisplayName("en_US_SURFER_DUDE", name, Locale::getEnglish());
|
||
|
confirmStringsEqual("28) get display name", name, "English (United States, SURFER_DUDE)");
|
||
|
}
|
||
|
|
||
|
// register another multiple factory
|
||
|
{
|
||
|
UnicodeString xids[] = {
|
||
|
"en_US_SURFER",
|
||
|
"en_US_SURFER_GAL",
|
||
|
"en_US_SILICON",
|
||
|
"en_US_SILICON_GEEK",
|
||
|
};
|
||
|
int32_t count = sizeof(xids)/sizeof(UnicodeString);
|
||
|
|
||
|
ICUServiceFactory* f = new TestMultipleKeyStringFactory(xids, count, "Rad dude");
|
||
|
service.registerFactory(f, status);
|
||
|
}
|
||
|
|
||
|
// this time, we have seven display names
|
||
|
// Rad dude's surfer gal 'replaces' Later's surfer gal
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector names(userv_deleteStringPair, NULL, status);
|
||
|
service.getDisplayNames(names, Locale("es"), status);
|
||
|
for (int i = 0; i < names.size(); ++i) {
|
||
|
const StringPair* pair = (const StringPair*)names[i];
|
||
|
logln(" " + pair->displayName + " --> " + pair->id);
|
||
|
}
|
||
|
confirmIdentical("26) display names", names.size(), 7);
|
||
|
}
|
||
|
|
||
|
// we should get the display name corresponding to the actual id
|
||
|
// returned by the id we used.
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString actualID;
|
||
|
UnicodeString id = "en_us_surfer_gal";
|
||
|
UnicodeString* gal = (UnicodeString*)service.get(id, &actualID, status);
|
||
|
if (gal != NULL) {
|
||
|
UnicodeString displayName;
|
||
|
logln("actual id: " + actualID);
|
||
|
service.getDisplayName(actualID, displayName, Locale::getEnglish());
|
||
|
logln("found actual: " + *gal + " with display name: " + displayName);
|
||
|
confirmBoolean("30) found display name for actual", !displayName.isBogus());
|
||
|
|
||
|
service.getDisplayName(id, displayName, Locale::getEnglish());
|
||
|
logln("found actual: " + *gal + " with display name: " + displayName);
|
||
|
confirmBoolean("31) found display name for query", displayName.isBogus());
|
||
|
|
||
|
delete gal;
|
||
|
} else {
|
||
|
errln("30) service could not find entry for " + id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// this should be handled by the 'dude' factory, since it overrides en_US_SURFER.
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString actualID;
|
||
|
UnicodeString id = "en_US_SURFER_BOZO";
|
||
|
UnicodeString* bozo = (UnicodeString*)service.get(id, &actualID, status);
|
||
|
if (bozo != NULL) {
|
||
|
UnicodeString displayName;
|
||
|
service.getDisplayName(actualID, displayName, Locale::getEnglish());
|
||
|
logln("found actual: " + *bozo + " with display name: " + displayName);
|
||
|
confirmBoolean("32) found display name for actual", !displayName.isBogus());
|
||
|
|
||
|
service.getDisplayName(id, displayName, Locale::getEnglish());
|
||
|
logln("found actual: " + *bozo + " with display name: " + displayName);
|
||
|
confirmBoolean("33) found display name for query", displayName.isBogus());
|
||
|
|
||
|
delete bozo;
|
||
|
} else {
|
||
|
errln("32) service could not find entry for " + id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// certainly not default...
|
||
|
{
|
||
|
confirmBoolean("34) is default ", !service.isDefault());
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, 0, status);
|
||
|
service.getVisibleIDs(ids, status);
|
||
|
for (int i = 0; i < ids.size(); ++i) {
|
||
|
const UnicodeString* id = (const UnicodeString*)ids[i];
|
||
|
msgstr(*id + "? ", service.get(*id, status));
|
||
|
}
|
||
|
|
||
|
logstr("valleygirl? ", service.get("en_US_VALLEY_GIRL", status));
|
||
|
logstr("valleyboy? ", service.get("en_US_VALLEY_BOY", status));
|
||
|
logstr("valleydude? ", service.get("en_US_VALLEY_DUDE", status));
|
||
|
logstr("surfergirl? ", service.get("en_US_SURFER_GIRL", status));
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
class CalifornioLanguageFactory : public ICUResourceBundleFactory
|
||
|
{
|
||
|
public:
|
||
|
static const char* californio; // = "en_US_CA";
|
||
|
static const char* valley; // = californio ## "_VALLEY";
|
||
|
static const char* surfer; // = californio ## "_SURFER";
|
||
|
static const char* geek; // = californio ## "_GEEK";
|
||
|
static Hashtable* supportedIDs; // = NULL;
|
||
|
|
||
|
static void cleanup(void) {
|
||
|
delete supportedIDs;
|
||
|
supportedIDs = NULL;
|
||
|
}
|
||
|
|
||
|
const Hashtable* getSupportedIDs(UErrorCode& status) const
|
||
|
{
|
||
|
if (supportedIDs == NULL) {
|
||
|
Hashtable* table = new Hashtable();
|
||
|
table->put(UnicodeString(californio), (void*)table, status);
|
||
|
table->put(UnicodeString(valley), (void*)table, status);
|
||
|
table->put(UnicodeString(surfer), (void*)table, status);
|
||
|
table->put(UnicodeString(geek), (void*)table, status);
|
||
|
|
||
|
// not necessarily atomic, but this is a test...
|
||
|
supportedIDs = table;
|
||
|
}
|
||
|
return supportedIDs;
|
||
|
}
|
||
|
|
||
|
UnicodeString& getDisplayName(const UnicodeString& id, const Locale& locale, UnicodeString& result) const
|
||
|
{
|
||
|
UnicodeString prefix = "";
|
||
|
UnicodeString suffix = "";
|
||
|
UnicodeString ls = locale.getName();
|
||
|
if (LocaleUtility::isFallbackOf(californio, ls)) {
|
||
|
if (!ls.caseCompare(valley, 0)) {
|
||
|
prefix = "Like, you know, it's so totally ";
|
||
|
} else if (!ls.caseCompare(surfer, 0)) {
|
||
|
prefix = "Dude, it's ";
|
||
|
} else if (!ls.caseCompare(geek, 0)) {
|
||
|
prefix = "I'd estimate it is approximately ";
|
||
|
} else {
|
||
|
prefix = "Huh? Maybe ";
|
||
|
}
|
||
|
}
|
||
|
if (LocaleUtility::isFallbackOf(californio, id)) {
|
||
|
if (!id.caseCompare(valley, 0)) {
|
||
|
suffix = "like the Valley, you know? Let's go to the mall!";
|
||
|
} else if (!id.caseCompare(surfer, 0)) {
|
||
|
suffix = "time to hit those gnarly waves, Dude!!!";
|
||
|
} else if (!id.caseCompare(geek, 0)) {
|
||
|
suffix = "all systems go. T-Minus 9, 8, 7...";
|
||
|
} else {
|
||
|
suffix = "No Habla Englais";
|
||
|
}
|
||
|
} else {
|
||
|
suffix = ICUResourceBundleFactory::getDisplayName(id, locale, result);
|
||
|
}
|
||
|
|
||
|
result = prefix + suffix;
|
||
|
return result;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
const char* CalifornioLanguageFactory::californio = "en_US_CA";
|
||
|
const char* CalifornioLanguageFactory::valley = "en_US_CA_VALLEY";
|
||
|
const char* CalifornioLanguageFactory::surfer = "en_US_CA_SURFER";
|
||
|
const char* CalifornioLanguageFactory::geek = "en_US_CA_GEEK";
|
||
|
Hashtable* CalifornioLanguageFactory::supportedIDs = NULL;
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::testRBF()
|
||
|
{
|
||
|
// resource bundle factory.
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
TestStringService service;
|
||
|
service.registerFactory(new ICUResourceBundleFactory(), status);
|
||
|
|
||
|
// list all of the resources
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, 0, status);
|
||
|
service.getVisibleIDs(ids, status);
|
||
|
logln("all visible ids:");
|
||
|
for (int i = 0; i < ids.size(); ++i) {
|
||
|
const UnicodeString* id = (const UnicodeString*)ids[i];
|
||
|
logln(*id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// get all the display names of these resources
|
||
|
// this should be fast since the display names were cached.
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector names(userv_deleteStringPair, NULL, status);
|
||
|
service.getDisplayNames(names, Locale::getGermany(), status);
|
||
|
logln("service display names for de_DE");
|
||
|
for (int i = 0; i < names.size(); ++i) {
|
||
|
const StringPair* pair = (const StringPair*)names[i];
|
||
|
logln(" " + pair->displayName + " --> " + pair->id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
service.registerFactory(new CalifornioLanguageFactory(), status);
|
||
|
|
||
|
// get all the display names of these resources
|
||
|
{
|
||
|
logln("californio language factory:");
|
||
|
const char* idNames[] = {
|
||
|
CalifornioLanguageFactory::californio,
|
||
|
CalifornioLanguageFactory::valley,
|
||
|
CalifornioLanguageFactory::surfer,
|
||
|
CalifornioLanguageFactory::geek,
|
||
|
};
|
||
|
int32_t count = sizeof(idNames)/sizeof(idNames[0]);
|
||
|
|
||
|
for (int i = 0; i < count; ++i) {
|
||
|
logln(UnicodeString("\n --- ") + idNames[i] + " ---");
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector names(userv_deleteStringPair, NULL, status);
|
||
|
service.getDisplayNames(names, idNames[i], status);
|
||
|
for (int i = 0; i < names.size(); ++i) {
|
||
|
const StringPair* pair = (const StringPair*)names[i];
|
||
|
logln(" " + pair->displayName + " --> " + pair->id);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
CalifornioLanguageFactory::cleanup();
|
||
|
}
|
||
|
|
||
|
class SimpleListener : public ServiceListener {
|
||
|
ICUServiceTest* _test;
|
||
|
int32_t _n;
|
||
|
UnicodeString _name;
|
||
|
|
||
|
public:
|
||
|
SimpleListener(ICUServiceTest* test, const UnicodeString& name) : _test(test), _n(0), _name(name) {}
|
||
|
|
||
|
virtual void serviceChanged(const ICUService& service) const {
|
||
|
UnicodeString serviceName = "listener ";
|
||
|
serviceName.append(_name);
|
||
|
serviceName.append(" n++");
|
||
|
serviceName.append(" service changed: " );
|
||
|
service.getName(serviceName);
|
||
|
_test->logln(serviceName);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::testNotification()
|
||
|
{
|
||
|
SimpleListener one(this, "one");
|
||
|
SimpleListener two(this, "two");
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
|
||
|
logln("simple registration notification");
|
||
|
TestStringService ls;
|
||
|
ls.addListener(&one, status);
|
||
|
ls.addListener(&two, status);
|
||
|
|
||
|
logln("registering foo... ");
|
||
|
ls.registerInstance(new UnicodeString("Foo"), "en_FOO", status);
|
||
|
logln("registering bar... ");
|
||
|
ls.registerInstance(new UnicodeString("Bar"), "en_BAR", status);
|
||
|
logln("getting foo...");
|
||
|
UnicodeString* result = (UnicodeString*)ls.get("en_FOO", status);
|
||
|
logln(*result);
|
||
|
delete result;
|
||
|
|
||
|
logln("removing listener 2...");
|
||
|
ls.removeListener(&two, status);
|
||
|
logln("registering baz...");
|
||
|
ls.registerInstance(new UnicodeString("Baz"), "en_BAZ", status);
|
||
|
logln("removing listener 1");
|
||
|
ls.removeListener(&one, status);
|
||
|
logln("registering burp...");
|
||
|
ls.registerInstance(new UnicodeString("Burp"), "en_BURP", status);
|
||
|
|
||
|
// should only get one notification even if register multiple times
|
||
|
logln("... trying multiple registration");
|
||
|
ls.addListener(&one, status);
|
||
|
ls.addListener(&one, status);
|
||
|
ls.addListener(&one, status);
|
||
|
ls.addListener(&two, status);
|
||
|
ls.registerInstance(new UnicodeString("Foo"), "en_FOO", status);
|
||
|
logln("... registered foo");
|
||
|
}
|
||
|
#if 0
|
||
|
// same thread, so we can't callback within notification, unlike Java
|
||
|
ServiceListener l3 = new ServiceListener() {
|
||
|
private int n;
|
||
|
public void serviceChanged(ICUService s) {
|
||
|
logln("listener 3 report " + n++ + " service changed...");
|
||
|
if (s.get("en_BOINK") == null) { // don't recurse on ourselves!!!
|
||
|
logln("registering boink...");
|
||
|
s.registerInstance("boink", "en_BOINK");
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
ls.addListener(l3);
|
||
|
logln("registering boo...");
|
||
|
ls.registerInstance("Boo", "en_BOO");
|
||
|
#endif
|
||
|
|
||
|
logln("...done");
|
||
|
}
|
||
|
|
||
|
class TestStringLocaleService : public ICULocaleService {
|
||
|
public:
|
||
|
virtual UObject* cloneInstance(UObject* instance) const {
|
||
|
return instance ? new UnicodeString(*(UnicodeString*)instance) : NULL;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
void ICUServiceTest::testLocale() {
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
TestStringLocaleService service;
|
||
|
|
||
|
UnicodeString* root = new UnicodeString("root");
|
||
|
UnicodeString* german = new UnicodeString("german");
|
||
|
UnicodeString* germany = new UnicodeString("german_Germany");
|
||
|
UnicodeString* japanese = new UnicodeString("japanese");
|
||
|
UnicodeString* japan = new UnicodeString("japanese_Japan");
|
||
|
|
||
|
service.registerInstance(root, "", status);
|
||
|
service.registerInstance(german, "de", status);
|
||
|
service.registerInstance(germany, Locale::getGermany(), status);
|
||
|
service.registerInstance(japanese, "ja", status);
|
||
|
service.registerInstance(japan, Locale::getJapan(), status);
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", status);
|
||
|
confirmEqual("test de_US", german, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", LocaleKey::KIND_ANY, status);
|
||
|
confirmEqual("test de_US 2", german, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", 1234, status);
|
||
|
confirmEqual("test de_US 3", german, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Locale actualReturn;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", &actualReturn, status);
|
||
|
confirmEqual("test de_US 5", german, target);
|
||
|
confirmEqual("test de_US 6", &actualReturn, &Locale::getGerman());
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Locale actualReturn;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", LocaleKey::KIND_ANY, &actualReturn, status);
|
||
|
confirmEqual("test de_US 7", &actualReturn, &Locale::getGerman());
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
Locale actualReturn;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", 1234, &actualReturn, status);
|
||
|
confirmEqual("test de_US 8", german, target);
|
||
|
confirmEqual("test de_US 9", &actualReturn, &Locale::getGerman());
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
UnicodeString* one = new UnicodeString("one/de_US");
|
||
|
UnicodeString* two = new UnicodeString("two/de_US");
|
||
|
|
||
|
service.registerInstance(one, Locale("de_US"), 1, status);
|
||
|
service.registerInstance(two, Locale("de_US"), 2, status);
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", 1, status);
|
||
|
confirmEqual("test de_US kind 1", one, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", 2, status);
|
||
|
confirmEqual("test de_US kind 2", two, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("de_US", status);
|
||
|
confirmEqual("test de_US kind 3", german, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString english = "en";
|
||
|
Locale localeResult;
|
||
|
UnicodeString result;
|
||
|
LocaleKey* lkey = LocaleKey::createWithCanonicalFallback(&english, NULL, 1234, status);
|
||
|
logln("lkey prefix: " + lkey->prefix(result));
|
||
|
result.remove();
|
||
|
logln("lkey descriptor: " + lkey->currentDescriptor(result));
|
||
|
result.remove();
|
||
|
logln(UnicodeString("lkey current locale: ") + lkey->currentLocale(localeResult).getName());
|
||
|
result.remove();
|
||
|
|
||
|
lkey->fallback();
|
||
|
logln("lkey descriptor 2: " + lkey->currentDescriptor(result));
|
||
|
result.remove();
|
||
|
|
||
|
lkey->fallback();
|
||
|
logln("lkey descriptor 3: " + lkey->currentDescriptor(result));
|
||
|
result.remove();
|
||
|
delete lkey; // tentatively weiv
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("za_PPP", status);
|
||
|
confirmEqual("test zappp", root, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
Locale loc = Locale::getDefault();
|
||
|
Locale::setDefault(Locale::getJapanese(), status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("za_PPP", status);
|
||
|
confirmEqual("test with ja locale", japanese, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, 0, status);
|
||
|
service.getVisibleIDs(ids, status);
|
||
|
logln("all visible ids:");
|
||
|
for (int i = 0; i < ids.size(); ++i) {
|
||
|
const UnicodeString* id = (const UnicodeString*)ids[i];
|
||
|
logln(*id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
Locale::setDefault(loc, status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UVector ids(uhash_deleteUnicodeString, uhash_compareUnicodeString, 0, status);
|
||
|
service.getVisibleIDs(ids, status);
|
||
|
logln("all visible ids:");
|
||
|
for (int i = 0; i < ids.size(); ++i) {
|
||
|
const UnicodeString* id = (const UnicodeString*)ids[i];
|
||
|
logln(*id);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* target = (UnicodeString*)service.get("za_PPP", status);
|
||
|
confirmEqual("test with en locale", root, target);
|
||
|
delete target;
|
||
|
}
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
StringEnumeration* locales = service.getAvailableLocales();
|
||
|
if (locales) {
|
||
|
confirmIdentical("test available locales", locales->count(status), 5);
|
||
|
logln("locales: ");
|
||
|
{
|
||
|
const char* p;
|
||
|
while ((p = locales->next(NULL, status))) {
|
||
|
logln(p);
|
||
|
}
|
||
|
}
|
||
|
logln(" ");
|
||
|
delete locales;
|
||
|
} else {
|
||
|
errln("could not create available locales");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
class WrapFactory : public ICUServiceFactory {
|
||
|
public:
|
||
|
static const UnicodeString& getGreetingID() {
|
||
|
if (greetingID == NULL) {
|
||
|
greetingID = new UnicodeString("greeting");
|
||
|
}
|
||
|
return *greetingID;
|
||
|
}
|
||
|
|
||
|
static void cleanup() {
|
||
|
delete greetingID;
|
||
|
greetingID = NULL;
|
||
|
}
|
||
|
|
||
|
UObject* create(const ICUServiceKey& key, const ICUService* service, UErrorCode& status) const {
|
||
|
if (U_SUCCESS(status)) {
|
||
|
UnicodeString temp;
|
||
|
if (key.currentID(temp).compare(getGreetingID()) == 0) {
|
||
|
UnicodeString* previous = (UnicodeString*)service->getKey((ICUServiceKey&)key, NULL, this, status);
|
||
|
if (previous) {
|
||
|
previous->insert(0, "A different greeting: \"");
|
||
|
previous->append("\"");
|
||
|
return previous;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
void updateVisibleIDs(Hashtable& result, UErrorCode& status) const {
|
||
|
if (U_SUCCESS(status)) {
|
||
|
result.put("greeting", (void*)this, status);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
UnicodeString& getDisplayName(const UnicodeString& id, const Locale& /* locale */, UnicodeString& result) const {
|
||
|
result.append("wrap '");
|
||
|
result.append(id);
|
||
|
result.append("'");
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* UObject boilerplate.
|
||
|
*/
|
||
|
virtual UClassID getDynamicClassID() const {
|
||
|
return getStaticClassID();
|
||
|
}
|
||
|
|
||
|
static UClassID getStaticClassID() {
|
||
|
return (UClassID)&fgClassID;
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
static const char fgClassID;
|
||
|
static UnicodeString* greetingID;
|
||
|
};
|
||
|
|
||
|
UnicodeString* WrapFactory::greetingID = NULL;
|
||
|
const char WrapFactory::fgClassID = '\0';
|
||
|
|
||
|
void
|
||
|
ICUServiceTest::testWrapFactory()
|
||
|
{
|
||
|
UnicodeString* greeting = new UnicodeString("Hello There");
|
||
|
UnicodeString greetingID = "greeting";
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
TestStringService service;
|
||
|
service.registerInstance(greeting, greetingID, status);
|
||
|
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* result = (UnicodeString*)service.get(greetingID, status);
|
||
|
if (result) {
|
||
|
logln("test one: " + *result);
|
||
|
delete result;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
service.registerFactory(new WrapFactory(), status);
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
UnicodeString* result = (UnicodeString*)service.get(greetingID, status);
|
||
|
UnicodeString target = "A different greeting: \"Hello There\"";
|
||
|
confirmEqual("wrap test: ", result, &target);
|
||
|
delete result;
|
||
|
}
|
||
|
|
||
|
WrapFactory::cleanup();
|
||
|
}
|
||
|
|
||
|
// misc coverage tests
|
||
|
void ICUServiceTest::testCoverage()
|
||
|
{
|
||
|
// ICUServiceKey
|
||
|
{
|
||
|
UnicodeString temp;
|
||
|
ICUServiceKey key("foobar");
|
||
|
logln("ID: " + key.getID());
|
||
|
logln("canonicalID: " + key.canonicalID(temp));
|
||
|
logln("currentID: " + key.currentID(temp.remove()));
|
||
|
logln("has fallback: " + UnicodeString(key.fallback() ? "true" : "false"));
|
||
|
|
||
|
if (key.getDynamicClassID() != ICUServiceKey::getStaticClassID()) {
|
||
|
errln("service key rtt failed.");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// SimpleFactory
|
||
|
{
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
|
||
|
UnicodeString* obj = new UnicodeString("An Object");
|
||
|
SimpleFactory* sf = new SimpleFactory(obj, "object");
|
||
|
|
||
|
UnicodeString temp;
|
||
|
logln(sf->getDisplayName("object", Locale::getDefault(), temp));
|
||
|
|
||
|
if (sf->getDynamicClassID() != SimpleFactory::getStaticClassID()) {
|
||
|
errln("simple factory rtti failed.");
|
||
|
}
|
||
|
|
||
|
// ICUService
|
||
|
TestStringService service;
|
||
|
service.registerFactory(sf, status);
|
||
|
|
||
|
{
|
||
|
UnicodeString* result = (UnicodeString*)service.get("object", status);
|
||
|
if (result) {
|
||
|
logln("object is: " + *result);
|
||
|
delete result;
|
||
|
} else {
|
||
|
errln("could not get object");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// ICULocaleService
|
||
|
|
||
|
// LocaleKey
|
||
|
{
|
||
|
UnicodeString primary("en_US");
|
||
|
UnicodeString fallback("ja_JP");
|
||
|
UErrorCode status = U_ZERO_ERROR;
|
||
|
LocaleKey* key = LocaleKey::createWithCanonicalFallback(&primary, &fallback, status);
|
||
|
|
||
|
if (key->getDynamicClassID() != LocaleKey::getStaticClassID()) {
|
||
|
errln("localekey rtti error");
|
||
|
}
|
||
|
|
||
|
if (!key->isFallbackOf("en_US_FOOBAR")) {
|
||
|
errln("localekey should be fallback for en_US_FOOBAR");
|
||
|
}
|
||
|
if (!key->isFallbackOf("en_US")) {
|
||
|
errln("localekey should be fallback for en_US");
|
||
|
}
|
||
|
if (key->isFallbackOf("en")) {
|
||
|
errln("localekey should not be fallback for en");
|
||
|
}
|
||
|
|
||
|
do {
|
||
|
Locale loc;
|
||
|
logln(UnicodeString("current locale: ") + key->currentLocale(loc).getName());
|
||
|
logln(UnicodeString("canonical locale: ") + key->canonicalLocale(loc).getName());
|
||
|
logln(UnicodeString("is fallback of en: ") + (key->isFallbackOf("en") ? "true" : " false"));
|
||
|
} while (key->fallback());
|
||
|
delete key;
|
||
|
|
||
|
// LocaleKeyFactory
|
||
|
key = LocaleKey::createWithCanonicalFallback(&primary, &fallback, status);
|
||
|
|
||
|
UnicodeString result;
|
||
|
LKFSubclass lkf(TRUE); // empty
|
||
|
Hashtable table;
|
||
|
|
||
|
UObject *obj = lkf.create(*key, NULL, status);
|
||
|
logln("obj: " + UnicodeString(obj ? "obj" : "null"));
|
||
|
logln(lkf.getDisplayName("en_US", Locale::getDefault(), result));
|
||
|
lkf.updateVisibleIDs(table, status);
|
||
|
delete obj;
|
||
|
if (table.count() != 1) {
|
||
|
errln("visible IDs does not contain en_US");
|
||
|
}
|
||
|
|
||
|
LKFSubclass invisibleLKF(FALSE);
|
||
|
obj = lkf.create(*key, NULL, status);
|
||
|
logln("obj: " + UnicodeString(obj ? "obj" : "null"));
|
||
|
logln(invisibleLKF.getDisplayName("en_US", Locale::getDefault(), result.remove()));
|
||
|
invisibleLKF.updateVisibleIDs(table, status);
|
||
|
if (table.count() != 0) {
|
||
|
errln("visible IDs contains en_US");
|
||
|
}
|
||
|
delete obj;
|
||
|
delete key;
|
||
|
}
|
||
|
|
||
|
|
||
|
#if 0
|
||
|
// ResourceBundleFactory
|
||
|
ICUResourceBundleFactory rbf = new ICUResourceBundleFactory();
|
||
|
logln("RB: " + rbf.create(lkey, null));
|
||
|
|
||
|
// ICUNotifier
|
||
|
ICUNotifier nf = new ICUNSubclass();
|
||
|
try {
|
||
|
nf.addListener(null);
|
||
|
errln("added null listener");
|
||
|
}
|
||
|
catch (NullPointerException e) {
|
||
|
logln(e.getMessage());
|
||
|
}
|
||
|
catch (Exception e) {
|
||
|
errln("got wrong exception");
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
nf.addListener(new WrongListener());
|
||
|
errln("added wrong listener");
|
||
|
}
|
||
|
catch (InternalError e) {
|
||
|
logln(e.getMessage());
|
||
|
}
|
||
|
catch (Exception e) {
|
||
|
errln("got wrong exception");
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
nf.removeListener(null);
|
||
|
errln("removed null listener");
|
||
|
}
|
||
|
catch (NullPointerException e) {
|
||
|
logln(e.getMessage());
|
||
|
}
|
||
|
catch (Exception e) {
|
||
|
errln("got wrong exception");
|
||
|
}
|
||
|
|
||
|
nf.removeListener(new MyListener());
|
||
|
nf.notifyChanged();
|
||
|
nf.addListener(new MyListener());
|
||
|
nf.removeListener(new MyListener());
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
|
||
|
/* !UCONFIG_NO_SERVICE */
|
||
|
#endif
|
||
|
|
||
|
|