ICU-20465 Calendar/DateFormat listen to tz extension

See #1176
This commit is contained in:
Frank Tang 2020-07-31 05:37:50 +00:00 committed by Frank Yung-Fong Tang
parent 8ca80c4b6d
commit 863582c2a4
27 changed files with 358 additions and 28 deletions

View file

@ -870,7 +870,7 @@ Calendar::createInstance(const TimeZone& zone, UErrorCode& success)
Calendar* U_EXPORT2
Calendar::createInstance(const Locale& aLocale, UErrorCode& success)
{
return createInstance(TimeZone::createDefault(), aLocale, success);
return createInstance(TimeZone::forLocaleOrDefault(aLocale), aLocale, success);
}
// ------------------------------------- Adopting

View file

@ -49,7 +49,7 @@ static const int32_t LIMITS[UCAL_FIELD_COUNT][4] = {
//-------------------------------------------------------------------------
CECalendar::CECalendar(const Locale& aLocale, UErrorCode& success)
: Calendar(TimeZone::createDefault(), aLocale, success)
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success)
{
setTimeInMillis(getNow(), success);
}

View file

@ -123,7 +123,7 @@ ChineseCalendar* ChineseCalendar::clone() const {
}
ChineseCalendar::ChineseCalendar(const Locale& aLocale, UErrorCode& success)
: Calendar(TimeZone::createDefault(), aLocale, success),
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success),
isLeapYear(FALSE),
fEpochYear(CHINESE_EPOCH_YEAR),
fZoneAstroCalc(getChineseCalZoneAstroCalc())
@ -133,7 +133,7 @@ ChineseCalendar::ChineseCalendar(const Locale& aLocale, UErrorCode& success)
ChineseCalendar::ChineseCalendar(const Locale& aLocale, int32_t epochYear,
const TimeZone* zoneAstroCalc, UErrorCode &success)
: Calendar(TimeZone::createDefault(), aLocale, success),
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success),
isLeapYear(FALSE),
fEpochYear(epochYear),
fZoneAstroCalc(zoneAstroCalc)

View file

@ -185,7 +185,7 @@ fIsGregorian(TRUE), fInvertGregorian(FALSE)
// -------------------------------------
GregorianCalendar::GregorianCalendar(const Locale& aLocale, UErrorCode& status)
: Calendar(TimeZone::createDefault(), aLocale, status),
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, status),
fGregorianCutover(kPapalCutover),
fCutoverJulianDay(kCutoverJulianDay), fNormalizedGregorianCutover(fGregorianCutover), fGregorianCutoverYear(1582),
fIsGregorian(TRUE), fInvertGregorian(FALSE)

View file

@ -155,7 +155,7 @@ U_NAMESPACE_BEGIN
* @internal
*/
HebrewCalendar::HebrewCalendar(const Locale& aLocale, UErrorCode& success)
: Calendar(TimeZone::createDefault(), aLocale, success)
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success)
{
setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly.

View file

@ -40,7 +40,7 @@ IndianCalendar* IndianCalendar::clone() const {
}
IndianCalendar::IndianCalendar(const Locale& aLocale, UErrorCode& success)
: Calendar(TimeZone::createDefault(), aLocale, success)
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success)
{
setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly.
}

View file

@ -232,7 +232,7 @@ IslamicCalendar* IslamicCalendar::clone() const {
}
IslamicCalendar::IslamicCalendar(const Locale& aLocale, UErrorCode& success, ECalculationType type)
: Calendar(TimeZone::createDefault(), aLocale, success),
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success),
cType(type)
{
setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly.

View file

@ -79,7 +79,7 @@ PersianCalendar* PersianCalendar::clone() const {
}
PersianCalendar::PersianCalendar(const Locale& aLocale, UErrorCode& success)
: Calendar(TimeZone::createDefault(), aLocale, success)
: Calendar(TimeZone::forLocaleOrDefault(aLocale), aLocale, success)
{
setTimeInMillis(getNow(), success); // Call this again now that the vtable is set up properly.
}

View file

@ -848,7 +848,8 @@ Calendar*
SimpleDateFormat::initializeCalendar(TimeZone* adoptZone, const Locale& locale, UErrorCode& status)
{
if(!U_FAILURE(status)) {
fCalendar = Calendar::createInstance(adoptZone?adoptZone:TimeZone::createDefault(), locale, status);
fCalendar = Calendar::createInstance(
adoptZone ? adoptZone : TimeZone::forLocaleOrDefault(locale), locale, status);
}
return fCalendar;
}

View file

@ -579,6 +579,24 @@ TimeZone::createDefault()
// -------------------------------------
TimeZone* U_EXPORT2
TimeZone::forLocaleOrDefault(const Locale& locale)
{
char buffer[ULOC_KEYWORDS_CAPACITY] = "";
UErrorCode localStatus = U_ZERO_ERROR;
int32_t count = locale.getKeywordValue("timezone", buffer, sizeof(buffer), localStatus);
if (U_FAILURE(localStatus) || localStatus == U_STRING_NOT_TERMINATED_WARNING) {
// the "timezone" keyword exceeds ULOC_KEYWORDS_CAPACITY; ignore and use default.
count = 0;
}
if (count > 0) {
return TimeZone::createTimeZone(UnicodeString(buffer, count, US_INV));
}
return TimeZone::createDefault();
}
// -------------------------------------
void U_EXPORT2
TimeZone::adoptDefault(TimeZone* zone)
{

View file

@ -317,6 +317,17 @@ public:
*/
static TimeZone* U_EXPORT2 createDefault(void);
/**
* If the locale contains the timezone keyword, creates a copy of that TimeZone.
* Otherwise, create the default timezone.
*
* @param locale a locale which may contains 'timezone' keyword/value.
* @return A TimeZone. Clients are responsible for deleting the time zone
* object returned.
* @internal
*/
static TimeZone* U_EXPORT2 forLocaleOrDefault(const Locale& locale);
/**
* Sets the default time zone (i.e., what's returned by createDefault()) to be the
* specified time zone. If NULL is specified for the time zone, the default time

View file

@ -346,6 +346,13 @@ void CalendarTest::runIndexedTest( int32_t index, UBool exec, const char* &name,
TestChineseCalendarMapping();
}
break;
case 37:
name = "TestTimeZoneInLocale";
if(exec) {
logln("TestTimeZoneInLocale---"); logln("");
TestTimeZoneInLocale();
}
break;
default: name = ""; break;
}
}
@ -2807,6 +2814,51 @@ void CalendarTest::TestCloneLocale(void) {
TEST_CHECK_STATUS;
}
void CalendarTest::TestTimeZoneInLocale(void) {
const char *tests[][3] = {
{ "en-u-tz-usden", "America/Denver", "gregorian" },
{ "es-u-tz-usden", "America/Denver", "gregorian" },
{ "ms-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "zh-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "fr-u-ca-buddhist-tz-phmnl", "Asia/Manila", "buddhist" },
{ "th-u-ca-chinese-tz-gblon", "Europe/London", "chinese" },
{ "de-u-ca-coptic-tz-ciabj", "Africa/Abidjan", "coptic" },
{ "ja-u-ca-dangi-tz-hkhkg", "Asia/Hong_Kong", "dangi" },
{ "da-u-ca-ethioaa-tz-ruunera", "Asia/Ust-Nera", "ethiopic-amete-alem" },
{ "ko-u-ca-ethiopic-tz-cvrai", "Atlantic/Cape_Verde", "ethiopic" },
{ "fil-u-ca-gregory-tz-aubne", "Australia/Brisbane", "gregorian" },
{ "fa-u-ca-hebrew-tz-brrbr", "America/Rio_Branco", "hebrew" },
{ "gr-u-ca-indian-tz-lccas", "America/St_Lucia", "indian" },
{ "or-u-ca-islamic-tz-cayyn", "America/Swift_Current", "islamic" },
{ "my-u-ca-islamic-umalqura-tz-kzala", "Asia/Almaty", "islamic-umalqura" },
{ "lo-u-ca-islamic-tbla-tz-bmbda", "Atlantic/Bermuda", "islamic-tbla" },
{ "km-u-ca-islamic-civil-tz-aqplm", "Antarctica/Palmer", "islamic-civil" },
{ "kk-u-ca-islamic-rgsa-tz-usanc", "America/Anchorage", "islamic" },
{ "ar-u-ca-iso8601-tz-bjptn", "Africa/Porto-Novo", "gregorian" },
{ "he-u-ca-japanese-tz-tzdar", "Africa/Dar_es_Salaam", "japanese" },
{ "bs-u-ca-persian-tz-etadd", "Africa/Addis_Ababa", "persian" },
{ "it-u-ca-roc-tz-aruaq", "America/Argentina/San_Juan", "roc" },
};
for (int32_t i = 0; i < UPRV_LENGTHOF(tests); ++i) {
UErrorCode status = U_ZERO_ERROR;
const char **testLine = tests[i];
Locale locale(testLine[0]);
UnicodeString expected(testLine[1], -1, US_INV);
UnicodeString actual;
LocalPointer<Calendar> calendar(
Calendar::createInstance(locale, status));
if (failure(status, "Calendar::createInstance", TRUE)) continue;
assertEquals("TimeZone from Calendar::createInstance",
expected, calendar->getTimeZone().getID(actual));
assertEquals("Calendar Type from Calendar::createInstance",
testLine[2], calendar->getType());
}
}
void CalendarTest::setAndTestCalendar(Calendar* cal, int32_t initMonth, int32_t initDay, int32_t initYear, UErrorCode& status) {
cal->clear();
cal->setLenient(FALSE);

View file

@ -242,6 +242,8 @@ public: // package
void TestCloneLocale(void);
void TestTimeZoneInLocale(void);
void TestHebrewMonthValidation(void);
/*

View file

@ -88,6 +88,7 @@ void DateFormatTest::runIndexedTest( int32_t index, UBool exec, const char* &nam
TESTCASE_AUTO(TestHebrewClone);
TESTCASE_AUTO(TestDateFormatSymbolsClone);
TESTCASE_AUTO(TestTimeZoneDisplayName);
TESTCASE_AUTO(TestTimeZoneInLocale);
TESTCASE_AUTO(TestRoundtripWithCalendar);
TESTCASE_AUTO(Test6338);
TESTCASE_AUTO(Test6726);
@ -3442,6 +3443,70 @@ void DateFormatTest::TestTimeZoneDisplayName()
}
}
void DateFormatTest::TestTimeZoneInLocale()
{
const char *tests[][3] = {
{ "en-u-tz-usden", "America/Denver", "gregorian" },
{ "es-u-tz-usden", "America/Denver", "gregorian" },
{ "ms-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "zh-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "fr-u-ca-buddhist-tz-phmnl", "Asia/Manila", "buddhist" },
{ "th-u-ca-chinese-tz-gblon", "Europe/London", "chinese" },
{ "de-u-ca-coptic-tz-ciabj", "Africa/Abidjan", "coptic" },
{ "ja-u-ca-dangi-tz-hkhkg", "Asia/Hong_Kong", "dangi" },
{ "da-u-ca-ethioaa-tz-ruunera", "Asia/Ust-Nera", "ethiopic-amete-alem" },
{ "ko-u-ca-ethiopic-tz-cvrai", "Atlantic/Cape_Verde", "ethiopic" },
{ "fil-u-ca-gregory-tz-aubne", "Australia/Brisbane", "gregorian" },
{ "fa-u-ca-hebrew-tz-brrbr", "America/Rio_Branco", "hebrew" },
{ "gr-u-ca-indian-tz-lccas", "America/St_Lucia", "indian" },
{ "or-u-ca-islamic-tz-cayyn", "America/Swift_Current", "islamic" },
{ "my-u-ca-islamic-umalqura-tz-kzala", "Asia/Almaty", "islamic-umalqura" },
{ "lo-u-ca-islamic-tbla-tz-bmbda", "Atlantic/Bermuda", "islamic-tbla" },
{ "km-u-ca-islamic-civil-tz-aqplm", "Antarctica/Palmer", "islamic-civil" },
{ "kk-u-ca-islamic-rgsa-tz-usanc", "America/Anchorage", "islamic" },
{ "ar-u-ca-iso8601-tz-bjptn", "Africa/Porto-Novo", "gregorian" },
{ "he-u-ca-japanese-tz-tzdar", "Africa/Dar_es_Salaam", "japanese" },
{ "bs-u-ca-persian-tz-etadd", "Africa/Addis_Ababa", "persian" },
{ "it-u-ca-roc-tz-aruaq", "America/Argentina/San_Juan", "roc" },
};
for (int32_t i = 0; i < UPRV_LENGTHOF(tests); ++i) {
UErrorCode status = U_ZERO_ERROR;
const char **testLine = tests[i];
Locale locale(testLine[0]);
UnicodeString expectedTimezone(testLine[1], -1, US_INV);
UnicodeString actual;
SimpleDateFormat smptfmt("Z", locale, status);
assertEquals("TimeZone from SimpleDateFormat constructor",
expectedTimezone, smptfmt.getTimeZone().getID(actual));
assertEquals("Calendar from SimpleDateFormat constructor",
testLine[2], smptfmt.getCalendar()->getType());
LocalPointer<DateFormat> datefmt(
DateFormat::createDateInstance(DateFormat::kDefault, locale));
assertEquals("TimeZone from DateFormat::createDateInstance",
expectedTimezone, datefmt->getTimeZone().getID(actual));
assertEquals("Calendar from DateFormat::createDateInstance",
testLine[2], datefmt->getCalendar()->getType());
LocalPointer<DateFormat> timefmt(
DateFormat::createTimeInstance(DateFormat::kDefault, locale));
assertEquals("TimeZone from TimeFormat::createTimeInstance",
expectedTimezone, timefmt->getTimeZone().getID(actual));
assertEquals("Calendar from DateFormat::createTimeInstance",
testLine[2], timefmt->getCalendar()->getType());
LocalPointer<DateFormat> datetimefmt(
DateFormat::createDateTimeInstance(
DateFormat::kDefault, DateFormat::kDefault, locale));
assertEquals("TimeZone from DateTimeFormat::createDateTimeInstance",
expectedTimezone, datetimefmt->getTimeZone().getID(actual));
assertEquals("Calendar from DateFormat::createDateTimeInstance",
testLine[2], datetimefmt->getCalendar()->getType());
}
}
void DateFormatTest::TestRoundtripWithCalendar(void) {
UErrorCode status = U_ZERO_ERROR;

View file

@ -226,6 +226,8 @@ public:
void TestTimeZoneDisplayName(void);
void TestTimeZoneInLocale(void);
void TestRoundtripWithCalendar(void);
public:

View file

@ -76,7 +76,7 @@ abstract class CECalendar extends Calendar {
* @param aLocale The locale for the new calendar.
*/
protected CECalendar(Locale aLocale) {
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -86,7 +86,7 @@ abstract class CECalendar extends Calendar {
* @param locale The locale for the new calendar.
*/
protected CECalendar(ULocale locale) {
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -1764,7 +1764,7 @@ public abstract class Calendar implements Serializable, Cloneable, Comparable<Ca
locale = ULocale.getDefault(Category.FORMAT);
}
if (tz == null) {
tz = TimeZone.getDefault();
tz = TimeZone.forULocaleOrDefault(locale);
}
Calendar cal = createInstance(locale);
@ -1796,7 +1796,7 @@ public abstract class Calendar implements Serializable, Cloneable, Comparable<Ca
private static Calendar createInstance(ULocale locale) {
Calendar cal = null;
TimeZone zone = TimeZone.getDefault();
TimeZone zone = TimeZone.forULocaleOrDefault(locale);
CalType calType = getCalendarTypeForLocale(locale);
if (calType == CalType.UNKNOWN) {
// fallback to Gregorian

View file

@ -284,7 +284,7 @@ public class ChineseCalendar extends Calendar {
* @stable ICU 4.0
*/
public ChineseCalendar(Locale aLocale) {
this(TimeZone.getDefault(), ULocale.forLocale(aLocale), CHINESE_EPOCH_YEAR, CHINA_ZONE);
this(TimeZone.forLocaleOrDefault(aLocale), ULocale.forLocale(aLocale), CHINESE_EPOCH_YEAR, CHINA_ZONE);
}
/**
@ -317,7 +317,7 @@ public class ChineseCalendar extends Calendar {
* @stable ICU 4.0
*/
public ChineseCalendar(ULocale locale) {
this(TimeZone.getDefault(), locale, CHINESE_EPOCH_YEAR, CHINA_ZONE);
this(TimeZone.forULocaleOrDefault(locale), locale, CHINESE_EPOCH_YEAR, CHINA_ZONE);
}
/**

View file

@ -177,7 +177,7 @@ public final class EthiopicCalendar extends CECalendar
* @stable ICU 3.4
*/
public EthiopicCalendar(Locale aLocale) {
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -188,7 +188,7 @@ public final class EthiopicCalendar extends CECalendar
* @stable ICU 3.4
*/
public EthiopicCalendar(ULocale locale) {
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -358,7 +358,7 @@ public class GregorianCalendar extends Calendar {
* @stable ICU 2.0
*/
public GregorianCalendar(Locale aLocale) {
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -368,7 +368,7 @@ public class GregorianCalendar extends Calendar {
* @stable ICU 3.2
*/
public GregorianCalendar(ULocale locale) {
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -305,7 +305,7 @@ public class HebrewCalendar extends Calendar {
* @stable ICU 2.8
*/
public HebrewCalendar(Locale aLocale) {
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -316,7 +316,7 @@ public class HebrewCalendar extends Calendar {
* @stable ICU 3.2
*/
public HebrewCalendar(ULocale locale) {
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -185,7 +185,7 @@ public class IndianCalendar extends Calendar {
* @stable ICU 3.8
*/
public IndianCalendar(Locale aLocale) {
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -196,7 +196,7 @@ public class IndianCalendar extends Calendar {
* @stable ICU 3.8
*/
public IndianCalendar(ULocale locale) {
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -215,7 +215,7 @@ public class IslamicCalendar extends Calendar {
*/
public IslamicCalendar(Locale aLocale)
{
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -227,7 +227,7 @@ public class IslamicCalendar extends Calendar {
*/
public IslamicCalendar(ULocale locale)
{
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -138,7 +138,7 @@ public class PersianCalendar extends Calendar {
@Deprecated
public PersianCalendar(Locale aLocale)
{
this(TimeZone.getDefault(), aLocale);
this(TimeZone.forLocaleOrDefault(aLocale), aLocale);
}
/**
@ -153,7 +153,7 @@ public class PersianCalendar extends Calendar {
@Deprecated
public PersianCalendar(ULocale locale)
{
this(TimeZone.getDefault(), locale);
this(TimeZone.forULocaleOrDefault(locale), locale);
}
/**

View file

@ -28,6 +28,7 @@ import com.ibm.icu.text.TimeZoneFormat.Style;
import com.ibm.icu.text.TimeZoneFormat.TimeType;
import com.ibm.icu.text.TimeZoneNames;
import com.ibm.icu.text.TimeZoneNames.NameType;
import com.ibm.icu.util.ULocale;
import com.ibm.icu.util.ULocale.Category;
/**
@ -915,6 +916,33 @@ abstract public class TimeZone implements Serializable, Cloneable, Freezable<Tim
return ZoneMeta.getEquivalentID(id, index);
}
/**
* If the locale contains the timezone keyword, creates a copy of that
* <code>TimeZone</code>.
* Otherwise, create the default <code>TimeZone</code>.
* @param locale a locale which may contains 'timezone' keyword/value.
* @return A <code>TimeZone</code>. Clients are responsible for deleting the
* <code>TimeZone</code> object returned.
* @internal
*/
public static TimeZone forULocaleOrDefault(ULocale locale) {
String tz = locale.getKeywordValue("timezone");
return (tz == null) ? getDefault() : getTimeZone(tz);
}
/**
* If the locale contains the timezone keyword, creates a copy of that
* <code>TimeZone</code>.
* Otherwise, create the default <code>TimeZone</code>.
* @param locale a locale which may contains 'timezone' keyword/value.
* @return A <code>TimeZone</code>. Clients are responsible for deleting the
* <code>TimeZone</code> object returned.
* @internal
*/
public static TimeZone forLocaleOrDefault(Locale locale) {
return forULocaleOrDefault(ULocale.forLocale(locale));
}
/**
* Gets the default <code>TimeZone</code> for this host.
* The source of the default <code>TimeZone</code>

View file

@ -2555,5 +2555,69 @@ public class CalendarRegressionTest extends com.ibm.icu.dev.test.TestFmwk {
errln("Fail: Expected year=" + year + ", actual=" + resultYear);
}
}
@Test
public void TestTimeZoneInLocale20465() {
String TESTS[][] = {
{ "en-u-tz-usden", "America/Denver", "gregorian" },
{ "es-u-tz-usden", "America/Denver", "gregorian" },
{ "ms-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "zh-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "fr-u-ca-buddhist-tz-phmnl", "Asia/Manila", "buddhist" },
{ "th-u-ca-chinese-tz-gblon", "Europe/London", "chinese" },
{ "de-u-ca-coptic-tz-ciabj", "Africa/Abidjan", "coptic" },
{ "ja-u-ca-dangi-tz-hkhkg", "Asia/Hong_Kong", "dangi" },
{ "da-u-ca-ethioaa-tz-ruunera", "Asia/Ust-Nera", "ethiopic-amete-alem" },
{ "ko-u-ca-ethiopic-tz-cvrai", "Atlantic/Cape_Verde", "ethiopic" },
{ "fil-u-ca-gregory-tz-aubne", "Australia/Brisbane", "gregorian" },
{ "fa-u-ca-hebrew-tz-brrbr", "America/Rio_Branco", "hebrew" },
{ "gr-u-ca-indian-tz-lccas", "America/St_Lucia", "indian" },
{ "or-u-ca-islamic-tz-cayyn", "America/Swift_Current", "islamic" },
{ "my-u-ca-islamic-umalqura-tz-kzala", "Asia/Almaty", "islamic-umalqura" },
{ "lo-u-ca-islamic-tbla-tz-bmbda", "Atlantic/Bermuda", "islamic-tbla" },
{ "km-u-ca-islamic-civil-tz-aqplm", "Antarctica/Palmer", "islamic-civil" },
{ "kk-u-ca-islamic-rgsa-tz-usanc", "America/Anchorage", "islamic" },
{ "ar-u-ca-iso8601-tz-bjptn", "Africa/Porto-Novo", "gregorian" },
{ "he-u-ca-japanese-tz-tzdar", "Africa/Dar_es_Salaam", "japanese" },
{ "bs-u-ca-persian-tz-etadd", "Africa/Addis_Ababa", "persian" },
{ "it-u-ca-roc-tz-aruaq", "America/Argentina/San_Juan", "roc" },
};
TimeZone other = TimeZone.getTimeZone("America/Louisville");
for (int i = 0; i < TESTS.length; ++i) {
ULocale ulocale = new ULocale(TESTS[i][0]);
Locale locale = new Locale(TESTS[i][0]);
Calendar cal = Calendar.getInstance(locale);
assertEquals(
"TimeZone from Calendar.getInstance(Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], cal.getTimeZone().getID());
assertEquals(
"Calendar from Calendar.getInstance(Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], cal.getType());
cal = Calendar.getInstance(ulocale);
assertEquals(
"TimeZone from Calendar.getInstance(ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], cal.getTimeZone().getID());
assertEquals(
"Calendar from Calendar.getInstance(ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], cal.getType());
cal = Calendar.getInstance(other, locale);
assertEquals(
"TimeZone from Calendar.getInstance(TimeZone zone=\"uslui\", Locale loc=\"" + TESTS[i][0] + "\")",
other.getID(), cal.getTimeZone().getID());
assertEquals(
"Calendar from Calendar.getInstance(TimeZone zone=\"uslui\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], cal.getType());
cal = Calendar.getInstance(other, ulocale);
assertEquals(
"TimeZone from Calendar.getInstance(TimeZone zone=\"uslui\", ULocale loc=\"" + TESTS[i][0] + "\")",
other.getID(), cal.getTimeZone().getID());
assertEquals(
"Calendar from Calendar.getInstance(TimeZone zone=\"uslui\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], cal.getType());
}
}
}
//eof

View file

@ -779,6 +779,93 @@ public class DateFormatTest extends TestFmwk {
}
}
@Test
public void TestTimeZoneInLocale() {
String TESTS[][] = {
{ "en-u-tz-usden", "America/Denver", "gregorian" },
{ "es-u-tz-usden", "America/Denver", "gregorian" },
{ "ms-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "zh-u-tz-mykul", "Asia/Kuala_Lumpur", "gregorian" },
{ "fr-u-ca-buddhist-tz-phmnl", "Asia/Manila", "buddhist" },
{ "th-u-ca-chinese-tz-gblon", "Europe/London", "chinese" },
{ "de-u-ca-coptic-tz-ciabj", "Africa/Abidjan", "coptic" },
{ "ja-u-ca-dangi-tz-hkhkg", "Asia/Hong_Kong", "dangi" },
{ "da-u-ca-ethioaa-tz-ruunera", "Asia/Ust-Nera", "ethiopic-amete-alem" },
{ "ko-u-ca-ethiopic-tz-cvrai", "Atlantic/Cape_Verde", "ethiopic" },
{ "fil-u-ca-gregory-tz-aubne", "Australia/Brisbane", "gregorian" },
{ "fa-u-ca-hebrew-tz-brrbr", "America/Rio_Branco", "hebrew" },
{ "gr-u-ca-indian-tz-lccas", "America/St_Lucia", "indian" },
{ "or-u-ca-islamic-tz-cayyn", "America/Swift_Current", "islamic" },
{ "my-u-ca-islamic-umalqura-tz-kzala", "Asia/Almaty", "islamic-umalqura" },
{ "lo-u-ca-islamic-tbla-tz-bmbda", "Atlantic/Bermuda", "islamic-tbla" },
{ "km-u-ca-islamic-civil-tz-aqplm", "Antarctica/Palmer", "islamic-civil" },
{ "kk-u-ca-islamic-rgsa-tz-usanc", "America/Anchorage", "islamic" },
{ "ar-u-ca-iso8601-tz-bjptn", "Africa/Porto-Novo", "gregorian" },
{ "he-u-ca-japanese-tz-tzdar", "Africa/Dar_es_Salaam", "japanese" },
{ "bs-u-ca-persian-tz-etadd", "Africa/Addis_Ababa", "persian" },
{ "it-u-ca-roc-tz-aruaq", "America/Argentina/San_Juan", "roc" },
};
for (int i = 0; i < TESTS.length; ++i) {
ULocale ulocale = new ULocale(TESTS[i][0]);
Locale locale = new Locale(TESTS[i][0]);
SimpleDateFormat smptfmt = new SimpleDateFormat("Z", locale);
assertEquals(
"TimeZone from SimpleDateFormat(\"Z\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], smptfmt.getTimeZone().getID());
assertEquals(
"Calendar from SimpleDateFormat(\"Z\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], smptfmt.getCalendar().getType());
smptfmt = new SimpleDateFormat("Z", "", ulocale);
assertEquals(
"TimeZone from SimpleDateFormat(\"Z\", \"\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], smptfmt.getTimeZone().getID());
assertEquals(
"Calendar from SimpleDateFormat(\"Z\", \"\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], smptfmt.getCalendar().getType());
smptfmt = new SimpleDateFormat("Z", ulocale);
assertEquals(
"TimeZone from SimpleDateFormat(\"Z\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], smptfmt.getTimeZone().getID());
assertEquals(
"Calendar from SimpleDateFormat(\"Z\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], smptfmt.getCalendar().getType());
DateFormat dfmt = DateFormat.getInstanceForSkeleton("Z", locale);
assertEquals(
"TimeZone from DateFormat.getInstanceForSkeleton(\"Z\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], dfmt.getTimeZone().getID());
assertEquals(
"Calendar from DateFormat.getInstanceForSkeleton(\"Z\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], dfmt.getCalendar().getType());
dfmt = DateFormat.getInstanceForSkeleton("Z", ulocale);
assertEquals(
"TimeZone from DateFormat.getInstanceForSkeleton(\"Z\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], dfmt.getTimeZone().getID());
assertEquals(
"Calendar from DateFormat.getInstanceForSkeleton(\"Z\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], dfmt.getCalendar().getType());
dfmt = DateFormat.getPatternInstance("Z", locale);
assertEquals(
"TimeZone from DateFormat.getPatternInstance(\"Z\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], dfmt.getTimeZone().getID());
assertEquals(
"Calendar from DateFormat.getPatternInstance(\"Z\", Locale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], dfmt.getCalendar().getType());
dfmt = DateFormat.getPatternInstance("Z", ulocale);
assertEquals(
"TimeZone from DateFormat.getPatternInstance(\"Z\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][1], dfmt.getTimeZone().getID());
assertEquals(
"Calendar from DateFormat.getPatternInstance(\"Z\", ULocale loc=\"" + TESTS[i][0] + "\")",
TESTS[i][2], dfmt.getCalendar().getType());
}
}
private static final String GMT_BG = "\u0413\u0440\u0438\u043D\u0443\u0438\u0447";
private static final String GMT_ZH = "GMT";
//private static final String GMT_ZH = "\u683C\u6797\u5C3C\u6CBB\u6807\u51C6\u65F6\u95F4";