mirror of
https://github.com/unicode-org/icu.git
synced 2025-04-04 21:15:35 +00:00
6153 lines
251 KiB
C++
6153 lines
251 KiB
C++
// © 2016 and later: Unicode, Inc. and others.
|
|
// License & terms of use: http://www.unicode.org/copyright.html
|
|
/************************************************************************
|
|
* COPYRIGHT:
|
|
* Copyright (c) 1997-2016, International Business Machines Corporation
|
|
* and others. All Rights Reserved.
|
|
************************************************************************/
|
|
#include "unicode/utypes.h"
|
|
|
|
#if !UCONFIG_NO_FORMATTING
|
|
|
|
#include "caltest.h"
|
|
#include "unicode/dtfmtsym.h"
|
|
#include "unicode/gregocal.h"
|
|
#include "unicode/localpointer.h"
|
|
#include "hebrwcal.h"
|
|
#include "unicode/smpdtfmt.h"
|
|
#include "unicode/simpletz.h"
|
|
#include "dbgutil.h"
|
|
#include "unicode/udat.h"
|
|
#include "unicode/ustring.h"
|
|
#include "cstring.h"
|
|
#include "unicode/localpointer.h"
|
|
#include "chnsecal.h"
|
|
#include "intltest.h"
|
|
#include "coptccal.h"
|
|
#include "ethpccal.h"
|
|
#include "islamcal.h"
|
|
|
|
#define mkcstr(U) u_austrcpy(calloc(8, u_strlen(U) + 1), U)
|
|
|
|
#define TEST_CHECK_STATUS UPRV_BLOCK_MACRO_BEGIN { \
|
|
if (U_FAILURE(status)) { \
|
|
if (status == U_MISSING_RESOURCE_ERROR) { \
|
|
dataerrln("%s:%d: Test failure. status=%s", __FILE__, __LINE__, u_errorName(status)); \
|
|
} else { \
|
|
errln("%s:%d: Test failure. status=%s", __FILE__, __LINE__, u_errorName(status)); \
|
|
} \
|
|
return; \
|
|
} \
|
|
} UPRV_BLOCK_MACRO_END
|
|
|
|
#define TEST_CHECK_STATUS_LOCALE(testlocale) UPRV_BLOCK_MACRO_BEGIN { \
|
|
if (U_FAILURE(status)) { \
|
|
if (status == U_MISSING_RESOURCE_ERROR) { \
|
|
dataerrln("%s:%d: Test failure, locale %s. status=%s", __FILE__, __LINE__, testlocale, u_errorName(status)); \
|
|
} else { \
|
|
errln("%s:%d: Test failure, locale %s. status=%s", __FILE__, __LINE__, testlocale, u_errorName(status)); \
|
|
} \
|
|
return; \
|
|
} \
|
|
} UPRV_BLOCK_MACRO_END
|
|
|
|
#define TEST_ASSERT(expr) UPRV_BLOCK_MACRO_BEGIN { \
|
|
if ((expr)==false) { \
|
|
errln("%s:%d: Test failure \n", __FILE__, __LINE__); \
|
|
} \
|
|
} UPRV_BLOCK_MACRO_END
|
|
|
|
// *****************************************************************************
|
|
// class CalendarTest
|
|
// *****************************************************************************
|
|
|
|
UnicodeString CalendarTest::calToStr(const Calendar & cal)
|
|
{
|
|
UnicodeString out;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
int i;
|
|
UDate d;
|
|
for(i = 0;i<UCAL_FIELD_COUNT;i++) {
|
|
out += (UnicodeString("") + fieldName(static_cast<UCalendarDateFields>(i)) + "=" + cal.get(static_cast<UCalendarDateFields>(i), status) + UnicodeString(" "));
|
|
}
|
|
out += "[" + UnicodeString(cal.getType()) + "]";
|
|
|
|
if(cal.inDaylightTime(status)) {
|
|
out += UnicodeString(" (in DST), zone=");
|
|
}
|
|
else {
|
|
out += UnicodeString(", zone=");
|
|
}
|
|
|
|
UnicodeString str2;
|
|
out += cal.getTimeZone().getDisplayName(str2);
|
|
d = cal.getTime(status);
|
|
out += UnicodeString(" :","") + d;
|
|
|
|
return out;
|
|
}
|
|
|
|
void CalendarTest::runIndexedTest( int32_t index, UBool exec, const char* &name, char* /*par*/ )
|
|
{
|
|
if (exec) logln("TestSuite TestCalendar");
|
|
|
|
TESTCASE_AUTO_BEGIN;
|
|
TESTCASE_AUTO(TestDOW943);
|
|
TESTCASE_AUTO(TestClonesUnique908);
|
|
TESTCASE_AUTO(TestGregorianChange768);
|
|
TESTCASE_AUTO(TestDisambiguation765);
|
|
TESTCASE_AUTO(TestGMTvsLocal4064654);
|
|
TESTCASE_AUTO(TestAddSetOrder621);
|
|
TESTCASE_AUTO(TestAdd520);
|
|
TESTCASE_AUTO(TestFieldSet4781);
|
|
// TESTCASE_AUTO(TestSerialize337);
|
|
TESTCASE_AUTO(TestSecondsZero121);
|
|
TESTCASE_AUTO(TestAddSetGet0610);
|
|
TESTCASE_AUTO(TestFields060);
|
|
TESTCASE_AUTO(TestEpochStartFields);
|
|
TESTCASE_AUTO(TestDOWProgression);
|
|
TESTCASE_AUTO(TestGenericAPI);
|
|
TESTCASE_AUTO(TestAddRollExtensive);
|
|
TESTCASE_AUTO(TestDOW_LOCALandYEAR_WOY);
|
|
TESTCASE_AUTO(TestWOY);
|
|
TESTCASE_AUTO(TestRog);
|
|
TESTCASE_AUTO(TestYWOY);
|
|
TESTCASE_AUTO(TestJD);
|
|
TESTCASE_AUTO(TestDebug);
|
|
TESTCASE_AUTO(Test6703);
|
|
TESTCASE_AUTO(Test3785);
|
|
TESTCASE_AUTO(Test1624);
|
|
TESTCASE_AUTO(TestTimeStamp);
|
|
TESTCASE_AUTO(TestISO8601);
|
|
TESTCASE_AUTO(TestAmbiguousWallTimeAPIs);
|
|
TESTCASE_AUTO(TestRepeatedWallTime);
|
|
TESTCASE_AUTO(TestSkippedWallTime);
|
|
TESTCASE_AUTO(TestCloneLocale);
|
|
TESTCASE_AUTO(TestIslamicUmAlQura);
|
|
TESTCASE_AUTO(TestIslamicTabularDates);
|
|
TESTCASE_AUTO(TestHebrewMonthValidation);
|
|
TESTCASE_AUTO(TestWeekData);
|
|
TESTCASE_AUTO(TestAddAcrossZoneTransition);
|
|
TESTCASE_AUTO(TestChineseCalendarMapping);
|
|
TESTCASE_AUTO(TestTimeZoneInLocale);
|
|
TESTCASE_AUTO(TestBasicConversionISO8601);
|
|
TESTCASE_AUTO(TestBasicConversionJapanese);
|
|
TESTCASE_AUTO(TestBasicConversionBuddhist);
|
|
TESTCASE_AUTO(TestBasicConversionTaiwan);
|
|
TESTCASE_AUTO(TestBasicConversionPersian);
|
|
TESTCASE_AUTO(TestBasicConversionIslamic);
|
|
TESTCASE_AUTO(TestBasicConversionIslamicTBLA);
|
|
TESTCASE_AUTO(TestBasicConversionIslamicCivil);
|
|
TESTCASE_AUTO(TestBasicConversionIslamicRGSA);
|
|
TESTCASE_AUTO(TestBasicConversionIslamicUmalqura);
|
|
TESTCASE_AUTO(TestBasicConversionHebrew);
|
|
TESTCASE_AUTO(TestBasicConversionChinese);
|
|
TESTCASE_AUTO(TestBasicConversionDangi);
|
|
TESTCASE_AUTO(TestBasicConversionIndian);
|
|
TESTCASE_AUTO(TestBasicConversionCoptic);
|
|
TESTCASE_AUTO(TestBasicConversionEthiopic);
|
|
TESTCASE_AUTO(TestBasicConversionEthiopicAmeteAlem);
|
|
TESTCASE_AUTO(TestGregorianCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestChineseCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestDangiCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestHebrewCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestIslamicCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestIslamicCivilCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestIslamicUmalquraCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestIslamicRGSACalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestIslamicTBLACalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestPersianCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestIndianCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestTaiwanCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestJapaneseCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestBuddhistCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestCopticCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestEthiopicCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestEthiopicAmeteAlemCalendarInTemporalLeapYear);
|
|
TESTCASE_AUTO(TestChineseCalendarGetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestDangiCalendarGetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestHebrewCalendarGetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestCopticCalendarGetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestEthiopicCalendarGetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestEthiopicAmeteAlemCalendarGetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestGregorianCalendarSetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestChineseCalendarSetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestHebrewCalendarSetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestCopticCalendarSetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestEthiopicCalendarSetTemporalMonthCode);
|
|
TESTCASE_AUTO(TestMostCalendarsOrdinalMonthSet);
|
|
TESTCASE_AUTO(TestChineseCalendarOrdinalMonthSet);
|
|
TESTCASE_AUTO(TestDangiCalendarOrdinalMonthSet);
|
|
TESTCASE_AUTO(TestHebrewCalendarOrdinalMonthSet);
|
|
TESTCASE_AUTO(TestCalendarAddOrdinalMonth);
|
|
TESTCASE_AUTO(TestCalendarRollOrdinalMonth);
|
|
TESTCASE_AUTO(TestLimitsOrdinalMonth);
|
|
TESTCASE_AUTO(TestActualLimitsOrdinalMonth);
|
|
TESTCASE_AUTO(TestMaxActualLimitsWithoutGet23006);
|
|
TESTCASE_AUTO(TestChineseCalendarMonthInSpecialYear);
|
|
TESTCASE_AUTO(TestClearMonth);
|
|
|
|
TESTCASE_AUTO(TestFWWithISO8601);
|
|
TESTCASE_AUTO(TestDangiOverflowIsLeapMonthBetween22507);
|
|
TESTCASE_AUTO(TestRollWeekOfYear);
|
|
TESTCASE_AUTO(TestFirstDayOfWeek);
|
|
|
|
TESTCASE_AUTO(Test22633ChineseOverflow);
|
|
TESTCASE_AUTO(Test22962ChineseOverflow);
|
|
TESTCASE_AUTO(Test22962BuddhistOverflow);
|
|
TESTCASE_AUTO(Test22633IndianOverflow);
|
|
TESTCASE_AUTO(Test22633IslamicUmalquraOverflow);
|
|
TESTCASE_AUTO(Test22633PersianOverflow);
|
|
TESTCASE_AUTO(Test22633HebrewOverflow);
|
|
TESTCASE_AUTO(Test22633AMPMOverflow);
|
|
TESTCASE_AUTO(Test22633SetGetTimeOverflow);
|
|
TESTCASE_AUTO(Test22633Set2FieldsGetTimeOverflow);
|
|
TESTCASE_AUTO(Test22633SetAddGetTimeOverflow);
|
|
TESTCASE_AUTO(Test22633SetRollGetTimeOverflow);
|
|
TESTCASE_AUTO(Test22633AddTwiceGetTimeOverflow);
|
|
TESTCASE_AUTO(Test22633RollTwiceGetTimeOverflow);
|
|
|
|
TESTCASE_AUTO(Test22633HebrewLargeNegativeDay);
|
|
TESTCASE_AUTO(Test23069HebrewHanukkah);
|
|
TESTCASE_AUTO(Test22730JapaneseOverflow);
|
|
TESTCASE_AUTO(Test22730CopticOverflow);
|
|
TESTCASE_AUTO(Test22962ComputeJulianDayOverflow);
|
|
|
|
TESTCASE_AUTO(TestAddOverflow);
|
|
|
|
TESTCASE_AUTO(Test22750Roll);
|
|
|
|
TESTCASE_AUTO(TestChineseCalendarComputeMonthStart);
|
|
TESTCASE_AUTO(Test22962MonthAddOneOverflow);
|
|
|
|
TESTCASE_AUTO_END;
|
|
}
|
|
|
|
// ---------------------------------------------------------------------------------
|
|
|
|
UnicodeString CalendarTest::fieldName(UCalendarDateFields f) {
|
|
switch (f) {
|
|
#define FIELD_NAME_STR(x) case x: return (#x)+5
|
|
FIELD_NAME_STR( UCAL_ERA );
|
|
FIELD_NAME_STR( UCAL_YEAR );
|
|
FIELD_NAME_STR( UCAL_MONTH );
|
|
FIELD_NAME_STR( UCAL_WEEK_OF_YEAR );
|
|
FIELD_NAME_STR( UCAL_WEEK_OF_MONTH );
|
|
FIELD_NAME_STR( UCAL_DATE );
|
|
FIELD_NAME_STR( UCAL_DAY_OF_YEAR );
|
|
FIELD_NAME_STR( UCAL_DAY_OF_WEEK );
|
|
FIELD_NAME_STR( UCAL_DAY_OF_WEEK_IN_MONTH );
|
|
FIELD_NAME_STR( UCAL_AM_PM );
|
|
FIELD_NAME_STR( UCAL_HOUR );
|
|
FIELD_NAME_STR( UCAL_HOUR_OF_DAY );
|
|
FIELD_NAME_STR( UCAL_MINUTE );
|
|
FIELD_NAME_STR( UCAL_SECOND );
|
|
FIELD_NAME_STR( UCAL_MILLISECOND );
|
|
FIELD_NAME_STR( UCAL_ZONE_OFFSET );
|
|
FIELD_NAME_STR( UCAL_DST_OFFSET );
|
|
FIELD_NAME_STR( UCAL_YEAR_WOY );
|
|
FIELD_NAME_STR( UCAL_DOW_LOCAL );
|
|
FIELD_NAME_STR( UCAL_EXTENDED_YEAR );
|
|
FIELD_NAME_STR( UCAL_JULIAN_DAY );
|
|
FIELD_NAME_STR( UCAL_MILLISECONDS_IN_DAY );
|
|
#undef FIELD_NAME_STR
|
|
default:
|
|
return UnicodeString("") + static_cast<int32_t>(f);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Test various API methods for API completeness.
|
|
*/
|
|
void
|
|
CalendarTest::TestGenericAPI()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UDate d;
|
|
UnicodeString str;
|
|
UBool eq = false,b4 = false,af = false;
|
|
|
|
UDate when = date(90, UCAL_APRIL, 15);
|
|
|
|
UnicodeString tzid("TestZone");
|
|
int32_t tzoffset = 123400;
|
|
|
|
SimpleTimeZone *zone = new SimpleTimeZone(tzoffset, tzid);
|
|
Calendar *cal = Calendar::createInstance(zone->clone(), status);
|
|
if (failure(status, "Calendar::createInstance #1", true)) return;
|
|
|
|
if (*zone != cal->getTimeZone()) errln("FAIL: Calendar::getTimeZone failed");
|
|
|
|
Calendar *cal2 = Calendar::createInstance(cal->getTimeZone(), status);
|
|
if (failure(status, "Calendar::createInstance #2")) return;
|
|
cal->setTime(when, status);
|
|
cal2->setTime(when, status);
|
|
if (failure(status, "Calendar::setTime")) return;
|
|
|
|
if (!(*cal == *cal2)) errln("FAIL: Calendar::operator== failed");
|
|
if ((*cal != *cal2)) errln("FAIL: Calendar::operator!= failed");
|
|
if (!cal->equals(*cal2, status) ||
|
|
cal->before(*cal2, status) ||
|
|
cal->after(*cal2, status) ||
|
|
U_FAILURE(status)) errln("FAIL: equals/before/after failed");
|
|
|
|
logln(UnicodeString("cal=") + cal->getTime(status) + (calToStr(*cal)));
|
|
logln(UnicodeString("cal2=") + cal2->getTime(status) + (calToStr(*cal2)));
|
|
logln("cal2->setTime(when+1000)");
|
|
cal2->setTime(when + 1000, status);
|
|
logln(UnicodeString("cal2=") + cal2->getTime(status) + (calToStr(*cal2)));
|
|
|
|
if (failure(status, "Calendar::setTime")) return;
|
|
if (cal->equals(*cal2, status) ||
|
|
cal2->before(*cal, status) ||
|
|
cal->after(*cal2, status) ||
|
|
U_FAILURE(status)) errln("FAIL: equals/before/after failed after setTime(+1000)");
|
|
|
|
logln("cal->roll(UCAL_SECOND)");
|
|
cal->roll(UCAL_SECOND, static_cast<UBool>(true), status);
|
|
logln(UnicodeString("cal=") + cal->getTime(status) + (calToStr(*cal)));
|
|
cal->roll(UCAL_SECOND, static_cast<int32_t>(0), status);
|
|
logln(UnicodeString("cal=") + cal->getTime(status) + (calToStr(*cal)));
|
|
if (failure(status, "Calendar::roll")) return;
|
|
|
|
if (!(eq=cal->equals(*cal2, status)) ||
|
|
(b4=cal->before(*cal2, status)) ||
|
|
(af=cal->after(*cal2, status)) ||
|
|
U_FAILURE(status)) {
|
|
errln("FAIL: equals[%c]/before[%c]/after[%c] failed after roll 1 second [should be T/F/F]",
|
|
eq?'T':'F',
|
|
b4?'T':'F',
|
|
af?'T':'F');
|
|
logln(UnicodeString("cal=") + cal->getTime(status) + (calToStr(*cal)));
|
|
logln(UnicodeString("cal2=") + cal2->getTime(status) + (calToStr(*cal2)));
|
|
}
|
|
|
|
// Roll back to January
|
|
cal->roll(UCAL_MONTH, static_cast<int32_t>(1 + UCAL_DECEMBER - cal->get(UCAL_MONTH, status)), status);
|
|
if (failure(status, "Calendar::roll")) return;
|
|
if (cal->equals(*cal2, status) ||
|
|
cal2->before(*cal, status) ||
|
|
cal->after(*cal2, status) ||
|
|
U_FAILURE(status)) errln("FAIL: equals/before/after failed after rollback to January");
|
|
|
|
TimeZone *z = cal->orphanTimeZone();
|
|
if (z->getID(str) != tzid ||
|
|
z->getRawOffset() != tzoffset)
|
|
errln("FAIL: orphanTimeZone failed");
|
|
|
|
int32_t i;
|
|
for (i=0; i<2; ++i)
|
|
{
|
|
UBool lenient = ( i > 0 );
|
|
cal->setLenient(lenient);
|
|
if (lenient != cal->isLenient()) errln("FAIL: setLenient/isLenient failed");
|
|
// Later: Check for lenient behavior
|
|
}
|
|
|
|
for (i=UCAL_SUNDAY; i<=UCAL_SATURDAY; ++i)
|
|
{
|
|
cal->setFirstDayOfWeek(static_cast<UCalendarDaysOfWeek>(i));
|
|
if (cal->getFirstDayOfWeek() != i) errln("FAIL: set/getFirstDayOfWeek failed");
|
|
UErrorCode aStatus = U_ZERO_ERROR;
|
|
if (cal->getFirstDayOfWeek(aStatus) != i || U_FAILURE(aStatus)) errln("FAIL: getFirstDayOfWeek(status) failed");
|
|
}
|
|
|
|
for (i=1; i<=7; ++i)
|
|
{
|
|
cal->setMinimalDaysInFirstWeek(static_cast<uint8_t>(i));
|
|
if (cal->getMinimalDaysInFirstWeek() != i) errln("FAIL: set/getFirstDayOfWeek failed");
|
|
}
|
|
|
|
for (i=0; i<UCAL_FIELD_COUNT; ++i)
|
|
{
|
|
if (cal->getMinimum(static_cast<UCalendarDateFields>(i)) > cal->getGreatestMinimum(static_cast<UCalendarDateFields>(i)))
|
|
errln(UnicodeString("FAIL: getMinimum larger than getGreatestMinimum for field ") + i);
|
|
if (cal->getLeastMaximum(static_cast<UCalendarDateFields>(i)) > cal->getMaximum(static_cast<UCalendarDateFields>(i)))
|
|
errln(UnicodeString("FAIL: getLeastMaximum larger than getMaximum for field ") + i);
|
|
if (cal->getMinimum(static_cast<UCalendarDateFields>(i)) >= cal->getMaximum(static_cast<UCalendarDateFields>(i)))
|
|
errln(UnicodeString("FAIL: getMinimum not less than getMaximum for field ") + i);
|
|
}
|
|
|
|
cal->adoptTimeZone(TimeZone::createDefault());
|
|
cal->clear();
|
|
cal->set(1984, 5, 24);
|
|
if (cal->getTime(status) != date(84, 5, 24) || U_FAILURE(status))
|
|
errln("FAIL: Calendar::set(3 args) failed");
|
|
|
|
cal->clear();
|
|
cal->set(1985, 3, 2, 11, 49);
|
|
if (cal->getTime(status) != date(85, 3, 2, 11, 49) || U_FAILURE(status))
|
|
errln("FAIL: Calendar::set(5 args) failed");
|
|
|
|
cal->clear();
|
|
cal->set(1995, 9, 12, 1, 39, 55);
|
|
if (cal->getTime(status) != date(95, 9, 12, 1, 39, 55) || U_FAILURE(status))
|
|
errln("FAIL: Calendar::set(6 args) failed");
|
|
|
|
cal->getTime(status);
|
|
if (failure(status, "Calendar::getTime")) return;
|
|
for (i=0; i<UCAL_FIELD_COUNT; ++i)
|
|
{
|
|
switch(i) {
|
|
case UCAL_YEAR: case UCAL_MONTH: case UCAL_DATE:
|
|
case UCAL_HOUR_OF_DAY: case UCAL_MINUTE: case UCAL_SECOND:
|
|
case UCAL_EXTENDED_YEAR:
|
|
if (!cal->isSet(static_cast<UCalendarDateFields>(i))) errln("FAIL: Calendar::isSet F, should be T " + fieldName(static_cast<UCalendarDateFields>(i)));
|
|
break;
|
|
default:
|
|
if (cal->isSet(static_cast<UCalendarDateFields>(i))) errln("FAIL: Calendar::isSet = T, should be F " + fieldName(static_cast<UCalendarDateFields>(i)));
|
|
}
|
|
cal->clear(static_cast<UCalendarDateFields>(i));
|
|
if (cal->isSet(static_cast<UCalendarDateFields>(i))) errln("FAIL: Calendar::clear/isSet failed " + fieldName(static_cast<UCalendarDateFields>(i)));
|
|
}
|
|
|
|
if(cal->getActualMinimum(Calendar::SECOND, status) != 0){
|
|
errln("Calendar is suppose to return 0 for getActualMinimum");
|
|
}
|
|
|
|
Calendar *cal3 = Calendar::createInstance(status);
|
|
cal3->roll(Calendar::SECOND, static_cast<int32_t>(0), status);
|
|
if (failure(status, "Calendar::roll(EDateFields, int32_t, UErrorCode)")) return;
|
|
|
|
delete cal;
|
|
delete cal2;
|
|
delete cal3;
|
|
|
|
int32_t count;
|
|
const Locale* loc = Calendar::getAvailableLocales(count);
|
|
if (count < 1 || loc == nullptr)
|
|
{
|
|
dataerrln("FAIL: getAvailableLocales failed");
|
|
}
|
|
else
|
|
{
|
|
for (i=0; i<count; ++i)
|
|
{
|
|
cal = Calendar::createInstance(loc[i], status);
|
|
if (U_FAILURE(status)) {
|
|
errcheckln(status, UnicodeString("FAIL: Calendar::createInstance #3, locale ") + loc[i].getName() + " , error " + u_errorName(status));
|
|
return;
|
|
}
|
|
delete cal;
|
|
}
|
|
}
|
|
|
|
cal = Calendar::createInstance(TimeZone::createDefault(), Locale::getEnglish(), status);
|
|
if (failure(status, "Calendar::createInstance #4")) return;
|
|
delete cal;
|
|
|
|
cal = Calendar::createInstance(*zone, Locale::getEnglish(), status);
|
|
if (failure(status, "Calendar::createInstance #5")) return;
|
|
delete cal;
|
|
|
|
GregorianCalendar *gc = new GregorianCalendar(*zone, status);
|
|
if (failure(status, "new GregorianCalendar")) return;
|
|
delete gc;
|
|
|
|
gc = new GregorianCalendar(Locale::getEnglish(), status);
|
|
if (failure(status, "new GregorianCalendar")) return;
|
|
delete gc;
|
|
|
|
gc = new GregorianCalendar(Locale::getEnglish(), status);
|
|
delete gc;
|
|
|
|
gc = new GregorianCalendar(*zone, Locale::getEnglish(), status);
|
|
if (failure(status, "new GregorianCalendar")) return;
|
|
delete gc;
|
|
|
|
gc = new GregorianCalendar(zone, status);
|
|
if (failure(status, "new GregorianCalendar")) return;
|
|
delete gc;
|
|
|
|
gc = new GregorianCalendar(1998, 10, 14, 21, 43, status);
|
|
if (gc->getTime(status) != (d =date(98, 10, 14, 21, 43) )|| U_FAILURE(status))
|
|
errln("FAIL: new GregorianCalendar(ymdhm) failed with " + UnicodeString(u_errorName(status)) + ", cal=" + gc->getTime(status) + (calToStr(*gc)) + ", d=" + d);
|
|
else
|
|
logln(UnicodeString("GOOD: cal=") + gc->getTime(status) + calToStr(*gc) + ", d=" + d);
|
|
delete gc;
|
|
|
|
gc = new GregorianCalendar(1998, 10, 14, 21, 43, 55, status);
|
|
if (gc->getTime(status) != (d=date(98, 10, 14, 21, 43, 55)) || U_FAILURE(status))
|
|
errln("FAIL: new GregorianCalendar(ymdhms) failed with " + UnicodeString(u_errorName(status)));
|
|
|
|
GregorianCalendar gc2(Locale::getEnglish(), status);
|
|
if (failure(status, "new GregorianCalendar")) return;
|
|
gc2 = *gc;
|
|
if (gc2 != *gc || !(gc2 == *gc)) errln("FAIL: GregorianCalendar assignment/operator==/operator!= failed");
|
|
delete gc;
|
|
delete z;
|
|
|
|
/* Code coverage for Calendar class. */
|
|
cal = Calendar::createInstance(status);
|
|
if (failure(status, "Calendar::createInstance #6")) {
|
|
return;
|
|
}else {
|
|
cal->roll(UCAL_HOUR, static_cast<int32_t>(100), status);
|
|
cal->clear(UCAL_HOUR);
|
|
#if !UCONFIG_NO_SERVICE
|
|
URegistryKey key = cal->registerFactory(nullptr, status);
|
|
cal->unregister(key, status);
|
|
#endif
|
|
}
|
|
delete cal;
|
|
|
|
status = U_ZERO_ERROR;
|
|
cal = Calendar::createInstance(Locale("he_IL@calendar=hebrew"), status);
|
|
if (failure(status, "Calendar::createInstance #7")) {
|
|
return;
|
|
} else {
|
|
cal->roll(Calendar::MONTH, static_cast<int32_t>(100), status);
|
|
}
|
|
|
|
LocalPointer<StringEnumeration> values(
|
|
Calendar::getKeywordValuesForLocale("calendar", Locale("he"), false, status));
|
|
if (values.isNull() || U_FAILURE(status)) {
|
|
dataerrln("FAIL: Calendar::getKeywordValuesForLocale(he): %s", u_errorName(status));
|
|
} else {
|
|
UBool containsHebrew = false;
|
|
const char *charValue;
|
|
int32_t valueLength;
|
|
while ((charValue = values->next(&valueLength, status)) != nullptr) {
|
|
if (valueLength == 6 && uprv_strcmp(charValue, "hebrew") == 0) {
|
|
containsHebrew = true;
|
|
}
|
|
}
|
|
if (!containsHebrew) {
|
|
errln("Calendar::getKeywordValuesForLocale(he)->next() does not contain \"hebrew\"");
|
|
}
|
|
|
|
values->reset(status);
|
|
containsHebrew = false;
|
|
UnicodeString hebrew = UNICODE_STRING_SIMPLE("hebrew");
|
|
const char16_t *ucharValue;
|
|
while ((ucharValue = values->unext(&valueLength, status)) != nullptr) {
|
|
UnicodeString value(false, ucharValue, valueLength);
|
|
if (value == hebrew) {
|
|
containsHebrew = true;
|
|
}
|
|
}
|
|
if (!containsHebrew) {
|
|
errln("Calendar::getKeywordValuesForLocale(he)->unext() does not contain \"hebrew\"");
|
|
}
|
|
|
|
values->reset(status);
|
|
containsHebrew = false;
|
|
const UnicodeString *stringValue;
|
|
while ((stringValue = values->snext(status)) != nullptr) {
|
|
if (*stringValue == hebrew) {
|
|
containsHebrew = true;
|
|
}
|
|
}
|
|
if (!containsHebrew) {
|
|
errln("Calendar::getKeywordValuesForLocale(he)->snext() does not contain \"hebrew\"");
|
|
}
|
|
}
|
|
delete cal;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* This test confirms the correct behavior of add when incrementing
|
|
* through subsequent days.
|
|
*/
|
|
void
|
|
CalendarTest::TestRog()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar* gc = new GregorianCalendar(status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
int32_t year = 1997, month = UCAL_APRIL, date = 1;
|
|
gc->set(year, month, date);
|
|
gc->set(UCAL_HOUR_OF_DAY, 23);
|
|
gc->set(UCAL_MINUTE, 0);
|
|
gc->set(UCAL_SECOND, 0);
|
|
gc->set(UCAL_MILLISECOND, 0);
|
|
for (int32_t i = 0; i < 9; i++, gc->add(UCAL_DATE, 1, status)) {
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
if (gc->get(UCAL_YEAR, status) != year ||
|
|
gc->get(UCAL_MONTH, status) != month ||
|
|
gc->get(UCAL_DATE, status) != (date + i)) errln("FAIL: Date wrong");
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
delete gc;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Test the handling of the day of the week, checking for correctness and
|
|
* for correct minimum and maximum values.
|
|
*/
|
|
void
|
|
CalendarTest::TestDOW943()
|
|
{
|
|
dowTest(false);
|
|
dowTest(true);
|
|
}
|
|
|
|
void CalendarTest::dowTest(UBool lenient)
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar* cal = new GregorianCalendar(status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
logln("cal - Aug 12, 1997\n");
|
|
cal->set(1997, UCAL_AUGUST, 12);
|
|
cal->getTime(status);
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
logln((lenient ? UnicodeString("LENIENT0: ") : UnicodeString("nonlenient0: ")) + (calToStr(*cal)));
|
|
cal->setLenient(lenient);
|
|
logln("cal - Dec 1, 1996\n");
|
|
cal->set(1996, UCAL_DECEMBER, 1);
|
|
logln((lenient ? UnicodeString("LENIENT: ") : UnicodeString("nonlenient: ")) + (calToStr(*cal)));
|
|
int32_t dow = cal->get(UCAL_DAY_OF_WEEK, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed [%s]", u_errorName(status)); return; }
|
|
int32_t min = cal->getMinimum(UCAL_DAY_OF_WEEK);
|
|
int32_t max = cal->getMaximum(UCAL_DAY_OF_WEEK);
|
|
if (dow < min ||
|
|
dow > max) errln(UnicodeString("FAIL: Day of week ") + dow + " out of range");
|
|
if (dow != UCAL_SUNDAY) errln("FAIL: Day of week should be SUNDAY[%d] not %d", UCAL_SUNDAY, dow);
|
|
if (min != UCAL_SUNDAY ||
|
|
max != UCAL_SATURDAY) errln("FAIL: Min/max bad");
|
|
delete cal;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Confirm that cloned Calendar objects do not inadvertently share substructures.
|
|
*/
|
|
void
|
|
CalendarTest::TestClonesUnique908()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *c = Calendar::createInstance(status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
Calendar *d = c->clone();
|
|
c->set(UCAL_MILLISECOND, 123);
|
|
d->set(UCAL_MILLISECOND, 456);
|
|
if (c->get(UCAL_MILLISECOND, status) != 123 ||
|
|
d->get(UCAL_MILLISECOND, status) != 456) {
|
|
errln("FAIL: Clones share fields");
|
|
}
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
delete c;
|
|
delete d;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Confirm that the Gregorian cutoff value works as advertised.
|
|
*/
|
|
void
|
|
CalendarTest::TestGregorianChange768()
|
|
{
|
|
UBool b;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UnicodeString str;
|
|
GregorianCalendar* c = new GregorianCalendar(status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
logln(UnicodeString("With cutoff ") + dateToString(c->getGregorianChange(), str));
|
|
b = c->isLeapYear(1800);
|
|
logln(UnicodeString(" isLeapYear(1800) = ") + (b ? "true" : "false"));
|
|
logln(UnicodeString(" (should be false)"));
|
|
if (b) errln("FAIL");
|
|
c->setGregorianChange(date(0, 0, 1), status);
|
|
if (U_FAILURE(status)) { errln("GregorianCalendar::setGregorianChange failed"); return; }
|
|
logln(UnicodeString("With cutoff ") + dateToString(c->getGregorianChange(), str));
|
|
b = c->isLeapYear(1800);
|
|
logln(UnicodeString(" isLeapYear(1800) = ") + (b ? "true" : "false"));
|
|
logln(UnicodeString(" (should be true)"));
|
|
if (!b) errln("FAIL");
|
|
delete c;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Confirm the functioning of the field disambiguation algorithm.
|
|
*/
|
|
void
|
|
CalendarTest::TestDisambiguation765()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *c = Calendar::createInstance("en_US", status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
c->setLenient(false);
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_DATE, 3);
|
|
verify765("1997 third day of June = ", c, 1997, UCAL_JUNE, 3);
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_DAY_OF_WEEK_IN_MONTH, 1);
|
|
verify765("1997 first Tuesday in June = ", c, 1997, UCAL_JUNE, 3);
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_DAY_OF_WEEK_IN_MONTH, - 1);
|
|
verify765("1997 last Tuesday in June = ", c, 1997, UCAL_JUNE, 24);
|
|
|
|
status = U_ZERO_ERROR;
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_DAY_OF_WEEK_IN_MONTH, 0);
|
|
c->getTime(status);
|
|
verify765("1997 zero-th Tuesday in June = ", status);
|
|
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_WEEK_OF_MONTH, 1);
|
|
verify765("1997 Tuesday in week 1 of June = ", c, 1997, UCAL_JUNE, 3);
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_WEEK_OF_MONTH, 5);
|
|
verify765("1997 Tuesday in week 5 of June = ", c, 1997, UCAL_JULY, 1);
|
|
|
|
status = U_ZERO_ERROR;
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_MONTH, UCAL_JUNE);
|
|
c->set(UCAL_WEEK_OF_MONTH, 0);
|
|
c->setMinimalDaysInFirstWeek(1);
|
|
c->getTime(status);
|
|
verify765("1997 Tuesday in week 0 of June = ", status);
|
|
|
|
/* Note: The following test used to expect YEAR 1997, WOY 1 to
|
|
* resolve to a date in Dec 1996; that is, to behave as if
|
|
* YEAR_WOY were 1997. With the addition of a new explicit
|
|
* YEAR_WOY field, YEAR_WOY must itself be set if that is what is
|
|
* desired. Using YEAR in combination with WOY is ambiguous, and
|
|
* results in the first WOY/DOW day of the year satisfying the
|
|
* given fields (there may be up to two such days). In this case,
|
|
* it properly resolves to Tue Dec 30 1997, which has a WOY value
|
|
* of 1 (for YEAR_WOY 1998) and a DOW of Tuesday, and falls in the
|
|
* _calendar_ year 1997, as specified. - aliu */
|
|
c->clear();
|
|
c->set(UCAL_YEAR_WOY, 1997); // aliu
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_WEEK_OF_YEAR, 1);
|
|
verify765("1997 Tuesday in week 1 of yearWOY = ", c, 1996, UCAL_DECEMBER, 31);
|
|
c->clear(); // - add test for YEAR
|
|
c->setMinimalDaysInFirstWeek(1);
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_WEEK_OF_YEAR, 1);
|
|
verify765("1997 Tuesday in week 1 of year = ", c, 1997, UCAL_DECEMBER, 30);
|
|
c->clear();
|
|
c->set(UCAL_YEAR, 1997);
|
|
c->set(UCAL_DAY_OF_WEEK, UCAL_TUESDAY);
|
|
c->set(UCAL_WEEK_OF_YEAR, 10);
|
|
verify765("1997 Tuesday in week 10 of year = ", c, 1997, UCAL_MARCH, 4);
|
|
//try {
|
|
|
|
// {sfb} week 0 is no longer a valid week of year
|
|
/*c->clear();
|
|
c->set(Calendar::YEAR, 1997);
|
|
c->set(Calendar::DAY_OF_WEEK, Calendar::TUESDAY);
|
|
//c->set(Calendar::WEEK_OF_YEAR, 0);
|
|
c->set(Calendar::WEEK_OF_YEAR, 1);
|
|
verify765("1997 Tuesday in week 0 of year = ", c, 1996, Calendar::DECEMBER, 24);*/
|
|
|
|
//}
|
|
//catch(IllegalArgumentException ex) {
|
|
// errln("FAIL: Exception seen:");
|
|
// ex.printStackTrace(log);
|
|
//}
|
|
delete c;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
void
|
|
CalendarTest::verify765(const UnicodeString& msg, Calendar* c, int32_t year, int32_t month, int32_t day)
|
|
{
|
|
UnicodeString str;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
int32_t y = c->get(UCAL_YEAR, status);
|
|
int32_t m = c->get(UCAL_MONTH, status);
|
|
int32_t d = c->get(UCAL_DATE, status);
|
|
if ( y == year &&
|
|
m == month &&
|
|
d == day) {
|
|
if (U_FAILURE(status)) { errln("FAIL: Calendar::get failed"); return; }
|
|
logln("PASS: " + msg + dateToString(c->getTime(status), str));
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
}
|
|
else {
|
|
errln("FAIL: " + msg + dateToString(c->getTime(status), str) + "; expected " + year + "/" + static_cast<int32_t>(month + 1) + "/" + day +
|
|
"; got " + y + "/" + static_cast<int32_t>(m + 1) + "/" + d + " for Locale: " + c->getLocaleID(ULOC_ACTUAL_LOCALE, status));
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
}
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
void
|
|
CalendarTest::verify765(const UnicodeString& msg/*, IllegalArgumentException e*/, UErrorCode status)
|
|
{
|
|
if (status != U_ILLEGAL_ARGUMENT_ERROR) errln("FAIL: No IllegalArgumentException for " + msg);
|
|
else logln("PASS: " + msg + "IllegalArgument as expected");
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Confirm that the offset between local time and GMT behaves as expected.
|
|
*/
|
|
void
|
|
CalendarTest::TestGMTvsLocal4064654()
|
|
{
|
|
test4064654(1997, 1, 1, 12, 0, 0);
|
|
test4064654(1997, 4, 16, 18, 30, 0);
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
void
|
|
CalendarTest::test4064654(int32_t yr, int32_t mo, int32_t dt, int32_t hr, int32_t mn, int32_t sc)
|
|
{
|
|
UDate date;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UnicodeString str;
|
|
Calendar *gmtcal = Calendar::createInstance(status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
gmtcal->adoptTimeZone(TimeZone::createTimeZone("Africa/Casablanca"));
|
|
gmtcal->set(yr, mo - 1, dt, hr, mn, sc);
|
|
gmtcal->set(UCAL_MILLISECOND, 0);
|
|
date = gmtcal->getTime(status);
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
logln("date = " + dateToString(date, str));
|
|
Calendar *cal = Calendar::createInstance(status);
|
|
if (U_FAILURE(status)) { errln("Calendar::createInstance failed"); return; }
|
|
cal->setTime(date, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
|
|
int32_t offset = cal->getTimeZone().getOffset(static_cast<uint8_t>(cal->get(UCAL_ERA, status)),
|
|
cal->get(UCAL_YEAR, status),
|
|
cal->get(UCAL_MONTH, status),
|
|
cal->get(UCAL_DATE, status),
|
|
static_cast<uint8_t>(cal->get(UCAL_DAY_OF_WEEK, status)),
|
|
cal->get(UCAL_MILLISECOND, status), status);
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
logln("offset for " + dateToString(date, str) + "= " + (offset / 1000 / 60 / 60.0) + "hr");
|
|
int32_t utc = ((cal->get(UCAL_HOUR_OF_DAY, status) * 60 +
|
|
cal->get(UCAL_MINUTE, status)) * 60 +
|
|
cal->get(UCAL_SECOND, status)) * 1000 +
|
|
cal->get(UCAL_MILLISECOND, status) - offset;
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
int32_t expected = ((hr * 60 + mn) * 60 + sc) * 1000;
|
|
if (utc != expected) errln(UnicodeString("FAIL: Discrepancy of ") + (utc - expected) +
|
|
" millis = " + ((utc - expected) / 1000 / 60 / 60.0) + " hr");
|
|
delete gmtcal;
|
|
delete cal;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* The operations of adding and setting should not exhibit pathological
|
|
* dependence on the order of operations. This test checks for this.
|
|
*/
|
|
void
|
|
CalendarTest::TestAddSetOrder621()
|
|
{
|
|
UDate d = date(97, 4, 14, 13, 23, 45);
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *cal = Calendar::createInstance(status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
|
|
cal->setTime(d, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::setTime failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
cal->add(UCAL_DATE, - 5, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::add failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
cal->set(UCAL_HOUR_OF_DAY, 0);
|
|
cal->set(UCAL_MINUTE, 0);
|
|
cal->set(UCAL_SECOND, 0);
|
|
UnicodeString s;
|
|
dateToString(cal->getTime(status), s);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::getTime failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
delete cal;
|
|
|
|
cal = Calendar::createInstance(status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::createInstance failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
cal->setTime(d, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::setTime failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
cal->set(UCAL_HOUR_OF_DAY, 0);
|
|
cal->set(UCAL_MINUTE, 0);
|
|
cal->set(UCAL_SECOND, 0);
|
|
cal->add(UCAL_DATE, - 5, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::add failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
UnicodeString s2;
|
|
dateToString(cal->getTime(status), s2);
|
|
if (U_FAILURE(status)) {
|
|
errln("Calendar::getTime failed");
|
|
delete cal;
|
|
return;
|
|
}
|
|
if (s == s2)
|
|
logln("Pass: " + s + " == " + s2);
|
|
else
|
|
errln("FAIL: " + s + " != " + s2);
|
|
delete cal;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Confirm that adding to various fields works.
|
|
*/
|
|
void
|
|
CalendarTest::TestAdd520()
|
|
{
|
|
int32_t y = 1997, m = UCAL_FEBRUARY, d = 1;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar *temp = new GregorianCalendar(y, m, d, status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
check520(temp, y, m, d);
|
|
temp->add(UCAL_YEAR, 1, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
y++;
|
|
check520(temp, y, m, d);
|
|
temp->add(UCAL_MONTH, 1, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
m++;
|
|
check520(temp, y, m, d);
|
|
temp->add(UCAL_DATE, 1, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
d++;
|
|
check520(temp, y, m, d);
|
|
temp->add(UCAL_DATE, 2, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
d += 2;
|
|
check520(temp, y, m, d);
|
|
temp->add(UCAL_DATE, 28, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
d = 1;++m;
|
|
check520(temp, y, m, d);
|
|
delete temp;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Execute adding and rolling in GregorianCalendar extensively,
|
|
*/
|
|
void
|
|
CalendarTest::TestAddRollExtensive()
|
|
{
|
|
int32_t maxlimit = 40;
|
|
int32_t y = 1997, m = UCAL_FEBRUARY, d = 1, hr = 1, min = 1, sec = 0, ms = 0;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar *temp = new GregorianCalendar(y, m, d, status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
|
|
temp->set(UCAL_HOUR, hr);
|
|
temp->set(UCAL_MINUTE, min);
|
|
temp->set(UCAL_SECOND, sec);
|
|
temp->set(UCAL_MILLISECOND, ms);
|
|
temp->setMinimalDaysInFirstWeek(1);
|
|
|
|
UCalendarDateFields e;
|
|
|
|
logln("Testing GregorianCalendar add...");
|
|
e = UCAL_YEAR;
|
|
while (e < UCAL_FIELD_COUNT) {
|
|
int32_t i;
|
|
int32_t limit = maxlimit;
|
|
status = U_ZERO_ERROR;
|
|
for (i = 0; i < limit; i++) {
|
|
temp->add(e, 1, status);
|
|
if (U_FAILURE(status)) {
|
|
limit = i;
|
|
status = U_ZERO_ERROR;
|
|
break; // Suppress compile warning. Shouldn't be necessary, but it is.
|
|
}
|
|
}
|
|
for (i = 0; i < limit; i++) {
|
|
temp->add(e, -1, status);
|
|
if (U_FAILURE(status)) { errln("GregorianCalendar::add -1 failed"); return; }
|
|
}
|
|
check520(temp, y, m, d, hr, min, sec, ms, e);
|
|
|
|
e = static_cast<UCalendarDateFields>(static_cast<int32_t>(e) + 1);
|
|
}
|
|
|
|
logln("Testing GregorianCalendar roll...");
|
|
e = UCAL_YEAR;
|
|
while (e < UCAL_FIELD_COUNT) {
|
|
int32_t i;
|
|
int32_t limit = maxlimit;
|
|
status = U_ZERO_ERROR;
|
|
for (i = 0; i < limit; i++) {
|
|
logln(calToStr(*temp) + UnicodeString(" " ) + fieldName(e) + UnicodeString("++") );
|
|
temp->roll(e, 1, status);
|
|
if (U_FAILURE(status)) {
|
|
logln("caltest.cpp:%d e=%d, i=%d - roll(+) err %s\n", __LINE__, static_cast<int>(e), static_cast<int>(i), u_errorName(status));
|
|
logln(calToStr(*temp));
|
|
limit = i; status = U_ZERO_ERROR;
|
|
}
|
|
}
|
|
for (i = 0; i < limit; i++) {
|
|
logln("caltest.cpp:%d e=%d, i=%d\n", __LINE__, static_cast<int>(e), static_cast<int>(i));
|
|
logln(calToStr(*temp) + UnicodeString(" " ) + fieldName(e) + UnicodeString("--") );
|
|
temp->roll(e, -1, status);
|
|
if (U_FAILURE(status)) { errln(UnicodeString("GregorianCalendar::roll ") + CalendarTest::fieldName(e) + " count=" + UnicodeString('@'+i) + " by -1 failed with " + u_errorName(status) ); return; }
|
|
}
|
|
check520(temp, y, m, d, hr, min, sec, ms, e);
|
|
|
|
e = static_cast<UCalendarDateFields>(static_cast<int32_t>(e) + 1);
|
|
}
|
|
|
|
delete temp;
|
|
}
|
|
|
|
// -------------------------------------
|
|
void
|
|
CalendarTest::check520(Calendar* c,
|
|
int32_t y, int32_t m, int32_t d,
|
|
int32_t hr, int32_t min, int32_t sec,
|
|
int32_t ms, UCalendarDateFields field)
|
|
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
if (c->get(UCAL_YEAR, status) != y ||
|
|
c->get(UCAL_MONTH, status) != m ||
|
|
c->get(UCAL_DATE, status) != d ||
|
|
c->get(UCAL_HOUR, status) != hr ||
|
|
c->get(UCAL_MINUTE, status) != min ||
|
|
c->get(UCAL_SECOND, status) != sec ||
|
|
c->get(UCAL_MILLISECOND, status) != ms) {
|
|
errln(UnicodeString("U_FAILURE for field ") + static_cast<int32_t>(field) +
|
|
": Expected y/m/d h:m:s:ms of " +
|
|
y + "/" + (m + 1) + "/" + d + " " +
|
|
hr + ":" + min + ":" + sec + ":" + ms +
|
|
"; got " + c->get(UCAL_YEAR, status) +
|
|
"/" + (c->get(UCAL_MONTH, status) + 1) +
|
|
"/" + c->get(UCAL_DATE, status) +
|
|
" " + c->get(UCAL_HOUR, status) + ":" +
|
|
c->get(UCAL_MINUTE, status) + ":" +
|
|
c->get(UCAL_SECOND, status) + ":" +
|
|
c->get(UCAL_MILLISECOND, status)
|
|
);
|
|
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
else
|
|
logln(UnicodeString("Confirmed: ") + y + "/" +
|
|
(m + 1) + "/" + d + " " +
|
|
hr + ":" + min + ":" + sec + ":" + ms);
|
|
}
|
|
|
|
// -------------------------------------
|
|
void
|
|
CalendarTest::check520(Calendar* c,
|
|
int32_t y, int32_t m, int32_t d)
|
|
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
if (c->get(UCAL_YEAR, status) != y ||
|
|
c->get(UCAL_MONTH, status) != m ||
|
|
c->get(UCAL_DATE, status) != d) {
|
|
errln(UnicodeString("FAILURE: Expected y/m/d of ") +
|
|
y + "/" + (m + 1) + "/" + d + " " +
|
|
"; got " + c->get(UCAL_YEAR, status) +
|
|
"/" + (c->get(UCAL_MONTH, status) + 1) +
|
|
"/" + c->get(UCAL_DATE, status)
|
|
);
|
|
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
else
|
|
logln(UnicodeString("Confirmed: ") + y + "/" +
|
|
(m + 1) + "/" + d);
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Test that setting of fields works. In particular, make sure that all instances
|
|
* of GregorianCalendar don't share a static instance of the fields array.
|
|
*/
|
|
void
|
|
CalendarTest::TestFieldSet4781()
|
|
{
|
|
// try {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar *g = new GregorianCalendar(status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
GregorianCalendar *g2 = new GregorianCalendar(status);
|
|
if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
|
|
g2->set(UCAL_HOUR, 12, status);
|
|
g2->set(UCAL_MINUTE, 0, status);
|
|
g2->set(UCAL_SECOND, 0, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::set failed"); return; }
|
|
if (*g == *g2) logln("Same");
|
|
else logln("Different");
|
|
//}
|
|
//catch(IllegalArgumentException e) {
|
|
//errln("Unexpected exception seen: " + e);
|
|
//}
|
|
delete g;
|
|
delete g2;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/* We don't support serialization on C++
|
|
void
|
|
CalendarTest::TestSerialize337()
|
|
{
|
|
Calendar cal = Calendar::getInstance();
|
|
UBool ok = false;
|
|
try {
|
|
FileOutputStream f = new FileOutputStream(FILENAME);
|
|
ObjectOutput s = new ObjectOutputStream(f);
|
|
s.writeObject(PREFIX);
|
|
s.writeObject(cal);
|
|
s.writeObject(POSTFIX);
|
|
f.close();
|
|
FileInputStream in = new FileInputStream(FILENAME);
|
|
ObjectInputStream t = new ObjectInputStream(in);
|
|
UnicodeString& pre = (UnicodeString&) t.readObject();
|
|
Calendar c = (Calendar) t.readObject();
|
|
UnicodeString& post = (UnicodeString&) t.readObject();
|
|
in.close();
|
|
ok = pre.equals(PREFIX) &&
|
|
post.equals(POSTFIX) &&
|
|
cal->equals(c);
|
|
File fl = new File(FILENAME);
|
|
fl.delete();
|
|
}
|
|
catch(IOException e) {
|
|
errln("FAIL: Exception received:");
|
|
e.printStackTrace(log);
|
|
}
|
|
catch(ClassNotFoundException e) {
|
|
errln("FAIL: Exception received:");
|
|
e.printStackTrace(log);
|
|
}
|
|
if (!ok) errln("Serialization of Calendar object failed.");
|
|
}
|
|
|
|
UnicodeString& CalendarTest::PREFIX = "abc";
|
|
|
|
UnicodeString& CalendarTest::POSTFIX = "def";
|
|
|
|
UnicodeString& CalendarTest::FILENAME = "tmp337.bin";
|
|
*/
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Verify that the seconds of a Calendar can be zeroed out through the
|
|
* expected sequence of operations.
|
|
*/
|
|
void
|
|
CalendarTest::TestSecondsZero121()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *cal = new GregorianCalendar(status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
cal->setTime(Calendar::getNow(), status);
|
|
if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
|
|
cal->set(UCAL_SECOND, 0);
|
|
if (U_FAILURE(status)) { errln("Calendar::set failed"); return; }
|
|
UDate d = cal->getTime(status);
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
UnicodeString s;
|
|
dateToString(d, s);
|
|
if (s.indexOf("DATE_FORMAT_FAILURE") >= 0) {
|
|
dataerrln("Got: \"DATE_FORMAT_FAILURE\".");
|
|
} else if (s.indexOf(":00 ") < 0) {
|
|
errln("Expected to see :00 in " + s);
|
|
}
|
|
delete cal;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Verify that a specific sequence of adding and setting works as expected;
|
|
* it should not vary depending on when and whether the get method is
|
|
* called.
|
|
*/
|
|
void
|
|
CalendarTest::TestAddSetGet0610()
|
|
{
|
|
UnicodeString EXPECTED_0610("1993/0/5", "");
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
{
|
|
Calendar *calendar = new GregorianCalendar(status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
calendar->set(1993, UCAL_JANUARY, 4);
|
|
logln("1A) " + value(calendar));
|
|
calendar->add(UCAL_DATE, 1, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
UnicodeString v = value(calendar);
|
|
logln("1B) " + v);
|
|
logln("--) 1993/0/5");
|
|
if (!(v == EXPECTED_0610)) errln("Expected " + EXPECTED_0610 + "; saw " + v);
|
|
delete calendar;
|
|
}
|
|
{
|
|
Calendar *calendar = new GregorianCalendar(1993, UCAL_JANUARY, 4, status);
|
|
if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
|
|
logln("2A) " + value(calendar));
|
|
calendar->add(UCAL_DATE, 1, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
UnicodeString v = value(calendar);
|
|
logln("2B) " + v);
|
|
logln("--) 1993/0/5");
|
|
if (!(v == EXPECTED_0610)) errln("Expected " + EXPECTED_0610 + "; saw " + v);
|
|
delete calendar;
|
|
}
|
|
{
|
|
Calendar *calendar = new GregorianCalendar(1993, UCAL_JANUARY, 4, status);
|
|
if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
|
|
logln("3A) " + value(calendar));
|
|
calendar->getTime(status);
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
calendar->add(UCAL_DATE, 1, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
UnicodeString v = value(calendar);
|
|
logln("3B) " + v);
|
|
logln("--) 1993/0/5");
|
|
if (!(v == EXPECTED_0610)) errln("Expected " + EXPECTED_0610 + "; saw " + v);
|
|
delete calendar;
|
|
}
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
UnicodeString
|
|
CalendarTest::value(Calendar* calendar)
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
return UnicodeString("") + calendar->get(UCAL_YEAR, status) +
|
|
"/" + calendar->get(UCAL_MONTH, status) +
|
|
"/" + calendar->get(UCAL_DATE, status) +
|
|
(U_FAILURE(status) ? " FAIL: Calendar::get failed" : "");
|
|
}
|
|
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Verify that various fields on a known date are set correctly.
|
|
*/
|
|
void
|
|
CalendarTest::TestFields060()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
int32_t year = 1997;
|
|
int32_t month = UCAL_OCTOBER;
|
|
int32_t dDate = 22;
|
|
GregorianCalendar* calendar = nullptr;
|
|
calendar = new GregorianCalendar(year, month, dDate, status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
for (int32_t i = 0; i < EXPECTED_FIELDS_length;) {
|
|
UCalendarDateFields field = static_cast<UCalendarDateFields>(EXPECTED_FIELDS[i++]);
|
|
int32_t expected = EXPECTED_FIELDS[i++];
|
|
if (calendar->get(field, status) != expected) {
|
|
errln(UnicodeString("Expected field ") + static_cast<int32_t>(field) + " to have value " + expected +
|
|
"; received " + calendar->get(field, status) + " instead");
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
}
|
|
delete calendar;
|
|
}
|
|
|
|
int32_t CalendarTest::EXPECTED_FIELDS[] = {
|
|
UCAL_YEAR, 1997,
|
|
UCAL_MONTH, UCAL_OCTOBER,
|
|
UCAL_DATE, 22,
|
|
UCAL_DAY_OF_WEEK, UCAL_WEDNESDAY,
|
|
UCAL_DAY_OF_WEEK_IN_MONTH, 4,
|
|
UCAL_DAY_OF_YEAR, 295
|
|
};
|
|
|
|
const int32_t CalendarTest::EXPECTED_FIELDS_length = static_cast<int32_t>(sizeof(CalendarTest::EXPECTED_FIELDS) /
|
|
sizeof(CalendarTest::EXPECTED_FIELDS[0]));
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Verify that various fields on a known date are set correctly. In this
|
|
* case, the start of the epoch (January 1 1970).
|
|
*/
|
|
void
|
|
CalendarTest::TestEpochStartFields()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
TimeZone *z = TimeZone::createDefault();
|
|
Calendar *c = Calendar::createInstance(status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
UDate d = - z->getRawOffset();
|
|
GregorianCalendar *gc = new GregorianCalendar(status);
|
|
if (U_FAILURE(status)) { errln("Couldn't create GregorianCalendar"); return; }
|
|
gc->setTimeZone(*z);
|
|
gc->setTime(d, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
|
|
UBool idt = gc->inDaylightTime(status);
|
|
if (U_FAILURE(status)) { errln("GregorianCalendar::inDaylightTime failed"); return; }
|
|
if (idt) {
|
|
UnicodeString str;
|
|
logln("Warning: Skipping test because " + dateToString(d, str) + " is in DST.");
|
|
}
|
|
else {
|
|
c->setTime(d, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::setTime failed"); return; }
|
|
for (int32_t i = 0; i < UCAL_ZONE_OFFSET;++i) {
|
|
if (c->get(static_cast<UCalendarDateFields>(i), status) != EPOCH_FIELDS[i])
|
|
dataerrln(UnicodeString("Expected field ") + i + " to have value " + EPOCH_FIELDS[i] +
|
|
"; saw " + c->get(static_cast<UCalendarDateFields>(i), status) + " instead");
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
if (c->get(UCAL_ZONE_OFFSET, status) != z->getRawOffset())
|
|
{
|
|
errln(UnicodeString("Expected field ZONE_OFFSET to have value ") + z->getRawOffset() +
|
|
"; saw " + c->get(UCAL_ZONE_OFFSET, status) + " instead");
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
if (c->get(UCAL_DST_OFFSET, status) != 0)
|
|
{
|
|
errln(UnicodeString("Expected field DST_OFFSET to have value 0") +
|
|
"; saw " + c->get(UCAL_DST_OFFSET, status) + " instead");
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
}
|
|
}
|
|
delete c;
|
|
delete z;
|
|
delete gc;
|
|
}
|
|
|
|
int32_t CalendarTest::EPOCH_FIELDS[] = {
|
|
1, 1970, 0, 1, 1, 1, 1, 5, 1, 0, 0, 0, 0, 0, 0, - 28800000, 0
|
|
};
|
|
|
|
// -------------------------------------
|
|
|
|
/**
|
|
* Test that the days of the week progress properly when add is called repeatedly
|
|
* for increments of 24 days.
|
|
*/
|
|
void
|
|
CalendarTest::TestDOWProgression()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *cal = new GregorianCalendar(1972, UCAL_OCTOBER, 26, status);
|
|
if (failure(status, "new GregorianCalendar", true)) return;
|
|
marchByDelta(cal, 24);
|
|
delete cal;
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
void
|
|
CalendarTest::TestDOW_LOCALandYEAR_WOY()
|
|
{
|
|
/* Note: I've commented out the loop_addroll tests for YEAR and
|
|
* YEAR_WOY below because these two fields should NOT behave
|
|
* identically when adding. YEAR should keep the month/dom
|
|
* invariant. YEAR_WOY should keep the woy/dow invariant. I've
|
|
* added a new test that checks for this in place of the old call
|
|
* to loop_addroll. - aliu */
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
int32_t times = 20;
|
|
Calendar *cal=Calendar::createInstance(Locale::getGermany(), status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
SimpleDateFormat *sdf=new SimpleDateFormat(UnicodeString("YYYY'-W'ww-ee"), Locale::getGermany(), status);
|
|
if (U_FAILURE(status)) { dataerrln("Couldn't create SimpleDateFormat - %s", u_errorName(status)); return; }
|
|
|
|
// ICU no longer use localized date-time pattern characters by default.
|
|
// So we set pattern chars using 'J' instead of 'Y'.
|
|
DateFormatSymbols *dfs = new DateFormatSymbols(Locale::getGermany(), status);
|
|
dfs->setLocalPatternChars(UnicodeString("GyMdkHmsSEDFwWahKzJeugAZvcLQq"));
|
|
sdf->adoptDateFormatSymbols(dfs);
|
|
sdf->applyLocalizedPattern(UnicodeString("JJJJ'-W'ww-ee"), status);
|
|
if (U_FAILURE(status)) { errln("Couldn't apply localized pattern"); return; }
|
|
|
|
cal->clear();
|
|
cal->set(1997, UCAL_DECEMBER, 25);
|
|
doYEAR_WOYLoop(cal, sdf, times, status);
|
|
//loop_addroll(cal, /*sdf,*/ times, UCAL_YEAR_WOY, UCAL_YEAR, status);
|
|
yearAddTest(*cal, status); // aliu
|
|
loop_addroll(cal, /*sdf,*/ times, UCAL_DOW_LOCAL, UCAL_DAY_OF_WEEK, status);
|
|
if (U_FAILURE(status)) { errln("Error in parse/calculate test for 1997"); return; }
|
|
|
|
cal->clear();
|
|
cal->set(1998, UCAL_DECEMBER, 25);
|
|
doYEAR_WOYLoop(cal, sdf, times, status);
|
|
//loop_addroll(cal, /*sdf,*/ times, UCAL_YEAR_WOY, UCAL_YEAR, status);
|
|
yearAddTest(*cal, status); // aliu
|
|
loop_addroll(cal, /*sdf,*/ times, UCAL_DOW_LOCAL, UCAL_DAY_OF_WEEK, status);
|
|
if (U_FAILURE(status)) { errln("Error in parse/calculate test for 1998"); return; }
|
|
|
|
cal->clear();
|
|
cal->set(1582, UCAL_OCTOBER, 1);
|
|
doYEAR_WOYLoop(cal, sdf, times, status);
|
|
//loop_addroll(cal, /*sdf,*/ times, Calendar::YEAR_WOY, Calendar::YEAR, status);
|
|
yearAddTest(*cal, status); // aliu
|
|
loop_addroll(cal, /*sdf,*/ times, UCAL_DOW_LOCAL, UCAL_DAY_OF_WEEK, status);
|
|
if (U_FAILURE(status)) { errln("Error in parse/calculate test for 1582"); return; }
|
|
delete sdf;
|
|
delete cal;
|
|
}
|
|
|
|
/**
|
|
* Confirm that adding a YEAR and adding a YEAR_WOY work properly for
|
|
* the given Calendar at its current setting.
|
|
*/
|
|
void CalendarTest::yearAddTest(Calendar& cal, UErrorCode& status) {
|
|
/**
|
|
* When adding the YEAR, the month and day should remain constant.
|
|
* When adding the YEAR_WOY, the WOY and DOW should remain constant. - aliu
|
|
* Examples:
|
|
* Wed Jan 14 1998 / 1998-W03-03 Add(YEAR_WOY, 1) -> Wed Jan 20 1999 / 1999-W03-03
|
|
* Add(YEAR, 1) -> Thu Jan 14 1999 / 1999-W02-04
|
|
* Thu Jan 14 1999 / 1999-W02-04 Add(YEAR_WOY, 1) -> Thu Jan 13 2000 / 2000-W02-04
|
|
* Add(YEAR, 1) -> Fri Jan 14 2000 / 2000-W02-05
|
|
* Sun Oct 31 1582 / 1582-W42-07 Add(YEAR_WOY, 1) -> Sun Oct 23 1583 / 1583-W42-07
|
|
* Add(YEAR, 1) -> Mon Oct 31 1583 / 1583-W44-01
|
|
*/
|
|
int32_t y = cal.get(UCAL_YEAR, status);
|
|
int32_t mon = cal.get(UCAL_MONTH, status);
|
|
int32_t day = cal.get(UCAL_DATE, status);
|
|
int32_t ywy = cal.get(UCAL_YEAR_WOY, status);
|
|
int32_t woy = cal.get(UCAL_WEEK_OF_YEAR, status);
|
|
int32_t dow = cal.get(UCAL_DOW_LOCAL, status);
|
|
UDate t = cal.getTime(status);
|
|
|
|
if(U_FAILURE(status)){
|
|
errln(UnicodeString("Failed to create Calendar for locale. Error: ") + UnicodeString(u_errorName(status)));
|
|
return;
|
|
}
|
|
UnicodeString str, str2;
|
|
SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy / YYYY'-W'ww-ee"), status);
|
|
fmt.setCalendar(cal);
|
|
|
|
fmt.format(t, str.remove());
|
|
str += ".add(YEAR, 1) =>";
|
|
cal.add(UCAL_YEAR, 1, status);
|
|
int32_t y2 = cal.get(UCAL_YEAR, status);
|
|
int32_t mon2 = cal.get(UCAL_MONTH, status);
|
|
int32_t day2 = cal.get(UCAL_DATE, status);
|
|
fmt.format(cal.getTime(status), str);
|
|
if (y2 != (y+1) || mon2 != mon || day2 != day) {
|
|
str += UnicodeString(", expected year ") +
|
|
(y+1) + ", month " + (mon+1) + ", day " + day;
|
|
errln(UnicodeString("FAIL: ") + str);
|
|
logln( UnicodeString(" -> ") + CalendarTest::calToStr(cal) );
|
|
} else {
|
|
logln(str);
|
|
}
|
|
|
|
fmt.format(t, str.remove());
|
|
str += ".add(YEAR_WOY, 1)=>";
|
|
cal.setTime(t, status);
|
|
logln( UnicodeString(" <- ") + CalendarTest::calToStr(cal) );
|
|
cal.add(UCAL_YEAR_WOY, 1, status);
|
|
int32_t ywy2 = cal.get(UCAL_YEAR_WOY, status);
|
|
int32_t woy2 = cal.get(UCAL_WEEK_OF_YEAR, status);
|
|
int32_t dow2 = cal.get(UCAL_DOW_LOCAL, status);
|
|
fmt.format(cal.getTime(status), str);
|
|
if (ywy2 != (ywy+1) || woy2 != woy || dow2 != dow) {
|
|
str += UnicodeString(", expected yearWOY ") +
|
|
(ywy+1) + ", woy " + woy + ", dowLocal " + dow;
|
|
errln(UnicodeString("FAIL: ") + str);
|
|
logln( UnicodeString(" -> ") + CalendarTest::calToStr(cal) );
|
|
} else {
|
|
logln(str);
|
|
}
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
void CalendarTest::loop_addroll(Calendar *cal, /*SimpleDateFormat *sdf,*/ int times, UCalendarDateFields field, UCalendarDateFields field2, UErrorCode& errorCode) {
|
|
Calendar *calclone;
|
|
SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy / YYYY'-W'ww-ee"), errorCode);
|
|
fmt.setCalendar(*cal);
|
|
int i;
|
|
|
|
for(i = 0; i<times; i++) {
|
|
calclone = cal->clone();
|
|
UDate start = cal->getTime(errorCode);
|
|
cal->add(field,1,errorCode);
|
|
if (U_FAILURE(errorCode)) { errln("Error in add"); delete calclone; return; }
|
|
calclone->add(field2,1,errorCode);
|
|
if (U_FAILURE(errorCode)) { errln("Error in add"); delete calclone; return; }
|
|
if(cal->getTime(errorCode) != calclone->getTime(errorCode)) {
|
|
UnicodeString str("FAIL: Results of add differ. "), str2;
|
|
str += fmt.format(start, str2) + " ";
|
|
str += UnicodeString("Add(") + fieldName(field) + ", 1) -> " +
|
|
fmt.format(cal->getTime(errorCode), str2.remove()) + "; ";
|
|
str += UnicodeString("Add(") + fieldName(field2) + ", 1) -> " +
|
|
fmt.format(calclone->getTime(errorCode), str2.remove());
|
|
errln(str);
|
|
delete calclone;
|
|
return;
|
|
}
|
|
delete calclone;
|
|
}
|
|
|
|
for(i = 0; i<times; i++) {
|
|
calclone = cal->clone();
|
|
cal->roll(field, static_cast<int32_t>(1), errorCode);
|
|
if (U_FAILURE(errorCode)) { errln("Error in roll"); delete calclone; return; }
|
|
calclone->roll(field2, static_cast<int32_t>(1), errorCode);
|
|
if (U_FAILURE(errorCode)) { errln("Error in roll"); delete calclone; return; }
|
|
if(cal->getTime(errorCode) != calclone->getTime(errorCode)) {
|
|
delete calclone;
|
|
errln("Results of roll differ!");
|
|
return;
|
|
}
|
|
delete calclone;
|
|
}
|
|
}
|
|
|
|
// -------------------------------------
|
|
|
|
void
|
|
CalendarTest::doYEAR_WOYLoop(Calendar *cal, SimpleDateFormat *sdf,
|
|
int32_t times, UErrorCode& errorCode) {
|
|
|
|
UnicodeString us;
|
|
UDate tst, original;
|
|
Calendar *tstres = new GregorianCalendar(Locale::getGermany(), errorCode);
|
|
for(int i=0; i<times; ++i) {
|
|
sdf->format(Formattable(cal->getTime(errorCode),Formattable::kIsDate), us, errorCode);
|
|
//logln("expected: "+us);
|
|
if (U_FAILURE(errorCode)) { errln("Format error"); return; }
|
|
tst=sdf->parse(us,errorCode);
|
|
if (U_FAILURE(errorCode)) { errln("Parse error"); return; }
|
|
tstres->clear();
|
|
tstres->setTime(tst, errorCode);
|
|
//logln((UnicodeString)"Parsed week of year is "+tstres->get(UCAL_WEEK_OF_YEAR, errorCode));
|
|
if (U_FAILURE(errorCode)) { errln("Set time error"); return; }
|
|
original = cal->getTime(errorCode);
|
|
us.remove();
|
|
sdf->format(Formattable(tst,Formattable::kIsDate), us, errorCode);
|
|
//logln("got: "+us);
|
|
if (U_FAILURE(errorCode)) { errln("Get time error"); return; }
|
|
if(original!=tst) {
|
|
us.remove();
|
|
sdf->format(Formattable(original, Formattable::kIsDate), us, errorCode);
|
|
errln("FAIL: Parsed time doesn't match with regular");
|
|
logln("expected "+us + " " + calToStr(*cal));
|
|
us.remove();
|
|
sdf->format(Formattable(tst, Formattable::kIsDate), us, errorCode);
|
|
logln("got "+us + " " + calToStr(*tstres));
|
|
}
|
|
tstres->clear();
|
|
tstres->set(UCAL_YEAR_WOY, cal->get(UCAL_YEAR_WOY, errorCode));
|
|
tstres->set(UCAL_WEEK_OF_YEAR, cal->get(UCAL_WEEK_OF_YEAR, errorCode));
|
|
tstres->set(UCAL_DOW_LOCAL, cal->get(UCAL_DOW_LOCAL, errorCode));
|
|
if(cal->get(UCAL_YEAR, errorCode) != tstres->get(UCAL_YEAR, errorCode)) {
|
|
errln("FAIL: Different Year!");
|
|
logln(UnicodeString("Expected ") + cal->get(UCAL_YEAR, errorCode));
|
|
logln(UnicodeString("Got ") + tstres->get(UCAL_YEAR, errorCode));
|
|
return;
|
|
}
|
|
if(cal->get(UCAL_DAY_OF_YEAR, errorCode) != tstres->get(UCAL_DAY_OF_YEAR, errorCode)) {
|
|
errln("FAIL: Different Day Of Year!");
|
|
logln(UnicodeString("Expected ") + cal->get(UCAL_DAY_OF_YEAR, errorCode));
|
|
logln(UnicodeString("Got ") + tstres->get(UCAL_DAY_OF_YEAR, errorCode));
|
|
return;
|
|
}
|
|
//logln(calToStr(*cal));
|
|
cal->add(UCAL_DATE, 1, errorCode);
|
|
if (U_FAILURE(errorCode)) { errln("Add error"); return; }
|
|
us.remove();
|
|
}
|
|
delete (tstres);
|
|
}
|
|
// -------------------------------------
|
|
|
|
void
|
|
CalendarTest::marchByDelta(Calendar* cal, int32_t delta)
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *cur = cal->clone();
|
|
int32_t initialDOW = cur->get(UCAL_DAY_OF_WEEK, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
int32_t DOW, newDOW = initialDOW;
|
|
do {
|
|
UnicodeString str;
|
|
DOW = newDOW;
|
|
logln(UnicodeString("DOW = ") + DOW + " " + dateToString(cur->getTime(status), str));
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
cur->add(UCAL_DAY_OF_WEEK, delta, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::add failed"); return; }
|
|
newDOW = cur->get(UCAL_DAY_OF_WEEK, status);
|
|
if (U_FAILURE(status)) { errln("Calendar::get failed"); return; }
|
|
int32_t expectedDOW = 1 + (DOW + delta - 1) % 7;
|
|
if (newDOW != expectedDOW) {
|
|
errln(UnicodeString("Day of week should be ") + expectedDOW + " instead of " + newDOW +
|
|
" on " + dateToString(cur->getTime(status), str));
|
|
if (U_FAILURE(status)) { errln("Calendar::getTime failed"); return; }
|
|
return;
|
|
}
|
|
}
|
|
while (newDOW != initialDOW);
|
|
delete cur;
|
|
}
|
|
|
|
#define CHECK(status, msg) UPRV_BLOCK_MACRO_BEGIN { \
|
|
if (U_FAILURE(status)) { \
|
|
errcheckln(status, msg); \
|
|
return; \
|
|
} \
|
|
} UPRV_BLOCK_MACRO_END
|
|
|
|
void CalendarTest::TestWOY() {
|
|
/*
|
|
FDW = Mon, MDFW = 4:
|
|
Sun Dec 26 1999, WOY 51
|
|
Mon Dec 27 1999, WOY 52
|
|
Tue Dec 28 1999, WOY 52
|
|
Wed Dec 29 1999, WOY 52
|
|
Thu Dec 30 1999, WOY 52
|
|
Fri Dec 31 1999, WOY 52
|
|
Sat Jan 01 2000, WOY 52 ***
|
|
Sun Jan 02 2000, WOY 52 ***
|
|
Mon Jan 03 2000, WOY 1
|
|
Tue Jan 04 2000, WOY 1
|
|
Wed Jan 05 2000, WOY 1
|
|
Thu Jan 06 2000, WOY 1
|
|
Fri Jan 07 2000, WOY 1
|
|
Sat Jan 08 2000, WOY 1
|
|
Sun Jan 09 2000, WOY 1
|
|
Mon Jan 10 2000, WOY 2
|
|
|
|
FDW = Mon, MDFW = 2:
|
|
Sun Dec 26 1999, WOY 52
|
|
Mon Dec 27 1999, WOY 1 ***
|
|
Tue Dec 28 1999, WOY 1 ***
|
|
Wed Dec 29 1999, WOY 1 ***
|
|
Thu Dec 30 1999, WOY 1 ***
|
|
Fri Dec 31 1999, WOY 1 ***
|
|
Sat Jan 01 2000, WOY 1
|
|
Sun Jan 02 2000, WOY 1
|
|
Mon Jan 03 2000, WOY 2
|
|
Tue Jan 04 2000, WOY 2
|
|
Wed Jan 05 2000, WOY 2
|
|
Thu Jan 06 2000, WOY 2
|
|
Fri Jan 07 2000, WOY 2
|
|
Sat Jan 08 2000, WOY 2
|
|
Sun Jan 09 2000, WOY 2
|
|
Mon Jan 10 2000, WOY 3
|
|
*/
|
|
|
|
UnicodeString str;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
int32_t i;
|
|
|
|
GregorianCalendar cal(status);
|
|
SimpleDateFormat fmt(UnicodeString("EEE MMM dd yyyy', WOY' w"), status);
|
|
if (failure(status, "Cannot construct calendar/format", true)) return;
|
|
|
|
UCalendarDaysOfWeek fdw = static_cast<UCalendarDaysOfWeek>(0);
|
|
|
|
//for (int8_t pass=2; pass<=2; ++pass) {
|
|
for (int8_t pass=1; pass<=2; ++pass) {
|
|
switch (pass) {
|
|
case 1:
|
|
fdw = UCAL_MONDAY;
|
|
cal.setFirstDayOfWeek(fdw);
|
|
cal.setMinimalDaysInFirstWeek(4);
|
|
fmt.adoptCalendar(cal.clone());
|
|
break;
|
|
case 2:
|
|
fdw = UCAL_MONDAY;
|
|
cal.setFirstDayOfWeek(fdw);
|
|
cal.setMinimalDaysInFirstWeek(2);
|
|
fmt.adoptCalendar(cal.clone());
|
|
break;
|
|
}
|
|
|
|
//for (i=2; i<=6; ++i) {
|
|
for (i=0; i<16; ++i) {
|
|
UDate t, t2;
|
|
int32_t t_y, t_woy, t_dow;
|
|
cal.clear();
|
|
cal.set(1999, UCAL_DECEMBER, 26 + i);
|
|
fmt.format(t = cal.getTime(status), str.remove());
|
|
CHECK(status, "Fail: getTime failed");
|
|
logln(UnicodeString("* ") + str);
|
|
int32_t dow = cal.get(UCAL_DAY_OF_WEEK, status);
|
|
int32_t woy = cal.get(UCAL_WEEK_OF_YEAR, status);
|
|
int32_t year = cal.get(UCAL_YEAR, status);
|
|
int32_t mon = cal.get(UCAL_MONTH, status);
|
|
logln(calToStr(cal));
|
|
CHECK(status, "Fail: get failed");
|
|
int32_t dowLocal = dow - fdw;
|
|
if (dowLocal < 0) dowLocal += 7;
|
|
dowLocal++;
|
|
int32_t yearWoy = year;
|
|
if (mon == UCAL_JANUARY) {
|
|
if (woy >= 52) --yearWoy;
|
|
} else {
|
|
if (woy == 1) ++yearWoy;
|
|
}
|
|
|
|
// Basic fields->time check y/woy/dow
|
|
// Since Y/WOY is ambiguous, we do a check of the fields,
|
|
// not of the specific time.
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR, year);
|
|
cal.set(UCAL_WEEK_OF_YEAR, woy);
|
|
cal.set(UCAL_DAY_OF_WEEK, dow);
|
|
t_y = cal.get(UCAL_YEAR, status);
|
|
t_woy = cal.get(UCAL_WEEK_OF_YEAR, status);
|
|
t_dow = cal.get(UCAL_DAY_OF_WEEK, status);
|
|
CHECK(status, "Fail: get failed");
|
|
if (t_y != year || t_woy != woy || t_dow != dow) {
|
|
str = "Fail: y/woy/dow fields->time => ";
|
|
fmt.format(cal.getTime(status), str);
|
|
errln(str);
|
|
logln(calToStr(cal));
|
|
logln("[get!=set] Y%d!=%d || woy%d!=%d || dow%d!=%d\n",
|
|
t_y, year, t_woy, woy, t_dow, dow);
|
|
} else {
|
|
logln("y/woy/dow fields->time OK");
|
|
}
|
|
|
|
// Basic fields->time check y/woy/dow_local
|
|
// Since Y/WOY is ambiguous, we do a check of the fields,
|
|
// not of the specific time.
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR, year);
|
|
cal.set(UCAL_WEEK_OF_YEAR, woy);
|
|
cal.set(UCAL_DOW_LOCAL, dowLocal);
|
|
t_y = cal.get(UCAL_YEAR, status);
|
|
t_woy = cal.get(UCAL_WEEK_OF_YEAR, status);
|
|
t_dow = cal.get(UCAL_DOW_LOCAL, status);
|
|
CHECK(status, "Fail: get failed");
|
|
if (t_y != year || t_woy != woy || t_dow != dowLocal) {
|
|
str = "Fail: y/woy/dow_local fields->time => ";
|
|
fmt.format(cal.getTime(status), str);
|
|
errln(str);
|
|
}
|
|
|
|
// Basic fields->time check y_woy/woy/dow
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR_WOY, yearWoy);
|
|
cal.set(UCAL_WEEK_OF_YEAR, woy);
|
|
cal.set(UCAL_DAY_OF_WEEK, dow);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: getTime failed");
|
|
if (t != t2) {
|
|
str = "Fail: y_woy/woy/dow fields->time => ";
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
logln(calToStr(cal));
|
|
logln("%.f != %.f\n", t, t2);
|
|
} else {
|
|
logln("y_woy/woy/dow OK");
|
|
}
|
|
|
|
// Basic fields->time check y_woy/woy/dow_local
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR_WOY, yearWoy);
|
|
cal.set(UCAL_WEEK_OF_YEAR, woy);
|
|
cal.set(UCAL_DOW_LOCAL, dowLocal);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: getTime failed");
|
|
if (t != t2) {
|
|
str = "Fail: y_woy/woy/dow_local fields->time => ";
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
}
|
|
|
|
logln("Testing DOW_LOCAL.. dow%d\n", dow);
|
|
// Make sure DOW_LOCAL disambiguates over DOW
|
|
int32_t wrongDow = dow - 3;
|
|
if (wrongDow < 1) wrongDow += 7;
|
|
cal.setTime(t, status);
|
|
cal.set(UCAL_DAY_OF_WEEK, wrongDow);
|
|
cal.set(UCAL_DOW_LOCAL, dowLocal);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: set/getTime failed");
|
|
if (t != t2) {
|
|
str = "Fail: DOW_LOCAL fields->time => ";
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
logln(calToStr(cal));
|
|
logln("%.f : DOW%d, DOW_LOCAL%d -> %.f\n",
|
|
t, wrongDow, dowLocal, t2);
|
|
}
|
|
|
|
// Make sure DOW disambiguates over DOW_LOCAL
|
|
int32_t wrongDowLocal = dowLocal - 3;
|
|
if (wrongDowLocal < 1) wrongDowLocal += 7;
|
|
cal.setTime(t, status);
|
|
cal.set(UCAL_DOW_LOCAL, wrongDowLocal);
|
|
cal.set(UCAL_DAY_OF_WEEK, dow);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: set/getTime failed");
|
|
if (t != t2) {
|
|
str = "Fail: DOW fields->time => ";
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
}
|
|
|
|
// Make sure YEAR_WOY disambiguates over YEAR
|
|
cal.setTime(t, status);
|
|
cal.set(UCAL_YEAR, year - 2);
|
|
cal.set(UCAL_YEAR_WOY, yearWoy);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: set/getTime failed");
|
|
if (t != t2) {
|
|
str = "Fail: YEAR_WOY fields->time => ";
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
}
|
|
|
|
// Make sure YEAR disambiguates over YEAR_WOY
|
|
cal.setTime(t, status);
|
|
cal.set(UCAL_YEAR_WOY, yearWoy - 2);
|
|
cal.set(UCAL_YEAR, year);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: set/getTime failed");
|
|
if (t != t2) {
|
|
str = "Fail: YEAR fields->time => ";
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
}
|
|
}
|
|
}
|
|
|
|
/*
|
|
FDW = Mon, MDFW = 4:
|
|
Sun Dec 26 1999, WOY 51
|
|
Mon Dec 27 1999, WOY 52
|
|
Tue Dec 28 1999, WOY 52
|
|
Wed Dec 29 1999, WOY 52
|
|
Thu Dec 30 1999, WOY 52
|
|
Fri Dec 31 1999, WOY 52
|
|
Sat Jan 01 2000, WOY 52
|
|
Sun Jan 02 2000, WOY 52
|
|
*/
|
|
|
|
// Roll the DOW_LOCAL within week 52
|
|
for (i=27; i<=33; ++i) {
|
|
int32_t amount;
|
|
for (amount=-7; amount<=7; ++amount) {
|
|
str = "roll(";
|
|
cal.set(1999, UCAL_DECEMBER, i);
|
|
UDate t, t2;
|
|
fmt.format(cal.getTime(status), str);
|
|
CHECK(status, "Fail: getTime failed");
|
|
str += UnicodeString(", ") + amount + ") = ";
|
|
|
|
cal.roll(UCAL_DOW_LOCAL, amount, status);
|
|
CHECK(status, "Fail: roll failed");
|
|
|
|
t = cal.getTime(status);
|
|
int32_t newDom = i + amount;
|
|
while (newDom < 27) newDom += 7;
|
|
while (newDom > 33) newDom -= 7;
|
|
cal.set(1999, UCAL_DECEMBER, newDom);
|
|
t2 = cal.getTime(status);
|
|
CHECK(status, "Fail: getTime failed");
|
|
fmt.format(t, str);
|
|
|
|
if (t != t2) {
|
|
str.append(", exp ");
|
|
fmt.format(t2, str);
|
|
errln(str);
|
|
} else {
|
|
logln(str);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestYWOY()
|
|
{
|
|
UnicodeString str;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
|
|
GregorianCalendar cal(status);
|
|
if (failure(status, "construct GregorianCalendar", true)) return;
|
|
|
|
cal.setFirstDayOfWeek(UCAL_SUNDAY);
|
|
cal.setMinimalDaysInFirstWeek(1);
|
|
|
|
logln("Setting: ywoy=2004, woy=1, dow=MONDAY");
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR_WOY,2004);
|
|
cal.set(UCAL_WEEK_OF_YEAR,1);
|
|
cal.set(UCAL_DAY_OF_WEEK, UCAL_MONDAY);
|
|
|
|
logln(calToStr(cal));
|
|
if(cal.get(UCAL_YEAR, status) != 2003) {
|
|
errln("year not 2003");
|
|
}
|
|
|
|
logln("+ setting DOW to THURSDAY");
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR_WOY,2004);
|
|
cal.set(UCAL_WEEK_OF_YEAR,1);
|
|
cal.set(UCAL_DAY_OF_WEEK, UCAL_THURSDAY);
|
|
|
|
logln(calToStr(cal));
|
|
if(cal.get(UCAL_YEAR, status) != 2004) {
|
|
errln("year not 2004");
|
|
}
|
|
|
|
logln("+ setting DOW_LOCAL to 1");
|
|
cal.clear();
|
|
cal.set(UCAL_YEAR_WOY,2004);
|
|
cal.set(UCAL_WEEK_OF_YEAR,1);
|
|
cal.set(UCAL_DAY_OF_WEEK, UCAL_THURSDAY);
|
|
cal.set(UCAL_DOW_LOCAL, 1);
|
|
|
|
logln(calToStr(cal));
|
|
if(cal.get(UCAL_YEAR, status) != 2003) {
|
|
errln("year not 2003");
|
|
}
|
|
|
|
cal.setFirstDayOfWeek(UCAL_MONDAY);
|
|
cal.setMinimalDaysInFirstWeek(4);
|
|
UDate t = 946713600000.;
|
|
cal.setTime(t, status);
|
|
cal.set(UCAL_DAY_OF_WEEK, 4);
|
|
cal.set(UCAL_DOW_LOCAL, 6);
|
|
if(cal.getTime(status) != t) {
|
|
logln(calToStr(cal));
|
|
errln("FAIL: DOW_LOCAL did not take precedence");
|
|
}
|
|
|
|
}
|
|
|
|
void CalendarTest::TestJD()
|
|
{
|
|
int32_t jd;
|
|
static const int32_t kEpochStartAsJulianDay = 2440588;
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar cal(status);
|
|
if (failure(status, "construct GregorianCalendar", true)) return;
|
|
cal.setTimeZone(*TimeZone::getGMT());
|
|
cal.clear();
|
|
jd = cal.get(UCAL_JULIAN_DAY, status);
|
|
if(jd != kEpochStartAsJulianDay) {
|
|
errln("Wanted JD of %d at time=0, [epoch 1970] but got %d\n", kEpochStartAsJulianDay, jd);
|
|
} else {
|
|
logln("Wanted JD of %d at time=0, [epoch 1970], got %d\n", kEpochStartAsJulianDay, jd);
|
|
}
|
|
|
|
cal.setTime(Calendar::getNow(), status);
|
|
cal.clear();
|
|
cal.set(UCAL_JULIAN_DAY, kEpochStartAsJulianDay);
|
|
UDate epochTime = cal.getTime(status);
|
|
if(epochTime != 0) {
|
|
errln("Wanted time of 0 at jd=%d, got %.1lf\n", kEpochStartAsJulianDay, epochTime);
|
|
} else {
|
|
logln("Wanted time of 0 at jd=%d, got %.1lf\n", kEpochStartAsJulianDay, epochTime);
|
|
}
|
|
|
|
}
|
|
|
|
// make sure the ctestfw utilities are in sync with the Calendar
|
|
void CalendarTest::TestDebug()
|
|
{
|
|
for(int32_t t=0;t<=UDBG_ENUM_COUNT;t++) {
|
|
int32_t count = udbg_enumCount(static_cast<UDebugEnumType>(t));
|
|
if(count == -1) {
|
|
logln("enumCount(%d) returned -1", count);
|
|
continue;
|
|
}
|
|
for(int32_t i=0;i<=count;i++) {
|
|
if(t<=UDBG_HIGHEST_CONTIGUOUS_ENUM && i<count) {
|
|
if (i != udbg_enumArrayValue(static_cast<UDebugEnumType>(t), i)) {
|
|
errln("FAIL: udbg_enumArrayValue(%d,%d) returned %d, expected %d", t, i, udbg_enumArrayValue(static_cast<UDebugEnumType>(t), i), i);
|
|
}
|
|
} else {
|
|
logln("Testing count+1:");
|
|
}
|
|
const char* name = udbg_enumName(static_cast<UDebugEnumType>(t), i);
|
|
if(name==nullptr) {
|
|
if(i==count || t>UDBG_HIGHEST_CONTIGUOUS_ENUM ) {
|
|
logln(" null name - expected.\n");
|
|
} else {
|
|
errln("FAIL: udbg_enumName(%d,%d) returned nullptr", t, i);
|
|
}
|
|
name = "(null)";
|
|
}
|
|
logln("udbg_enumArrayValue(%d,%d) = %s, returned %d", t, i,
|
|
name, udbg_enumArrayValue(static_cast<UDebugEnumType>(t), i));
|
|
logln("udbg_enumString = " + udbg_enumString(static_cast<UDebugEnumType>(t), i));
|
|
}
|
|
if (udbg_enumExpectedCount(static_cast<UDebugEnumType>(t)) != count && t <= UDBG_HIGHEST_CONTIGUOUS_ENUM) {
|
|
errln("FAIL: udbg_enumExpectedCount(%d): %d, != UCAL_FIELD_COUNT=%d ", t, udbg_enumExpectedCount(static_cast<UDebugEnumType>(t)), count);
|
|
} else {
|
|
logln("udbg_ucal_fieldCount: %d, UCAL_FIELD_COUNT=udbg_enumCount %d ", udbg_enumExpectedCount(static_cast<UDebugEnumType>(t)), count);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
#undef CHECK
|
|
|
|
// List of interesting locales
|
|
const char *CalendarTest::testLocaleID(int32_t i)
|
|
{
|
|
switch(i) {
|
|
case 0: return "he_IL@calendar=hebrew";
|
|
case 1: return "en_US@calendar=hebrew";
|
|
case 2: return "fr_FR@calendar=hebrew";
|
|
case 3: return "fi_FI@calendar=hebrew";
|
|
case 4: return "nl_NL@calendar=hebrew";
|
|
case 5: return "hu_HU@calendar=hebrew";
|
|
case 6: return "nl_BE@currency=MTL;calendar=islamic";
|
|
case 7: return "th_TH_TRADITIONAL@calendar=gregorian";
|
|
case 8: return "ar_JO@calendar=islamic-civil";
|
|
case 9: return "fi_FI@calendar=islamic";
|
|
case 10: return "fr_CH@calendar=islamic-civil";
|
|
case 11: return "he_IL@calendar=islamic-civil";
|
|
case 12: return "hu_HU@calendar=buddhist";
|
|
case 13: return "hu_HU@calendar=islamic";
|
|
case 14: return "en_US@calendar=japanese";
|
|
default: return nullptr;
|
|
}
|
|
}
|
|
|
|
int32_t CalendarTest::testLocaleCount()
|
|
{
|
|
static int32_t gLocaleCount = -1;
|
|
if(gLocaleCount < 0) {
|
|
int32_t i;
|
|
for(i=0;testLocaleID(i) != nullptr;i++) {
|
|
// do nothing
|
|
}
|
|
gLocaleCount = i;
|
|
}
|
|
return gLocaleCount;
|
|
}
|
|
|
|
static UDate doMinDateOfCalendar(Calendar* adopt, UBool &isGregorian, UErrorCode& status) {
|
|
if(U_FAILURE(status)) return 0.0;
|
|
|
|
adopt->clear();
|
|
adopt->set(UCAL_EXTENDED_YEAR, adopt->getActualMinimum(UCAL_EXTENDED_YEAR, status));
|
|
UDate ret = adopt->getTime(status);
|
|
isGregorian = dynamic_cast<GregorianCalendar*>(adopt) != nullptr;
|
|
delete adopt;
|
|
return ret;
|
|
}
|
|
|
|
UDate CalendarTest::minDateOfCalendar(const Locale& locale, UBool &isGregorian, UErrorCode& status) {
|
|
if(U_FAILURE(status)) return 0.0;
|
|
return doMinDateOfCalendar(Calendar::createInstance(locale, status), isGregorian, status);
|
|
}
|
|
|
|
UDate CalendarTest::minDateOfCalendar(const Calendar& cal, UBool &isGregorian, UErrorCode& status) {
|
|
if(U_FAILURE(status)) return 0.0;
|
|
return doMinDateOfCalendar(cal.clone(), isGregorian, status);
|
|
}
|
|
|
|
void CalendarTest::Test6703()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *cal;
|
|
|
|
Locale loc1("en@calendar=fubar");
|
|
cal = Calendar::createInstance(loc1, status);
|
|
if (failure(status, "Calendar::createInstance", true)) return;
|
|
delete cal;
|
|
|
|
status = U_ZERO_ERROR;
|
|
Locale loc2("en");
|
|
cal = Calendar::createInstance(loc2, status);
|
|
if (failure(status, "Calendar::createInstance")) return;
|
|
delete cal;
|
|
|
|
status = U_ZERO_ERROR;
|
|
Locale loc3("en@calendar=roc");
|
|
cal = Calendar::createInstance(loc3, status);
|
|
if (failure(status, "Calendar::createInstance")) return;
|
|
delete cal;
|
|
}
|
|
|
|
void CalendarTest::Test3785()
|
|
{
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UnicodeString uzone = UNICODE_STRING_SIMPLE("Europe/Paris");
|
|
UnicodeString exp1 = UNICODE_STRING_SIMPLE("Mon 30 Jumada II 1433 AH, 01:47:03");
|
|
UnicodeString exp2 = UNICODE_STRING_SIMPLE("Mon 1 Rajab 1433 AH, 01:47:04");
|
|
|
|
LocalUDateFormatPointer df(udat_open(UDAT_NONE, UDAT_NONE, "en@calendar=islamic", uzone.getTerminatedBuffer(),
|
|
uzone.length(), nullptr, 0, &status));
|
|
if (df.isNull() || U_FAILURE(status)) return;
|
|
|
|
char16_t upattern[64];
|
|
u_uastrcpy(upattern, "EEE d MMMM y G, HH:mm:ss");
|
|
udat_applyPattern(df.getAlias(), false, upattern, u_strlen(upattern));
|
|
|
|
char16_t ubuffer[1024];
|
|
UDate ud0 = 1337557623000.0;
|
|
|
|
status = U_ZERO_ERROR;
|
|
udat_format(df.getAlias(), ud0, ubuffer, 1024, nullptr, &status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Error formatting date 1\n");
|
|
return;
|
|
}
|
|
//printf("formatted: '%s'\n", mkcstr(ubuffer));
|
|
|
|
UnicodeString act1(ubuffer);
|
|
if ( act1 != exp1 ) {
|
|
errln("Unexpected result from date 1 format\n");
|
|
}
|
|
ud0 += 1000.0; // add one second
|
|
|
|
status = U_ZERO_ERROR;
|
|
udat_format(df.getAlias(), ud0, ubuffer, 1024, nullptr, &status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Error formatting date 2\n");
|
|
return;
|
|
}
|
|
//printf("formatted: '%s'\n", mkcstr(ubuffer));
|
|
UnicodeString act2(ubuffer);
|
|
if ( act2 != exp2 ) {
|
|
errln("Unexpected result from date 2 format\n");
|
|
}
|
|
}
|
|
|
|
void CalendarTest::Test1624() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale loc("he_IL@calendar=hebrew");
|
|
HebrewCalendar hc(loc,status);
|
|
|
|
for (int32_t year = 5600; year < 5800; year++ ) {
|
|
|
|
for (int32_t month = HebrewCalendar::TISHRI; month <= HebrewCalendar::ELUL; month++) {
|
|
// skip the adar 1 month if year is not a leap year
|
|
if (HebrewCalendar::isLeapYear(year) == false && month == HebrewCalendar::ADAR_1) {
|
|
continue;
|
|
}
|
|
int32_t day = 15;
|
|
hc.set(year,month,day);
|
|
int32_t dayHC = hc.get(UCAL_DATE,status);
|
|
int32_t monthHC = hc.get(UCAL_MONTH,status);
|
|
int32_t yearHC = hc.get(UCAL_YEAR,status);
|
|
|
|
if (failure(status, "HebrewCalendar.get()", true)) continue;
|
|
|
|
if (dayHC != day) {
|
|
errln(" ==> day %d incorrect, should be: %d\n",dayHC,day);
|
|
break;
|
|
}
|
|
if (monthHC != month) {
|
|
errln(" ==> month %d incorrect, should be: %d\n",monthHC,month);
|
|
break;
|
|
}
|
|
if (yearHC != year) {
|
|
errln(" ==> day %d incorrect, should be: %d\n",yearHC,year);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestTimeStamp() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
UDate start = 0.0, time;
|
|
Calendar *cal;
|
|
|
|
// Create a new Gregorian Calendar.
|
|
cal = Calendar::createInstance("en_US@calendar=gregorian", status);
|
|
if (U_FAILURE(status)) {
|
|
dataerrln("Error creating Gregorian calendar.");
|
|
return;
|
|
}
|
|
|
|
for (int i = 0; i < 20000; i++) {
|
|
// Set the Gregorian Calendar to a specific date for testing.
|
|
cal->set(2009, UCAL_JULY, 3, 0, 49, 46);
|
|
|
|
time = cal->getTime(status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Error calling getTime()");
|
|
break;
|
|
}
|
|
|
|
if (i == 0) {
|
|
start = time;
|
|
} else {
|
|
if (start != time) {
|
|
errln("start and time not equal.");
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
delete cal;
|
|
}
|
|
|
|
void CalendarTest::TestISO8601() {
|
|
const char* TEST_LOCALES[] = {
|
|
"en_US@calendar=iso8601",
|
|
"en_US@calendar=Iso8601",
|
|
"th_TH@calendar=iso8601",
|
|
"ar_EG@calendar=iso8601",
|
|
nullptr
|
|
};
|
|
|
|
int32_t TEST_DATA[][3] = {
|
|
{2008, 1, 2008},
|
|
{2009, 1, 2009},
|
|
{2010, 53, 2009},
|
|
{2011, 52, 2010},
|
|
{2012, 52, 2011},
|
|
{2013, 1, 2013},
|
|
{2014, 1, 2014},
|
|
{0, 0, 0},
|
|
};
|
|
|
|
for (int i = 0; TEST_LOCALES[i] != nullptr; i++) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar *cal = Calendar::createInstance(TEST_LOCALES[i], status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Error: Failed to create a calendar for locale: %s", TEST_LOCALES[i]);
|
|
continue;
|
|
}
|
|
if (uprv_strcmp(cal->getType(), "iso8601") != 0) {
|
|
errln("Error: iso8601 calendar is not used for locale: %s", TEST_LOCALES[i]);
|
|
continue;
|
|
}
|
|
for (int j = 0; TEST_DATA[j][0] != 0; j++) {
|
|
cal->set(TEST_DATA[j][0], UCAL_JANUARY, 1);
|
|
int32_t weekNum = cal->get(UCAL_WEEK_OF_YEAR, status);
|
|
int32_t weekYear = cal->get(UCAL_YEAR_WOY, status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Error: Failed to get week of year");
|
|
break;
|
|
}
|
|
if (weekNum != TEST_DATA[j][1] || weekYear != TEST_DATA[j][2]) {
|
|
errln("Error: Incorrect week of year on January 1st, %d for locale %s: Returned [weekNum=%d, weekYear=%d], Expected [weekNum=%d, weekYear=%d]",
|
|
TEST_DATA[j][0], TEST_LOCALES[i], weekNum, weekYear, TEST_DATA[j][1], TEST_DATA[j][2]);
|
|
}
|
|
}
|
|
delete cal;
|
|
}
|
|
|
|
}
|
|
|
|
void
|
|
CalendarTest::TestAmbiguousWallTimeAPIs() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Calendar* cal = Calendar::createInstance(status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Error creating a calendar instance.");
|
|
return;
|
|
}
|
|
|
|
if (cal->getRepeatedWallTimeOption() != UCAL_WALLTIME_LAST) {
|
|
errln("Fail: Default repeted time option is not UCAL_WALLTIME_LAST");
|
|
}
|
|
if (cal->getSkippedWallTimeOption() != UCAL_WALLTIME_LAST) {
|
|
errln("Fail: Default skipped time option is not UCAL_WALLTIME_LAST");
|
|
}
|
|
|
|
Calendar* cal2 = cal->clone();
|
|
|
|
if (*cal != *cal2) {
|
|
errln("Fail: Cloned calendar != the original");
|
|
}
|
|
if (!cal->equals(*cal2, status)) {
|
|
errln("Fail: The time of cloned calendar is not equal to the original");
|
|
} else if (U_FAILURE(status)) {
|
|
errln("Fail: Error equals");
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
|
|
cal2->setRepeatedWallTimeOption(UCAL_WALLTIME_FIRST);
|
|
cal2->setSkippedWallTimeOption(UCAL_WALLTIME_FIRST);
|
|
|
|
if (*cal == *cal2) {
|
|
errln("Fail: Cloned and modified calendar == the original");
|
|
}
|
|
if (!cal->equals(*cal2, status)) {
|
|
errln("Fail: The time of cloned calendar is not equal to the original after changing wall time options");
|
|
} else if (U_FAILURE(status)) {
|
|
errln("Fail: Error equals after changing wall time options");
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
|
|
if (cal2->getRepeatedWallTimeOption() != UCAL_WALLTIME_FIRST) {
|
|
errln("Fail: Repeted time option is not UCAL_WALLTIME_FIRST");
|
|
}
|
|
if (cal2->getSkippedWallTimeOption() != UCAL_WALLTIME_FIRST) {
|
|
errln("Fail: Skipped time option is not UCAL_WALLTIME_FIRST");
|
|
}
|
|
|
|
cal2->setRepeatedWallTimeOption(UCAL_WALLTIME_NEXT_VALID);
|
|
if (cal2->getRepeatedWallTimeOption() != UCAL_WALLTIME_FIRST) {
|
|
errln("Fail: Repeated wall time option was updated other than UCAL_WALLTIME_FIRST");
|
|
}
|
|
|
|
delete cal;
|
|
delete cal2;
|
|
}
|
|
|
|
class CalFields {
|
|
public:
|
|
CalFields(int32_t year, int32_t month, int32_t day, int32_t hour, int32_t min, int32_t sec, int32_t ms = 0);
|
|
CalFields(const Calendar& cal, UErrorCode& status);
|
|
void setTo(Calendar& cal) const;
|
|
char* toString(char* buf, int32_t len) const;
|
|
bool operator==(const CalFields& rhs) const;
|
|
bool operator!=(const CalFields& rhs) const;
|
|
UBool isEquivalentTo(const Calendar& cal, UErrorCode& status) const;
|
|
|
|
private:
|
|
int32_t year;
|
|
int32_t month;
|
|
int32_t day;
|
|
int32_t hour;
|
|
int32_t min;
|
|
int32_t sec;
|
|
int32_t ms;
|
|
};
|
|
|
|
CalFields::CalFields(int32_t year, int32_t month, int32_t day, int32_t hour, int32_t min, int32_t sec, int32_t ms)
|
|
: year(year), month(month), day(day), hour(hour), min(min), sec(sec), ms(ms) {
|
|
}
|
|
|
|
CalFields::CalFields(const Calendar& cal, UErrorCode& status) {
|
|
year = cal.get(UCAL_YEAR, status);
|
|
month = cal.get(UCAL_MONTH, status) + 1;
|
|
day = cal.get(UCAL_DAY_OF_MONTH, status);
|
|
hour = cal.get(UCAL_HOUR_OF_DAY, status);
|
|
min = cal.get(UCAL_MINUTE, status);
|
|
sec = cal.get(UCAL_SECOND, status);
|
|
ms = cal.get(UCAL_MILLISECOND, status);
|
|
}
|
|
|
|
void
|
|
CalFields::setTo(Calendar& cal) const {
|
|
cal.clear();
|
|
cal.set(year, month - 1, day, hour, min, sec);
|
|
cal.set(UCAL_MILLISECOND, ms);
|
|
}
|
|
|
|
char*
|
|
CalFields::toString(char* buf, int32_t len) const {
|
|
char local[32];
|
|
snprintf(local, sizeof(local), "%04d-%02d-%02d %02d:%02d:%02d.%03d", year, month, day, hour, min, sec, ms);
|
|
uprv_strncpy(buf, local, len - 1);
|
|
buf[len - 1] = 0;
|
|
return buf;
|
|
}
|
|
|
|
bool
|
|
CalFields::operator==(const CalFields& rhs) const {
|
|
return year == rhs.year
|
|
&& month == rhs.month
|
|
&& day == rhs.day
|
|
&& hour == rhs.hour
|
|
&& min == rhs.min
|
|
&& sec == rhs.sec
|
|
&& ms == rhs.ms;
|
|
}
|
|
|
|
bool
|
|
CalFields::operator!=(const CalFields& rhs) const {
|
|
return !(*this == rhs);
|
|
}
|
|
|
|
UBool
|
|
CalFields::isEquivalentTo(const Calendar& cal, UErrorCode& status) const {
|
|
return year == cal.get(UCAL_YEAR, status)
|
|
&& month == cal.get(UCAL_MONTH, status) + 1
|
|
&& day == cal.get(UCAL_DAY_OF_MONTH, status)
|
|
&& hour == cal.get(UCAL_HOUR_OF_DAY, status)
|
|
&& min == cal.get(UCAL_MINUTE, status)
|
|
&& sec == cal.get(UCAL_SECOND, status)
|
|
&& ms == cal.get(UCAL_MILLISECOND, status);
|
|
}
|
|
|
|
typedef struct {
|
|
const char* tzid;
|
|
const CalFields in;
|
|
const CalFields expLastGMT;
|
|
const CalFields expFirstGMT;
|
|
} RepeatedWallTimeTestData;
|
|
|
|
static const RepeatedWallTimeTestData RPDATA[] =
|
|
{
|
|
// Time zone Input wall time WALLTIME_LAST in GMT WALLTIME_FIRST in GMT
|
|
{"America/New_York", CalFields(2011,11,6,0,59,59), CalFields(2011,11,6,4,59,59), CalFields(2011,11,6,4,59,59)},
|
|
{"America/New_York", CalFields(2011,11,6,1,0,0), CalFields(2011,11,6,6,0,0), CalFields(2011,11,6,5,0,0)},
|
|
{"America/New_York", CalFields(2011,11,6,1,0,1), CalFields(2011,11,6,6,0,1), CalFields(2011,11,6,5,0,1)},
|
|
{"America/New_York", CalFields(2011,11,6,1,30,0), CalFields(2011,11,6,6,30,0), CalFields(2011,11,6,5,30,0)},
|
|
{"America/New_York", CalFields(2011,11,6,1,59,59), CalFields(2011,11,6,6,59,59), CalFields(2011,11,6,5,59,59)},
|
|
{"America/New_York", CalFields(2011,11,6,2,0,0), CalFields(2011,11,6,7,0,0), CalFields(2011,11,6,7,0,0)},
|
|
{"America/New_York", CalFields(2011,11,6,2,0,1), CalFields(2011,11,6,7,0,1), CalFields(2011,11,6,7,0,1)},
|
|
|
|
{"Australia/Lord_Howe", CalFields(2011,4,3,1,29,59), CalFields(2011,4,2,14,29,59), CalFields(2011,4,2,14,29,59)},
|
|
{"Australia/Lord_Howe", CalFields(2011,4,3,1,30,0), CalFields(2011,4,2,15,0,0), CalFields(2011,4,2,14,30,0)},
|
|
{"Australia/Lord_Howe", CalFields(2011,4,3,1,45,0), CalFields(2011,4,2,15,15,0), CalFields(2011,4,2,14,45,0)},
|
|
{"Australia/Lord_Howe", CalFields(2011,4,3,1,59,59), CalFields(2011,4,2,15,29,59), CalFields(2011,4,2,14,59,59)},
|
|
{"Australia/Lord_Howe", CalFields(2011,4,3,2,0,0), CalFields(2011,4,2,15,30,0), CalFields(2011,4,2,15,30,0)},
|
|
{"Australia/Lord_Howe", CalFields(2011,4,3,2,0,1), CalFields(2011,4,2,15,30,1), CalFields(2011,4,2,15,30,1)},
|
|
|
|
{nullptr, CalFields(0,0,0,0,0,0), CalFields(0,0,0,0,0,0), CalFields(0,0,0,0,0,0)}
|
|
};
|
|
|
|
void CalendarTest::TestRepeatedWallTime() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar calGMT((const TimeZone&)*TimeZone::getGMT(), status);
|
|
GregorianCalendar calDefault(status);
|
|
GregorianCalendar calLast(status);
|
|
GregorianCalendar calFirst(status);
|
|
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Failed to create a calendar object.");
|
|
return;
|
|
}
|
|
|
|
calLast.setRepeatedWallTimeOption(UCAL_WALLTIME_LAST);
|
|
calFirst.setRepeatedWallTimeOption(UCAL_WALLTIME_FIRST);
|
|
|
|
for (int32_t i = 0; RPDATA[i].tzid != nullptr; i++) {
|
|
char buf[32];
|
|
TimeZone *tz = TimeZone::createTimeZone(RPDATA[i].tzid);
|
|
|
|
// UCAL_WALLTIME_LAST
|
|
status = U_ZERO_ERROR;
|
|
calLast.setTimeZone(*tz);
|
|
RPDATA[i].in.setTo(calLast);
|
|
calGMT.setTime(calLast.getTime(status), status);
|
|
CalFields outLastGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (UCAL_WALLTIME_LAST) - ")
|
|
+ RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "]");
|
|
} else {
|
|
if (outLastGMT != RPDATA[i].expLastGMT) {
|
|
dataerrln(UnicodeString("Fail: UCAL_WALLTIME_LAST ") + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "] is parsed as "
|
|
+ outLastGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + RPDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
|
|
// default
|
|
status = U_ZERO_ERROR;
|
|
calDefault.setTimeZone(*tz);
|
|
RPDATA[i].in.setTo(calDefault);
|
|
calGMT.setTime(calDefault.getTime(status), status);
|
|
CalFields outDefGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (default) - ")
|
|
+ RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "]");
|
|
} else {
|
|
if (outDefGMT != RPDATA[i].expLastGMT) {
|
|
dataerrln(UnicodeString("Fail: (default) ") + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "] is parsed as "
|
|
+ outDefGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + RPDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
|
|
// UCAL_WALLTIME_FIRST
|
|
status = U_ZERO_ERROR;
|
|
calFirst.setTimeZone(*tz);
|
|
RPDATA[i].in.setTo(calFirst);
|
|
calGMT.setTime(calFirst.getTime(status), status);
|
|
CalFields outFirstGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (UCAL_WALLTIME_FIRST) - ")
|
|
+ RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "]");
|
|
} else {
|
|
if (outFirstGMT != RPDATA[i].expFirstGMT) {
|
|
dataerrln(UnicodeString("Fail: UCAL_WALLTIME_FIRST ") + RPDATA[i].in.toString(buf, sizeof(buf)) + "[" + RPDATA[i].tzid + "] is parsed as "
|
|
+ outFirstGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + RPDATA[i].expFirstGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
delete tz;
|
|
}
|
|
}
|
|
|
|
typedef struct {
|
|
const char* tzid;
|
|
const CalFields in;
|
|
UBool isValid;
|
|
const CalFields expLastGMT;
|
|
const CalFields expFirstGMT;
|
|
const CalFields expNextAvailGMT;
|
|
} SkippedWallTimeTestData;
|
|
|
|
static SkippedWallTimeTestData SKDATA[] =
|
|
{
|
|
// Time zone Input wall time valid? WALLTIME_LAST in GMT WALLTIME_FIRST in GMT WALLTIME_NEXT_VALID in GMT
|
|
{"America/New_York", CalFields(2011,3,13,1,59,59), true, CalFields(2011,3,13,6,59,59), CalFields(2011,3,13,6,59,59), CalFields(2011,3,13,6,59,59)},
|
|
{"America/New_York", CalFields(2011,3,13,2,0,0), false, CalFields(2011,3,13,7,0,0), CalFields(2011,3,13,6,0,0), CalFields(2011,3,13,7,0,0)},
|
|
{"America/New_York", CalFields(2011,3,13,2,1,0), false, CalFields(2011,3,13,7,1,0), CalFields(2011,3,13,6,1,0), CalFields(2011,3,13,7,0,0)},
|
|
{"America/New_York", CalFields(2011,3,13,2,30,0), false, CalFields(2011,3,13,7,30,0), CalFields(2011,3,13,6,30,0), CalFields(2011,3,13,7,0,0)},
|
|
{"America/New_York", CalFields(2011,3,13,2,59,59), false, CalFields(2011,3,13,7,59,59), CalFields(2011,3,13,6,59,59), CalFields(2011,3,13,7,0,0)},
|
|
{"America/New_York", CalFields(2011,3,13,3,0,0), true, CalFields(2011,3,13,7,0,0), CalFields(2011,3,13,7,0,0), CalFields(2011,3,13,7,0,0)},
|
|
|
|
{"Pacific/Apia", CalFields(2011,12,29,23,59,59), true, CalFields(2011,12,30,9,59,59), CalFields(2011,12,30,9,59,59), CalFields(2011,12,30,9,59,59)},
|
|
{"Pacific/Apia", CalFields(2011,12,30,0,0,0), false, CalFields(2011,12,30,10,0,0), CalFields(2011,12,29,10,0,0), CalFields(2011,12,30,10,0,0)},
|
|
{"Pacific/Apia", CalFields(2011,12,30,12,0,0), false, CalFields(2011,12,30,22,0,0), CalFields(2011,12,29,22,0,0), CalFields(2011,12,30,10,0,0)},
|
|
{"Pacific/Apia", CalFields(2011,12,30,23,59,59), false, CalFields(2011,12,31,9,59,59), CalFields(2011,12,30,9,59,59), CalFields(2011,12,30,10,0,0)},
|
|
{"Pacific/Apia", CalFields(2011,12,31,0,0,0), true, CalFields(2011,12,30,10,0,0), CalFields(2011,12,30,10,0,0), CalFields(2011,12,30,10,0,0)},
|
|
|
|
{nullptr, CalFields(0,0,0,0,0,0), true, CalFields(0,0,0,0,0,0), CalFields(0,0,0,0,0,0), CalFields(0,0,0,0,0,0)}
|
|
};
|
|
|
|
|
|
void CalendarTest::TestSkippedWallTime() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar calGMT((const TimeZone&)*TimeZone::getGMT(), status);
|
|
GregorianCalendar calDefault(status);
|
|
GregorianCalendar calLast(status);
|
|
GregorianCalendar calFirst(status);
|
|
GregorianCalendar calNextAvail(status);
|
|
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Failed to create a calendar object.");
|
|
return;
|
|
}
|
|
|
|
calLast.setSkippedWallTimeOption(UCAL_WALLTIME_LAST);
|
|
calFirst.setSkippedWallTimeOption(UCAL_WALLTIME_FIRST);
|
|
calNextAvail.setSkippedWallTimeOption(UCAL_WALLTIME_NEXT_VALID);
|
|
|
|
for (int32_t i = 0; SKDATA[i].tzid != nullptr; i++) {
|
|
UDate d;
|
|
char buf[32];
|
|
TimeZone *tz = TimeZone::createTimeZone(SKDATA[i].tzid);
|
|
|
|
for (int32_t j = 0; j < 2; j++) {
|
|
UBool bLenient = (j == 0);
|
|
|
|
// UCAL_WALLTIME_LAST
|
|
status = U_ZERO_ERROR;
|
|
calLast.setLenient(bLenient);
|
|
calLast.setTimeZone(*tz);
|
|
SKDATA[i].in.setTo(calLast);
|
|
d = calLast.getTime(status);
|
|
if (bLenient || SKDATA[i].isValid) {
|
|
calGMT.setTime(d, status);
|
|
CalFields outLastGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calLast/calGMT (UCAL_WALLTIME_LAST) - ")
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
} else {
|
|
if (outLastGMT != SKDATA[i].expLastGMT) {
|
|
dataerrln(UnicodeString("Fail: UCAL_WALLTIME_LAST ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
|
|
+ outLastGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
} else if (U_SUCCESS(status)) {
|
|
// strict, invalid wall time - must report an error
|
|
dataerrln(UnicodeString("Fail: An error expected (UCAL_WALLTIME_LAST)") +
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
}
|
|
|
|
// default
|
|
status = U_ZERO_ERROR;
|
|
calDefault.setLenient(bLenient);
|
|
calDefault.setTimeZone(*tz);
|
|
SKDATA[i].in.setTo(calDefault);
|
|
d = calDefault.getTime(status);
|
|
if (bLenient || SKDATA[i].isValid) {
|
|
calGMT.setTime(d, status);
|
|
CalFields outDefGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calDefault/calGMT (default) - ")
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
} else {
|
|
if (outDefGMT != SKDATA[i].expLastGMT) {
|
|
dataerrln(UnicodeString("Fail: (default) ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
|
|
+ outDefGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expLastGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
} else if (U_SUCCESS(status)) {
|
|
// strict, invalid wall time - must report an error
|
|
dataerrln(UnicodeString("Fail: An error expected (default)") +
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
}
|
|
|
|
// UCAL_WALLTIME_FIRST
|
|
status = U_ZERO_ERROR;
|
|
calFirst.setLenient(bLenient);
|
|
calFirst.setTimeZone(*tz);
|
|
SKDATA[i].in.setTo(calFirst);
|
|
d = calFirst.getTime(status);
|
|
if (bLenient || SKDATA[i].isValid) {
|
|
calGMT.setTime(d, status);
|
|
CalFields outFirstGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calFirst/calGMT (UCAL_WALLTIME_FIRST) - ")
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
} else {
|
|
if (outFirstGMT != SKDATA[i].expFirstGMT) {
|
|
dataerrln(UnicodeString("Fail: UCAL_WALLTIME_FIRST ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
|
|
+ outFirstGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expFirstGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
} else if (U_SUCCESS(status)) {
|
|
// strict, invalid wall time - must report an error
|
|
dataerrln(UnicodeString("Fail: An error expected (UCAL_WALLTIME_FIRST)") +
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
}
|
|
|
|
// UCAL_WALLTIME_NEXT_VALID
|
|
status = U_ZERO_ERROR;
|
|
calNextAvail.setLenient(bLenient);
|
|
calNextAvail.setTimeZone(*tz);
|
|
SKDATA[i].in.setTo(calNextAvail);
|
|
d = calNextAvail.getTime(status);
|
|
if (bLenient || SKDATA[i].isValid) {
|
|
calGMT.setTime(d, status);
|
|
CalFields outNextAvailGMT(calGMT, status);
|
|
if (U_FAILURE(status)) {
|
|
errln(UnicodeString("Fail: Failed to get/set time calNextAvail/calGMT (UCAL_WALLTIME_NEXT_VALID) - ")
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
} else {
|
|
if (outNextAvailGMT != SKDATA[i].expNextAvailGMT) {
|
|
dataerrln(UnicodeString("Fail: UCAL_WALLTIME_NEXT_VALID ") + SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "] is parsed as "
|
|
+ outNextAvailGMT.toString(buf, sizeof(buf)) + "[GMT]. Expected: " + SKDATA[i].expNextAvailGMT.toString(buf, sizeof(buf)) + "[GMT]");
|
|
}
|
|
}
|
|
} else if (U_SUCCESS(status)) {
|
|
// strict, invalid wall time - must report an error
|
|
dataerrln(UnicodeString("Fail: An error expected (UCAL_WALLTIME_NEXT_VALID)") +
|
|
+ SKDATA[i].in.toString(buf, sizeof(buf)) + "[" + SKDATA[i].tzid + "]");
|
|
}
|
|
}
|
|
|
|
delete tz;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestCloneLocale() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(TimeZone::getGMT()->clone(),
|
|
Locale::createFromName("en"), status));
|
|
TEST_CHECK_STATUS;
|
|
Locale l0 = cal->getLocale(ULOC_VALID_LOCALE, status);
|
|
TEST_CHECK_STATUS;
|
|
LocalPointer<Calendar> cal2(cal->clone());
|
|
Locale l = cal2->getLocale(ULOC_VALID_LOCALE, status);
|
|
if(l0!=l) {
|
|
errln("Error: cloned locale %s != original locale %s, status %s\n", l0.getName(), l.getName(), u_errorName(status));
|
|
}
|
|
TEST_CHECK_STATUS;
|
|
}
|
|
|
|
void CalendarTest::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-rgsa" },
|
|
{ "ar-u-ca-iso8601-tz-bjptn", "Africa/Porto-Novo", "iso8601" },
|
|
{ "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::AsssertCalendarFieldValue(
|
|
Calendar* cal, double time, const char* type,
|
|
int32_t era, int32_t year, int32_t month, int32_t week_of_year,
|
|
int32_t week_of_month, int32_t date, int32_t day_of_year, int32_t day_of_week,
|
|
int32_t day_of_week_in_month, int32_t am_pm, int32_t hour, int32_t hour_of_day,
|
|
int32_t minute, int32_t second, int32_t millisecond, int32_t zone_offset,
|
|
int32_t dst_offset, int32_t year_woy, int32_t dow_local, int32_t extended_year,
|
|
int32_t julian_day, int32_t milliseconds_in_day, int32_t is_leap_month) {
|
|
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->setTime(time, status);
|
|
assertEquals("getType", type, cal->getType());
|
|
|
|
assertEquals("UCAL_ERA", era, cal->get(UCAL_ERA, status));
|
|
assertEquals("UCAL_YEAR", year, cal->get(UCAL_YEAR, status));
|
|
assertEquals("UCAL_MONTH", month, cal->get(UCAL_MONTH, status));
|
|
assertEquals("UCAL_WEEK_OF_YEAR", week_of_year, cal->get(UCAL_WEEK_OF_YEAR, status));
|
|
assertEquals("UCAL_WEEK_OF_MONTH", week_of_month, cal->get(UCAL_WEEK_OF_MONTH, status));
|
|
assertEquals("UCAL_DATE", date, cal->get(UCAL_DATE, status));
|
|
assertEquals("UCAL_DAY_OF_YEAR", day_of_year, cal->get(UCAL_DAY_OF_YEAR, status));
|
|
assertEquals("UCAL_DAY_OF_WEEK", day_of_week, cal->get(UCAL_DAY_OF_WEEK, status));
|
|
assertEquals("UCAL_DAY_OF_WEEK_IN_MONTH", day_of_week_in_month, cal->get(UCAL_DAY_OF_WEEK_IN_MONTH, status));
|
|
assertEquals("UCAL_AM_PM", am_pm, cal->get(UCAL_AM_PM, status));
|
|
assertEquals("UCAL_HOUR", hour, cal->get(UCAL_HOUR, status));
|
|
assertEquals("UCAL_HOUR_OF_DAY", hour_of_day, cal->get(UCAL_HOUR_OF_DAY, status));
|
|
assertEquals("UCAL_MINUTE", minute, cal->get(UCAL_MINUTE, status));
|
|
assertEquals("UCAL_SECOND", second, cal->get(UCAL_SECOND, status));
|
|
assertEquals("UCAL_MILLISECOND", millisecond, cal->get(UCAL_MILLISECOND, status));
|
|
assertEquals("UCAL_ZONE_OFFSET", zone_offset, cal->get(UCAL_ZONE_OFFSET, status));
|
|
assertEquals("UCAL_DST_OFFSET", dst_offset, cal->get(UCAL_DST_OFFSET, status));
|
|
assertEquals("UCAL_YEAR_WOY", year_woy, cal->get(UCAL_YEAR_WOY, status));
|
|
assertEquals("UCAL_DOW_LOCAL", dow_local, cal->get(UCAL_DOW_LOCAL, status));
|
|
assertEquals("UCAL_EXTENDED_YEAR", extended_year, cal->get(UCAL_EXTENDED_YEAR, status));
|
|
assertEquals("UCAL_JULIAN_DAY", julian_day, cal->get(UCAL_JULIAN_DAY, status));
|
|
assertEquals("UCAL_MILLISECONDS_IN_DAY", milliseconds_in_day, cal->get(UCAL_MILLISECONDS_IN_DAY, status));
|
|
assertEquals("UCAL_IS_LEAP_MONTH", is_leap_month, cal->get(UCAL_IS_LEAP_MONTH, status));
|
|
}
|
|
|
|
static constexpr double test_time = 1667277891323; // Nov 1, 2022 4:44:51 GMT
|
|
|
|
void CalendarTest::TestBasicConversionGregorian() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=gregorian"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Gregorian calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "gregorian",
|
|
1, 2022, 10, 45, 1, 1, 305, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2022, 3, 2022, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionISO8601() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=iso8601"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get ISO8601 calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "iso8601",
|
|
1, 2022, 10, 44, 1, 1, 305, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2022, 2, 2022, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionJapanese() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=japanese"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Japanese calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "japanese",
|
|
236, 4, 10, 45, 1, 1, 305, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2022, 3, 2022, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionBuddhist() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=buddhist"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Buddhist calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "buddhist",
|
|
0, 2565, 10, 45, 1, 1, 305, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2022, 3, 2022, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionTaiwan() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=roc"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Taiwan calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "roc",
|
|
1, 111, 10, 45, 1, 1, 305, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2022, 3, 2022, 2459885, 17091323, 0);
|
|
|
|
}
|
|
void CalendarTest::TestBasicConversionPersian() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=persian"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Persian calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "persian",
|
|
0, 1401, 7, 33, 2, 10, 226, 3, 2, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1401, 3, 1401, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionIslamic() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=islamic"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Islamic calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "islamic",
|
|
0, 1444, 3, 15, 2, 7, 96, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1444, 3, 1444, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionIslamicTBLA() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=islamic-tbla"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get IslamicTBLA calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "islamic-tbla",
|
|
0, 1444, 3, 15, 2, 7, 96, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1444, 3, 1444, 2459885, 17091323, 0);
|
|
|
|
}
|
|
void CalendarTest::TestBasicConversionIslamicCivil() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=islamic-civil"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get IslamicCivil calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "islamic-civil",
|
|
0, 1444, 3, 15, 2, 6, 95, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1444, 3, 1444, 2459885, 17091323, 0);
|
|
|
|
}
|
|
void CalendarTest::TestBasicConversionIslamicRGSA() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=islamic-rgsa"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get IslamicRGSA calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "islamic-rgsa",
|
|
0, 1444, 3, 15, 2, 7, 96, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1444, 3, 1444, 2459885, 17091323, 0);
|
|
|
|
}
|
|
void CalendarTest::TestBasicConversionIslamicUmalqura() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=islamic-umalqura"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get IslamicUmalqura calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "islamic-umalqura",
|
|
0, 1444, 3, 15, 2, 7, 95, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1444, 3, 1444, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionHebrew() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=hebrew"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Hebrew calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "hebrew",
|
|
0, 5783, 1, 6, 2, 7, 37, 3, 1, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 5783, 3, 5783, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionChinese() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=chinese"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Chinese calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "chinese",
|
|
78, 39, 9, 40, 2, 8, 274, 3, 2, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 4659, 3, 4659, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionDangi() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=dangi"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Dangi calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "dangi",
|
|
78, 39, 9, 40, 2, 8, 274, 3, 2, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 4355, 3, 4355, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionIndian() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=indian"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Indian calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "indian",
|
|
0, 1944, 7, 33, 2, 10, 225, 3, 2, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1944, 3, 1944, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionCoptic() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=coptic"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Coptic calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "coptic",
|
|
1, 1739, 1, 8, 4, 22, 52, 3, 4, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 1739, 3, 1739, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionEthiopic() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=ethiopic"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get Ethiopic calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "ethiopic",
|
|
1, 2015, 1, 8, 4, 22, 52, 3, 4, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2015, 3, 2015, 2459885, 17091323, 0);
|
|
}
|
|
void CalendarTest::TestBasicConversionEthiopicAmeteAlem() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(icu::Calendar::createInstance(
|
|
*TimeZone::getGMT(), Locale("en@calendar=ethiopic-amete-alem"), status));
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: Cannot get EthiopicAmeteAlem calendar");
|
|
return;
|
|
}
|
|
AsssertCalendarFieldValue(
|
|
cal.getAlias(), test_time, "ethiopic-amete-alem",
|
|
0, 7515, 1, 8, 4, 22, 52, 3, 4, 0, 4, 4, 44, 51,
|
|
323, 0, 0, 2015, 3, 2015, 2459885, 17091323, 0);
|
|
}
|
|
|
|
|
|
void CalendarTest::setAndTestCalendar(Calendar* cal, int32_t initMonth, int32_t initDay, int32_t initYear, UErrorCode& status) {
|
|
cal->clear();
|
|
cal->setLenient(false);
|
|
cal->set(initYear, initMonth, initDay);
|
|
int32_t day = cal->get(UCAL_DAY_OF_MONTH, status);
|
|
int32_t month = cal->get(UCAL_MONTH, status);
|
|
int32_t year = cal->get(UCAL_YEAR, status);
|
|
if(U_FAILURE(status))
|
|
return;
|
|
|
|
if(initDay != day || initMonth != month || initYear != year)
|
|
{
|
|
errln(" year init values:\tmonth %i\tday %i\tyear %i", initMonth, initDay, initYear);
|
|
errln("values post set():\tmonth %i\tday %i\tyear %i",month, day, year);
|
|
}
|
|
}
|
|
|
|
void CalendarTest::setAndTestWholeYear(Calendar* cal, int32_t startYear, UErrorCode& status) {
|
|
for(int32_t startMonth = 0; startMonth < 12; startMonth++) {
|
|
for(int32_t startDay = 1; startDay < 31; startDay++ ) {
|
|
setAndTestCalendar(cal, startMonth, startDay, startYear, status);
|
|
if(U_FAILURE(status) && startDay == 30) {
|
|
status = U_ZERO_ERROR;
|
|
continue;
|
|
}
|
|
TEST_CHECK_STATUS;
|
|
}
|
|
}
|
|
}
|
|
|
|
// =====================================================================
|
|
|
|
typedef struct {
|
|
int16_t gYear;
|
|
int8_t gMon;
|
|
int8_t gDay;
|
|
int16_t uYear;
|
|
int8_t uMon;
|
|
int8_t uDay;
|
|
} GregoUmmAlQuraMap;
|
|
|
|
// data from
|
|
// Official Umm-al-Qura calendar of SA:
|
|
// home, http://www.ummulqura.org.sa/default.aspx
|
|
// converter, http://www.ummulqura.org.sa/Index.aspx
|
|
static const GregoUmmAlQuraMap guMappings[] = {
|
|
// gregorian, ummAlQura
|
|
// year mo da, year mo da
|
|
// (using 1-based months here)
|
|
{ 1882,11,12, 1300, 1, 1 },
|
|
{ 1892, 7,25, 1310, 1, 1 },
|
|
{ 1896, 6,12, 1314, 1, 1 },
|
|
{ 1898, 5,22, 1316, 1, 1 },
|
|
{ 1900, 4,30, 1318, 1, 1 },
|
|
{ 1901, 4,20, 1319, 1, 1 },
|
|
{ 1902, 4,10, 1320, 1, 1 },
|
|
{ 1903, 3,30, 1321, 1, 1 },
|
|
{ 1904, 3,19, 1322, 1, 1 },
|
|
{ 1905, 3, 8, 1323, 1, 1 },
|
|
{ 1906, 2,25, 1324, 1, 1 },
|
|
{ 1907, 2,14, 1325, 1, 1 },
|
|
{ 1908, 2, 4, 1326, 1, 1 },
|
|
{ 1909, 1,23, 1327, 1, 1 },
|
|
{ 1910, 1,13, 1328, 1, 1 },
|
|
{ 1911, 1, 2, 1329, 1, 1 },
|
|
{ 1911,12,22, 1330, 1, 1 },
|
|
{ 1912,12,10, 1331, 1, 1 },
|
|
{ 1913,11,30, 1332, 1, 1 },
|
|
{ 1914,11,19, 1333, 1, 1 },
|
|
{ 1915,11, 9, 1334, 1, 1 },
|
|
{ 1916,10,28, 1335, 1, 1 },
|
|
{ 1917,10,18, 1336, 1, 1 },
|
|
{ 1918,10, 7, 1337, 1, 1 },
|
|
{ 1919, 9,26, 1338, 1, 1 },
|
|
{ 1920, 9,14, 1339, 1, 1 },
|
|
{ 1921, 9, 4, 1340, 1, 1 },
|
|
{ 1922, 8,24, 1341, 1, 1 },
|
|
{ 1923, 8,14, 1342, 1, 1 },
|
|
{ 1924, 8, 2, 1343, 1, 1 },
|
|
{ 1925, 7,22, 1344, 1, 1 },
|
|
{ 1926, 7,11, 1345, 1, 1 },
|
|
{ 1927, 6,30, 1346, 1, 1 },
|
|
{ 1928, 6,19, 1347, 1, 1 },
|
|
{ 1929, 6, 9, 1348, 1, 1 },
|
|
{ 1930, 5,29, 1349, 1, 1 },
|
|
{ 1931, 5,19, 1350, 1, 1 },
|
|
{ 1932, 5, 7, 1351, 1, 1 },
|
|
{ 1933, 4,26, 1352, 1, 1 },
|
|
{ 1934, 4,15, 1353, 1, 1 },
|
|
{ 1935, 4, 5, 1354, 1, 1 },
|
|
{ 1936, 3,24, 1355, 1, 1 },
|
|
{ 1937, 3,14, 1356, 1, 1 },
|
|
{ 1938, 3, 4, 1357, 1, 1 },
|
|
{ 1939, 2,21, 1358, 1, 1 },
|
|
{ 1940, 2,10, 1359, 1, 1 },
|
|
{ 1941, 1,29, 1360, 1, 1 },
|
|
{ 1942, 1,18, 1361, 1, 1 },
|
|
{ 1943, 1, 8, 1362, 1, 1 },
|
|
{ 1943,12,28, 1363, 1, 1 },
|
|
{ 1944,12,17, 1364, 1, 1 },
|
|
{ 1945,12, 6, 1365, 1, 1 },
|
|
{ 1946,11,25, 1366, 1, 1 },
|
|
{ 1947,11,14, 1367, 1, 1 },
|
|
{ 1948,11, 3, 1368, 1, 1 },
|
|
{ 1949,10,23, 1369, 1, 1 },
|
|
{ 1950,10,13, 1370, 1, 1 },
|
|
{ 1951,10, 3, 1371, 1, 1 },
|
|
{ 1952, 9,21, 1372, 1, 1 },
|
|
{ 1953, 9,10, 1373, 1, 1 },
|
|
{ 1954, 8,30, 1374, 1, 1 },
|
|
{ 1955, 8,19, 1375, 1, 1 },
|
|
{ 1956, 8, 8, 1376, 1, 1 },
|
|
{ 1957, 7,29, 1377, 1, 1 },
|
|
{ 1958, 7,18, 1378, 1, 1 },
|
|
{ 1959, 7, 8, 1379, 1, 1 },
|
|
{ 1960, 6,26, 1380, 1, 1 },
|
|
{ 1961, 6,15, 1381, 1, 1 },
|
|
{ 1962, 6, 4, 1382, 1, 1 },
|
|
{ 1963, 5,24, 1383, 1, 1 },
|
|
{ 1964, 5,13, 1384, 1, 1 },
|
|
{ 1965, 5, 3, 1385, 1, 1 },
|
|
{ 1966, 4,22, 1386, 1, 1 },
|
|
{ 1967, 4,11, 1387, 1, 1 },
|
|
{ 1968, 3,30, 1388, 1, 1 },
|
|
{ 1969, 3,19, 1389, 1, 1 },
|
|
{ 1970, 3, 9, 1390, 1, 1 },
|
|
{ 1971, 2,27, 1391, 1, 1 },
|
|
{ 1972, 2,16, 1392, 1, 1 },
|
|
{ 1973, 2, 5, 1393, 1, 1 },
|
|
{ 1974, 1,25, 1394, 1, 1 },
|
|
{ 1975, 1,14, 1395, 1, 1 },
|
|
{ 1976, 1, 3, 1396, 1, 1 },
|
|
{ 1976,12,22, 1397, 1, 1 },
|
|
{ 1977,12,12, 1398, 1, 1 },
|
|
{ 1978,12, 1, 1399, 1, 1 },
|
|
{ 1979,11,21, 1400, 1, 1 },
|
|
{ 1980,11, 9, 1401, 1, 1 },
|
|
{ 1981,10,29, 1402, 1, 1 },
|
|
{ 1982,10,18, 1403, 1, 1 },
|
|
{ 1983,10, 8, 1404, 1, 1 },
|
|
{ 1984, 9,26, 1405, 1, 1 },
|
|
{ 1985, 9,16, 1406, 1, 1 },
|
|
{ 1986, 9, 6, 1407, 1, 1 },
|
|
{ 1987, 8,26, 1408, 1, 1 },
|
|
{ 1988, 8,14, 1409, 1, 1 },
|
|
{ 1989, 8, 3, 1410, 1, 1 },
|
|
{ 1990, 7,23, 1411, 1, 1 },
|
|
{ 1991, 7,13, 1412, 1, 1 },
|
|
{ 1992, 7, 2, 1413, 1, 1 },
|
|
{ 1993, 6,21, 1414, 1, 1 },
|
|
{ 1994, 6,11, 1415, 1, 1 },
|
|
{ 1995, 5,31, 1416, 1, 1 },
|
|
{ 1996, 5,19, 1417, 1, 1 },
|
|
{ 1997, 5, 8, 1418, 1, 1 },
|
|
{ 1998, 4,28, 1419, 1, 1 },
|
|
{ 1999, 4,17, 1420, 1, 1 },
|
|
{ 1999, 5,16, 1420, 2, 1 },
|
|
{ 1999, 6,15, 1420, 3, 1 },
|
|
{ 1999, 7,14, 1420, 4, 1 },
|
|
{ 1999, 8,12, 1420, 5, 1 },
|
|
{ 1999, 9,11, 1420, 6, 1 },
|
|
{ 1999,10,10, 1420, 7, 1 },
|
|
{ 1999,11, 9, 1420, 8, 1 },
|
|
{ 1999,12, 9, 1420, 9, 1 },
|
|
{ 2000, 1, 8, 1420,10, 1 },
|
|
{ 2000, 2, 7, 1420,11, 1 },
|
|
{ 2000, 3, 7, 1420,12, 1 },
|
|
{ 2000, 4, 6, 1421, 1, 1 },
|
|
{ 2000, 5, 5, 1421, 2, 1 },
|
|
{ 2000, 6, 3, 1421, 3, 1 },
|
|
{ 2000, 7, 3, 1421, 4, 1 },
|
|
{ 2000, 8, 1, 1421, 5, 1 },
|
|
{ 2000, 8,30, 1421, 6, 1 },
|
|
{ 2000, 9,28, 1421, 7, 1 },
|
|
{ 2000,10,28, 1421, 8, 1 },
|
|
{ 2000,11,27, 1421, 9, 1 },
|
|
{ 2000,12,27, 1421,10, 1 },
|
|
{ 2001, 1,26, 1421,11, 1 },
|
|
{ 2001, 2,24, 1421,12, 1 },
|
|
{ 2001, 3,26, 1422, 1, 1 },
|
|
{ 2001, 4,25, 1422, 2, 1 },
|
|
{ 2001, 5,24, 1422, 3, 1 },
|
|
{ 2001, 6,22, 1422, 4, 1 },
|
|
{ 2001, 7,22, 1422, 5, 1 },
|
|
{ 2001, 8,20, 1422, 6, 1 },
|
|
{ 2001, 9,18, 1422, 7, 1 },
|
|
{ 2001,10,17, 1422, 8, 1 },
|
|
{ 2001,11,16, 1422, 9, 1 },
|
|
{ 2001,12,16, 1422,10, 1 },
|
|
{ 2002, 1,15, 1422,11, 1 },
|
|
{ 2002, 2,13, 1422,12, 1 },
|
|
{ 2002, 3,15, 1423, 1, 1 },
|
|
{ 2002, 4,14, 1423, 2, 1 },
|
|
{ 2002, 5,13, 1423, 3, 1 },
|
|
{ 2002, 6,12, 1423, 4, 1 },
|
|
{ 2002, 7,11, 1423, 5, 1 },
|
|
{ 2002, 8,10, 1423, 6, 1 },
|
|
{ 2002, 9, 8, 1423, 7, 1 },
|
|
{ 2002,10, 7, 1423, 8, 1 },
|
|
{ 2002,11, 6, 1423, 9, 1 },
|
|
{ 2002,12, 5, 1423,10, 1 },
|
|
{ 2003, 1, 4, 1423,11, 1 },
|
|
{ 2003, 2, 2, 1423,12, 1 },
|
|
{ 2003, 3, 4, 1424, 1, 1 },
|
|
{ 2003, 4, 3, 1424, 2, 1 },
|
|
{ 2003, 5, 2, 1424, 3, 1 },
|
|
{ 2003, 6, 1, 1424, 4, 1 },
|
|
{ 2003, 7, 1, 1424, 5, 1 },
|
|
{ 2003, 7,30, 1424, 6, 1 },
|
|
{ 2003, 8,29, 1424, 7, 1 },
|
|
{ 2003, 9,27, 1424, 8, 1 },
|
|
{ 2003,10,26, 1424, 9, 1 },
|
|
{ 2003,11,25, 1424,10, 1 },
|
|
{ 2003,12,24, 1424,11, 1 },
|
|
{ 2004, 1,23, 1424,12, 1 },
|
|
{ 2004, 2,21, 1425, 1, 1 },
|
|
{ 2004, 3,22, 1425, 2, 1 },
|
|
{ 2004, 4,20, 1425, 3, 1 },
|
|
{ 2004, 5,20, 1425, 4, 1 },
|
|
{ 2004, 6,19, 1425, 5, 1 },
|
|
{ 2004, 7,18, 1425, 6, 1 },
|
|
{ 2004, 8,17, 1425, 7, 1 },
|
|
{ 2004, 9,15, 1425, 8, 1 },
|
|
{ 2004,10,15, 1425, 9, 1 },
|
|
{ 2004,11,14, 1425,10, 1 },
|
|
{ 2004,12,13, 1425,11, 1 },
|
|
{ 2005, 1,12, 1425,12, 1 },
|
|
{ 2005, 2,10, 1426, 1, 1 },
|
|
{ 2005, 3,11, 1426, 2, 1 },
|
|
{ 2005, 4,10, 1426, 3, 1 },
|
|
{ 2005, 5, 9, 1426, 4, 1 },
|
|
{ 2005, 6, 8, 1426, 5, 1 },
|
|
{ 2005, 7, 7, 1426, 6, 1 },
|
|
{ 2005, 8, 6, 1426, 7, 1 },
|
|
{ 2005, 9, 5, 1426, 8, 1 },
|
|
{ 2005,10, 4, 1426, 9, 1 },
|
|
{ 2005,11, 3, 1426,10, 1 },
|
|
{ 2005,12, 3, 1426,11, 1 },
|
|
{ 2006, 1, 1, 1426,12, 1 },
|
|
{ 2006, 1,31, 1427, 1, 1 },
|
|
{ 2006, 3, 1, 1427, 2, 1 },
|
|
{ 2006, 3,30, 1427, 3, 1 },
|
|
{ 2006, 4,29, 1427, 4, 1 },
|
|
{ 2006, 5,28, 1427, 5, 1 },
|
|
{ 2006, 6,27, 1427, 6, 1 },
|
|
{ 2006, 7,26, 1427, 7, 1 },
|
|
{ 2006, 8,25, 1427, 8, 1 },
|
|
{ 2006, 9,24, 1427, 9, 1 },
|
|
{ 2006,10,23, 1427,10, 1 },
|
|
{ 2006,11,22, 1427,11, 1 },
|
|
{ 2006,12,22, 1427,12, 1 },
|
|
{ 2007, 1,20, 1428, 1, 1 },
|
|
{ 2007, 2,19, 1428, 2, 1 },
|
|
{ 2007, 3,20, 1428, 3, 1 },
|
|
{ 2007, 4,18, 1428, 4, 1 },
|
|
{ 2007, 5,18, 1428, 5, 1 },
|
|
{ 2007, 6,16, 1428, 6, 1 },
|
|
{ 2007, 7,15, 1428, 7, 1 },
|
|
{ 2007, 8,14, 1428, 8, 1 },
|
|
{ 2007, 9,13, 1428, 9, 1 },
|
|
{ 2007,10,13, 1428,10, 1 },
|
|
{ 2007,11,11, 1428,11, 1 },
|
|
{ 2007,12,11, 1428,12, 1 },
|
|
{ 2008, 1,10, 1429, 1, 1 },
|
|
{ 2008, 2, 8, 1429, 2, 1 },
|
|
{ 2008, 3, 9, 1429, 3, 1 },
|
|
{ 2008, 4, 7, 1429, 4, 1 },
|
|
{ 2008, 5, 6, 1429, 5, 1 },
|
|
{ 2008, 6, 5, 1429, 6, 1 },
|
|
{ 2008, 7, 4, 1429, 7, 1 },
|
|
{ 2008, 8, 2, 1429, 8, 1 },
|
|
{ 2008, 9, 1, 1429, 9, 1 },
|
|
{ 2008,10, 1, 1429,10, 1 },
|
|
{ 2008,10,30, 1429,11, 1 },
|
|
{ 2008,11,29, 1429,12, 1 },
|
|
{ 2008,12,29, 1430, 1, 1 },
|
|
{ 2009, 1,27, 1430, 2, 1 },
|
|
{ 2009, 2,26, 1430, 3, 1 },
|
|
{ 2009, 3,28, 1430, 4, 1 },
|
|
{ 2009, 4,26, 1430, 5, 1 },
|
|
{ 2009, 5,25, 1430, 6, 1 },
|
|
{ 2009, 6,24, 1430, 7, 1 },
|
|
{ 2009, 7,23, 1430, 8, 1 },
|
|
{ 2009, 8,22, 1430, 9, 1 },
|
|
{ 2009, 9,20, 1430,10, 1 },
|
|
{ 2009,10,20, 1430,11, 1 },
|
|
{ 2009,11,18, 1430,12, 1 },
|
|
{ 2009,12,18, 1431, 1, 1 },
|
|
{ 2010, 1,16, 1431, 2, 1 },
|
|
{ 2010, 2,15, 1431, 3, 1 },
|
|
{ 2010, 3,17, 1431, 4, 1 },
|
|
{ 2010, 4,15, 1431, 5, 1 },
|
|
{ 2010, 5,15, 1431, 6, 1 },
|
|
{ 2010, 6,13, 1431, 7, 1 },
|
|
{ 2010, 7,13, 1431, 8, 1 },
|
|
{ 2010, 8,11, 1431, 9, 1 },
|
|
{ 2010, 9,10, 1431,10, 1 },
|
|
{ 2010,10, 9, 1431,11, 1 },
|
|
{ 2010,11, 7, 1431,12, 1 },
|
|
{ 2010,12, 7, 1432, 1, 1 },
|
|
{ 2011, 1, 5, 1432, 2, 1 },
|
|
{ 2011, 2, 4, 1432, 3, 1 },
|
|
{ 2011, 3, 6, 1432, 4, 1 },
|
|
{ 2011, 4, 5, 1432, 5, 1 },
|
|
{ 2011, 5, 4, 1432, 6, 1 },
|
|
{ 2011, 6, 3, 1432, 7, 1 },
|
|
{ 2011, 7, 2, 1432, 8, 1 },
|
|
{ 2011, 8, 1, 1432, 9, 1 },
|
|
{ 2011, 8,30, 1432,10, 1 },
|
|
{ 2011, 9,29, 1432,11, 1 },
|
|
{ 2011,10,28, 1432,12, 1 },
|
|
{ 2011,11,26, 1433, 1, 1 },
|
|
{ 2011,12,26, 1433, 2, 1 },
|
|
{ 2012, 1,24, 1433, 3, 1 },
|
|
{ 2012, 2,23, 1433, 4, 1 },
|
|
{ 2012, 3,24, 1433, 5, 1 },
|
|
{ 2012, 4,22, 1433, 6, 1 },
|
|
{ 2012, 5,22, 1433, 7, 1 },
|
|
{ 2012, 6,21, 1433, 8, 1 },
|
|
{ 2012, 7,20, 1433, 9, 1 },
|
|
{ 2012, 8,19, 1433,10, 1 },
|
|
{ 2012, 9,17, 1433,11, 1 },
|
|
{ 2012,10,17, 1433,12, 1 },
|
|
{ 2012,11,15, 1434, 1, 1 },
|
|
{ 2012,12,14, 1434, 2, 1 },
|
|
{ 2013, 1,13, 1434, 3, 1 },
|
|
{ 2013, 2,11, 1434, 4, 1 },
|
|
{ 2013, 3,13, 1434, 5, 1 },
|
|
{ 2013, 4,11, 1434, 6, 1 },
|
|
{ 2013, 5,11, 1434, 7, 1 },
|
|
{ 2013, 6,10, 1434, 8, 1 },
|
|
{ 2013, 7, 9, 1434, 9, 1 },
|
|
{ 2013, 8, 8, 1434,10, 1 },
|
|
{ 2013, 9, 7, 1434,11, 1 },
|
|
{ 2013,10, 6, 1434,12, 1 },
|
|
{ 2013,11, 4, 1435, 1, 1 },
|
|
{ 2013,12, 4, 1435, 2, 1 },
|
|
{ 2014, 1, 2, 1435, 3, 1 },
|
|
{ 2014, 2, 1, 1435, 4, 1 },
|
|
{ 2014, 3, 2, 1435, 5, 1 },
|
|
{ 2014, 4, 1, 1435, 6, 1 },
|
|
{ 2014, 4,30, 1435, 7, 1 },
|
|
{ 2014, 5,30, 1435, 8, 1 },
|
|
{ 2014, 6,28, 1435, 9, 1 },
|
|
{ 2014, 7,28, 1435,10, 1 },
|
|
{ 2014, 8,27, 1435,11, 1 },
|
|
{ 2014, 9,25, 1435,12, 1 },
|
|
{ 2014,10,25, 1436, 1, 1 },
|
|
{ 2014,11,23, 1436, 2, 1 },
|
|
{ 2014,12,23, 1436, 3, 1 },
|
|
{ 2015, 1,21, 1436, 4, 1 },
|
|
{ 2015, 2,20, 1436, 5, 1 },
|
|
{ 2015, 3,21, 1436, 6, 1 },
|
|
{ 2015, 4,20, 1436, 7, 1 },
|
|
{ 2015, 5,19, 1436, 8, 1 },
|
|
{ 2015, 6,18, 1436, 9, 1 },
|
|
{ 2015, 7,17, 1436,10, 1 },
|
|
{ 2015, 8,16, 1436,11, 1 },
|
|
{ 2015, 9,14, 1436,12, 1 },
|
|
{ 2015,10,14, 1437, 1, 1 },
|
|
{ 2015,11,13, 1437, 2, 1 },
|
|
{ 2015,12,12, 1437, 3, 1 },
|
|
{ 2016, 1,11, 1437, 4, 1 },
|
|
{ 2016, 2,10, 1437, 5, 1 },
|
|
{ 2016, 3,10, 1437, 6, 1 },
|
|
{ 2016, 4, 8, 1437, 7, 1 },
|
|
{ 2016, 5, 8, 1437, 8, 1 },
|
|
{ 2016, 6, 6, 1437, 9, 1 },
|
|
{ 2016, 7, 6, 1437,10, 1 },
|
|
{ 2016, 8, 4, 1437,11, 1 },
|
|
{ 2016, 9, 2, 1437,12, 1 },
|
|
{ 2016,10, 2, 1438, 1, 1 },
|
|
{ 2016,11, 1, 1438, 2, 1 },
|
|
{ 2016,11,30, 1438, 3, 1 },
|
|
{ 2016,12,30, 1438, 4, 1 },
|
|
{ 2017, 1,29, 1438, 5, 1 },
|
|
{ 2017, 2,28, 1438, 6, 1 },
|
|
{ 2017, 3,29, 1438, 7, 1 },
|
|
{ 2017, 4,27, 1438, 8, 1 },
|
|
{ 2017, 5,27, 1438, 9, 1 },
|
|
{ 2017, 6,25, 1438,10, 1 },
|
|
{ 2017, 7,24, 1438,11, 1 },
|
|
{ 2017, 8,23, 1438,12, 1 },
|
|
{ 2017, 9,21, 1439, 1, 1 },
|
|
{ 2017,10,21, 1439, 2, 1 },
|
|
{ 2017,11,19, 1439, 3, 1 },
|
|
{ 2017,12,19, 1439, 4, 1 },
|
|
{ 2018, 1,18, 1439, 5, 1 },
|
|
{ 2018, 2,17, 1439, 6, 1 },
|
|
{ 2018, 3,18, 1439, 7, 1 },
|
|
{ 2018, 4,17, 1439, 8, 1 },
|
|
{ 2018, 5,16, 1439, 9, 1 },
|
|
{ 2018, 6,15, 1439,10, 1 },
|
|
{ 2018, 7,14, 1439,11, 1 },
|
|
{ 2018, 8,12, 1439,12, 1 },
|
|
{ 2018, 9,11, 1440, 1, 1 },
|
|
{ 2019, 8,31, 1441, 1, 1 },
|
|
{ 2020, 8,20, 1442, 1, 1 },
|
|
{ 2021, 8, 9, 1443, 1, 1 },
|
|
{ 2022, 7,30, 1444, 1, 1 },
|
|
{ 2023, 7,19, 1445, 1, 1 },
|
|
{ 2024, 7, 7, 1446, 1, 1 },
|
|
{ 2025, 6,26, 1447, 1, 1 },
|
|
{ 2026, 6,16, 1448, 1, 1 },
|
|
{ 2027, 6, 6, 1449, 1, 1 },
|
|
{ 2028, 5,25, 1450, 1, 1 },
|
|
{ 2029, 5,14, 1451, 1, 1 },
|
|
{ 2030, 5, 4, 1452, 1, 1 },
|
|
{ 2031, 4,23, 1453, 1, 1 },
|
|
{ 2032, 4,11, 1454, 1, 1 },
|
|
{ 2033, 4, 1, 1455, 1, 1 },
|
|
{ 2034, 3,22, 1456, 1, 1 },
|
|
{ 2035, 3,11, 1457, 1, 1 },
|
|
{ 2036, 2,29, 1458, 1, 1 },
|
|
{ 2037, 2,17, 1459, 1, 1 },
|
|
{ 2038, 2, 6, 1460, 1, 1 },
|
|
{ 2039, 1,26, 1461, 1, 1 },
|
|
{ 2040, 1,15, 1462, 1, 1 },
|
|
{ 2041, 1, 4, 1463, 1, 1 },
|
|
{ 2041,12,25, 1464, 1, 1 },
|
|
{ 2042,12,14, 1465, 1, 1 },
|
|
{ 2043,12, 3, 1466, 1, 1 },
|
|
{ 2044,11,21, 1467, 1, 1 },
|
|
{ 2045,11,11, 1468, 1, 1 },
|
|
{ 2046,10,31, 1469, 1, 1 },
|
|
{ 2047,10,21, 1470, 1, 1 },
|
|
{ 2048,10, 9, 1471, 1, 1 },
|
|
{ 2049, 9,29, 1472, 1, 1 },
|
|
{ 2050, 9,18, 1473, 1, 1 },
|
|
{ 2051, 9, 7, 1474, 1, 1 },
|
|
{ 2052, 8,26, 1475, 1, 1 },
|
|
{ 2053, 8,15, 1476, 1, 1 },
|
|
{ 2054, 8, 5, 1477, 1, 1 },
|
|
{ 2055, 7,26, 1478, 1, 1 },
|
|
{ 2056, 7,14, 1479, 1, 1 },
|
|
{ 2057, 7, 3, 1480, 1, 1 },
|
|
{ 2058, 6,22, 1481, 1, 1 },
|
|
{ 2059, 6,11, 1482, 1, 1 },
|
|
{ 2061, 5,21, 1484, 1, 1 },
|
|
{ 2063, 4,30, 1486, 1, 1 },
|
|
{ 2065, 4, 7, 1488, 1, 1 },
|
|
{ 2067, 3,17, 1490, 1, 1 },
|
|
{ 2069, 2,23, 1492, 1, 1 },
|
|
{ 2071, 2, 2, 1494, 1, 1 },
|
|
{ 2073, 1,10, 1496, 1, 1 },
|
|
{ 2074,12,20, 1498, 1, 1 },
|
|
{ 2076,11,28, 1500, 1, 1 },
|
|
{ 0, 0, 0, 0, 0, 0 }, // terminator
|
|
};
|
|
|
|
static const char16_t zoneSA[] = {0x41,0x73,0x69,0x61,0x2F,0x52,0x69,0x79,0x61,0x64,0x68,0}; // "Asia/Riyadh"
|
|
|
|
void CalendarTest::TestIslamicUmAlQura() {
|
|
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale umalquraLoc("ar_SA@calendar=islamic-umalqura");
|
|
Locale gregoLoc("ar_SA@calendar=gregorian");
|
|
TimeZone* tzSA = TimeZone::createTimeZone(UnicodeString(true, zoneSA, -1));
|
|
Calendar* tstCal = Calendar::createInstance(*tzSA, umalquraLoc, status);
|
|
Calendar* gregCal = Calendar::createInstance(*tzSA, gregoLoc, status);
|
|
|
|
IslamicCalendar* iCal = dynamic_cast<IslamicCalendar*>(tstCal);
|
|
if(uprv_strcmp(iCal->getType(), "islamic-umalqura") != 0) {
|
|
errln("wrong type of calendar created - %s", iCal->getType());
|
|
}
|
|
|
|
int32_t firstYear = 1318;
|
|
int32_t lastYear = 1368; // just enough to be pretty sure
|
|
//int32_t lastYear = 1480; // the whole shootin' match
|
|
|
|
tstCal->clear();
|
|
tstCal->setLenient(false);
|
|
|
|
int32_t day=0, month=0, year=0, initDay = 27, initMonth = IslamicCalendar::RAJAB, initYear = 1434;
|
|
|
|
for( int32_t startYear = firstYear; startYear <= lastYear; startYear++) {
|
|
setAndTestWholeYear(tstCal, startYear, status);
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
|
|
initMonth = IslamicCalendar::RABI_2;
|
|
initDay = 5;
|
|
int32_t loopCnt = 25;
|
|
tstCal->clear();
|
|
setAndTestCalendar( tstCal, initMonth, initDay, initYear, status);
|
|
TEST_CHECK_STATUS;
|
|
|
|
for(int x=1; x<=loopCnt; x++) {
|
|
day = tstCal->get(UCAL_DAY_OF_MONTH,status);
|
|
month = tstCal->get(UCAL_MONTH,status);
|
|
year = tstCal->get(UCAL_YEAR,status);
|
|
TEST_CHECK_STATUS;
|
|
tstCal->roll(UCAL_DAY_OF_MONTH, static_cast<UBool>(true), status);
|
|
TEST_CHECK_STATUS;
|
|
}
|
|
|
|
if(day != (initDay + loopCnt - 1) || month != IslamicCalendar::RABI_2 || year != 1434)
|
|
errln("invalid values for RABI_2 date after roll of %d", loopCnt);
|
|
|
|
status = U_ZERO_ERROR;
|
|
tstCal->clear();
|
|
initMonth = 2;
|
|
initDay = 30;
|
|
setAndTestCalendar( tstCal, initMonth, initDay, initYear, status);
|
|
if(U_SUCCESS(status)) {
|
|
errln("error NOT detected status %i",status);
|
|
errln(" init values:\tmonth %i\tday %i\tyear %i", initMonth, initDay, initYear);
|
|
int32_t day = tstCal->get(UCAL_DAY_OF_MONTH, status);
|
|
int32_t month = tstCal->get(UCAL_MONTH, status);
|
|
int32_t year = tstCal->get(UCAL_YEAR, status);
|
|
errln("values post set():\tmonth %i\tday %i\tyear %i",month, day, year);
|
|
}
|
|
|
|
status = U_ZERO_ERROR;
|
|
tstCal->clear();
|
|
initMonth = 3;
|
|
initDay = 30;
|
|
setAndTestCalendar( tstCal, initMonth, initDay, initYear, status);
|
|
TEST_CHECK_STATUS;
|
|
|
|
SimpleDateFormat* formatter = new SimpleDateFormat("yyyy-MM-dd", Locale::getUS(), status);
|
|
UDate date = formatter->parse("1975-05-06", status);
|
|
Calendar* is_cal = Calendar::createInstance(umalquraLoc, status);
|
|
is_cal->setTime(date, status);
|
|
int32_t is_day = is_cal->get(UCAL_DAY_OF_MONTH,status);
|
|
int32_t is_month = is_cal->get(UCAL_MONTH,status);
|
|
int32_t is_year = is_cal->get(UCAL_YEAR,status);
|
|
TEST_CHECK_STATUS;
|
|
if(is_day != 24 || is_month != IslamicCalendar::RABI_2 || is_year != 1395)
|
|
errln("unexpected conversion date month %i not %i or day %i not 24 or year %i not 1395", is_month, IslamicCalendar::RABI_2, is_day, is_year);
|
|
|
|
UDate date2 = is_cal->getTime(status);
|
|
TEST_CHECK_STATUS;
|
|
if(date2 != date) {
|
|
errln("before(%f) and after(%f) dates don't match up!",date, date2);
|
|
}
|
|
|
|
// check against data
|
|
const GregoUmmAlQuraMap* guMapPtr;
|
|
gregCal->clear();
|
|
tstCal->clear();
|
|
for (guMapPtr = guMappings; guMapPtr->gYear != 0; guMapPtr++) {
|
|
status = U_ZERO_ERROR;
|
|
gregCal->set(guMapPtr->gYear, guMapPtr->gMon - 1, guMapPtr->gDay, 12, 0);
|
|
date = gregCal->getTime(status);
|
|
tstCal->setTime(date, status);
|
|
int32_t uYear = tstCal->get(UCAL_YEAR, status);
|
|
int32_t uMon = tstCal->get(UCAL_MONTH, status) + 1;
|
|
int32_t uDay = tstCal->get(UCAL_DATE, status);
|
|
if(U_FAILURE(status)) {
|
|
errln("For gregorian %4d-%02d-%02d, get status %s",
|
|
guMapPtr->gYear, guMapPtr->gMon, guMapPtr->gDay, u_errorName(status) );
|
|
} else if (uYear != guMapPtr->uYear || uMon != guMapPtr->uMon || uDay != guMapPtr->uDay) {
|
|
errln("For gregorian %4d-%02d-%02d, expect umalqura %4d-%02d-%02d, get %4d-%02d-%02d",
|
|
guMapPtr->gYear, guMapPtr->gMon, guMapPtr->gDay,
|
|
guMapPtr->uYear, guMapPtr->uMon, guMapPtr->uDay, uYear, uMon, uDay );
|
|
}
|
|
}
|
|
|
|
delete is_cal;
|
|
delete formatter;
|
|
delete gregCal;
|
|
delete tstCal;
|
|
delete tzSA;
|
|
}
|
|
|
|
void CalendarTest::TestIslamicTabularDates() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale islamicLoc("ar_SA@calendar=islamic-civil");
|
|
Locale tblaLoc("ar_SA@calendar=islamic-tbla");
|
|
SimpleDateFormat* formatter = new SimpleDateFormat("yyyy-MM-dd", Locale::getUS(), status);
|
|
UDate date = formatter->parse("1975-05-06", status);
|
|
|
|
Calendar* tstCal = Calendar::createInstance(islamicLoc, status);
|
|
tstCal->setTime(date, status);
|
|
int32_t is_day = tstCal->get(UCAL_DAY_OF_MONTH,status);
|
|
int32_t is_month = tstCal->get(UCAL_MONTH,status);
|
|
int32_t is_year = tstCal->get(UCAL_YEAR,status);
|
|
TEST_CHECK_STATUS;
|
|
delete tstCal;
|
|
|
|
tstCal = Calendar::createInstance(tblaLoc, status);
|
|
tstCal->setTime(date, status);
|
|
int32_t tbla_day = tstCal->get(UCAL_DAY_OF_MONTH,status);
|
|
int32_t tbla_month = tstCal->get(UCAL_MONTH,status);
|
|
int32_t tbla_year = tstCal->get(UCAL_YEAR,status);
|
|
TEST_CHECK_STATUS;
|
|
|
|
if(tbla_month != is_month || tbla_year != is_year)
|
|
errln("unexpected difference between islamic and tbla month %d : %d and/or year %d : %d",tbla_month,is_month,tbla_year,is_year);
|
|
|
|
if(tbla_day - is_day != 1)
|
|
errln("unexpected day difference between islamic and tbla: %d : %d ",tbla_day,is_day);
|
|
delete tstCal;
|
|
delete formatter;
|
|
}
|
|
|
|
void CalendarTest::TestHebrewMonthValidation() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale::createFromName("he_IL@calendar=hebrew"), status));
|
|
if (failure(status, "Calendar::createInstance, locale:he_IL@calendar=hebrew", true)) return;
|
|
Calendar *pCal = cal.getAlias();
|
|
|
|
UDate d;
|
|
pCal->setLenient(false);
|
|
|
|
// 5776 is a leap year and has month Adar I
|
|
pCal->set(5776, HebrewCalendar::ADAR_1, 1);
|
|
d = pCal->getTime(status);
|
|
if (U_FAILURE(status)) {
|
|
errln("Fail: 5776 Adar I 1 is a valid date.");
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
|
|
// 5777 is NOT a lear year and does not have month Adar I
|
|
pCal->set(5777, HebrewCalendar::ADAR_1, 1);
|
|
d = pCal->getTime(status);
|
|
(void)d;
|
|
if (status == U_ILLEGAL_ARGUMENT_ERROR) {
|
|
logln("Info: U_ILLEGAL_ARGUMENT_ERROR, because 5777 Adar I 1 is not a valid date.");
|
|
} else {
|
|
errln("Fail: U_ILLEGAL_ARGUMENT_ERROR should be set for input date 5777 Adar I 1.");
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestWeekData() {
|
|
// Each line contains two locales using the same set of week rule data.
|
|
const char* LOCALE_PAIRS[] = {
|
|
"en", "en_US",
|
|
"de", "de_DE",
|
|
"de_DE", "en_DE",
|
|
"en_GB", "und_GB",
|
|
"ar_EG", "en_EG",
|
|
"ar_SA", "fr_SA",
|
|
nullptr
|
|
};
|
|
|
|
UErrorCode status;
|
|
|
|
for (int32_t i = 0; LOCALE_PAIRS[i] != nullptr; i += 2) {
|
|
status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal1(Calendar::createInstance(LOCALE_PAIRS[i], status));
|
|
LocalPointer<Calendar> cal2(Calendar::createInstance(LOCALE_PAIRS[i + 1], status));
|
|
TEST_CHECK_STATUS_LOCALE(LOCALE_PAIRS[i]);
|
|
|
|
// First day of week
|
|
UCalendarDaysOfWeek dow1 = cal1->getFirstDayOfWeek(status);
|
|
UCalendarDaysOfWeek dow2 = cal2->getFirstDayOfWeek(status);
|
|
TEST_CHECK_STATUS;
|
|
TEST_ASSERT(dow1 == dow2);
|
|
|
|
// Minimum days in first week
|
|
uint8_t minDays1 = cal1->getMinimalDaysInFirstWeek();
|
|
uint8_t minDays2 = cal2->getMinimalDaysInFirstWeek();
|
|
TEST_ASSERT(minDays1 == minDays2);
|
|
|
|
// Weekdays and Weekends
|
|
for (int32_t d = UCAL_SUNDAY; d <= UCAL_SATURDAY; d++) {
|
|
status = U_ZERO_ERROR;
|
|
UCalendarWeekdayType wdt1 = cal1->getDayOfWeekType(static_cast<UCalendarDaysOfWeek>(d), status);
|
|
UCalendarWeekdayType wdt2 = cal2->getDayOfWeekType(static_cast<UCalendarDaysOfWeek>(d), status);
|
|
TEST_CHECK_STATUS;
|
|
TEST_ASSERT(wdt1 == wdt2);
|
|
}
|
|
}
|
|
}
|
|
|
|
typedef struct {
|
|
const char* zone;
|
|
const CalFields base;
|
|
int32_t deltaDays;
|
|
UCalendarWallTimeOption skippedWTOpt;
|
|
const CalFields expected;
|
|
} TestAddAcrossZoneTransitionData;
|
|
|
|
static const TestAddAcrossZoneTransitionData AAZTDATA[] =
|
|
{
|
|
// Time zone Base wall time day(s) Skipped time options
|
|
// Expected wall time
|
|
|
|
// Add 1 day, from the date before DST transition
|
|
{"America/Los_Angeles", CalFields(2014,3,8,1,59,59,999), 1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,1,59,59,999)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,1,59,59,999), 1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,1,59,59,999)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,1,59,59,999), 1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,1,59,59,999)},
|
|
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,2,0,0,0), 1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,1,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,2,0,0,0), 1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,2,0,0,0), 1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,2,30,0,0), 1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,1,30,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,2,30,0,0), 1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,3,30,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,2,30,0,0), 1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,3,0,0,0), 1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,3,0,0,0), 1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,8,3,0,0,0), 1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
// Subtract 1 day, from one day after DST transition
|
|
{"America/Los_Angeles", CalFields(2014,3,10,1,59,59,999), -1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,1,59,59,999)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,1,59,59,999), -1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,1,59,59,999)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,1,59,59,999), -1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,1,59,59,999)},
|
|
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,2,0,0,0), -1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,1,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,2,0,0,0), -1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,2,0,0,0), -1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,2,30,0,0), -1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,1,30,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,2,30,0,0), -1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,3,30,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,2,30,0,0), -1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,3,0,0,0), -1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,3,0,0,0), -1, UCAL_WALLTIME_LAST,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
{"America/Los_Angeles", CalFields(2014,3,10,3,0,0,0), -1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2014,3,9,3,0,0,0)},
|
|
|
|
|
|
// Test case for ticket#10544
|
|
{"America/Santiago", CalFields(2013,4,27,0,0,0,0), 134, UCAL_WALLTIME_FIRST,
|
|
CalFields(2013,9,7,23,0,0,0)},
|
|
|
|
{"America/Santiago", CalFields(2013,4,27,0,0,0,0), 134, UCAL_WALLTIME_LAST,
|
|
CalFields(2013,9,8,1,0,0,0)},
|
|
|
|
{"America/Santiago", CalFields(2013,4,27,0,0,0,0), 134, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2013,9,8,1,0,0,0)},
|
|
|
|
|
|
{"America/Santiago", CalFields(2013,4,27,0,30,0,0), 134, UCAL_WALLTIME_FIRST,
|
|
CalFields(2013,9,7,23,30,0,0)},
|
|
|
|
{"America/Santiago", CalFields(2013,4,27,0,30,0,0), 134, UCAL_WALLTIME_LAST,
|
|
CalFields(2013,9,8,1,30,0,0)},
|
|
|
|
{"America/Santiago", CalFields(2013,4,27,0,30,0,0), 134, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2013,9,8,1,0,0,0)},
|
|
|
|
|
|
// Extreme transition - Pacific/Apia completely skips 2011-12-30
|
|
{"Pacific/Apia", CalFields(2011,12,29,0,0,0,0), 1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2011,12,31,0,0,0,0)},
|
|
|
|
{"Pacific/Apia", CalFields(2011,12,29,0,0,0,0), 1, UCAL_WALLTIME_LAST,
|
|
CalFields(2011,12,31,0,0,0,0)},
|
|
|
|
{"Pacific/Apia", CalFields(2011,12,29,0,0,0,0), 1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2011,12,31,0,0,0,0)},
|
|
|
|
|
|
{"Pacific/Apia", CalFields(2011,12,31,12,0,0,0), -1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2011,12,29,12,0,0,0)},
|
|
|
|
{"Pacific/Apia", CalFields(2011,12,31,12,0,0,0), -1, UCAL_WALLTIME_LAST,
|
|
CalFields(2011,12,29,12,0,0,0)},
|
|
|
|
{"Pacific/Apia", CalFields(2011,12,31,12,0,0,0), -1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2011,12,29,12,0,0,0)},
|
|
|
|
|
|
// 30 minutes DST - Australia/Lord_Howe
|
|
{"Australia/Lord_Howe", CalFields(2013,10,5,2,15,0,0), 1, UCAL_WALLTIME_FIRST,
|
|
CalFields(2013,10,6,1,45,0,0)},
|
|
|
|
{"Australia/Lord_Howe", CalFields(2013,10,5,2,15,0,0), 1, UCAL_WALLTIME_LAST,
|
|
CalFields(2013,10,6,2,45,0,0)},
|
|
|
|
{"Australia/Lord_Howe", CalFields(2013,10,5,2,15,0,0), 1, UCAL_WALLTIME_NEXT_VALID,
|
|
CalFields(2013,10,6,2,30,0,0)},
|
|
|
|
{nullptr, CalFields(0,0,0,0,0,0,0), 0, UCAL_WALLTIME_LAST, CalFields(0,0,0,0,0,0,0)}
|
|
};
|
|
|
|
void CalendarTest::TestAddAcrossZoneTransition() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar cal(status);
|
|
TEST_CHECK_STATUS;
|
|
|
|
for (int32_t i = 0; AAZTDATA[i].zone; i++) {
|
|
status = U_ZERO_ERROR;
|
|
TimeZone *tz = TimeZone::createTimeZone(AAZTDATA[i].zone);
|
|
cal.adoptTimeZone(tz);
|
|
cal.setSkippedWallTimeOption(AAZTDATA[i].skippedWTOpt);
|
|
AAZTDATA[i].base.setTo(cal);
|
|
cal.add(UCAL_DATE, AAZTDATA[i].deltaDays, status);
|
|
TEST_CHECK_STATUS;
|
|
|
|
if (!AAZTDATA[i].expected.isEquivalentTo(cal, status)) {
|
|
CalFields res(cal, status);
|
|
TEST_CHECK_STATUS;
|
|
char buf[32];
|
|
const char *optDisp = AAZTDATA[i].skippedWTOpt == UCAL_WALLTIME_FIRST ? "FIRST" :
|
|
AAZTDATA[i].skippedWTOpt == UCAL_WALLTIME_LAST ? "LAST" : "NEXT_VALID";
|
|
dataerrln(UnicodeString("Error: base:") + AAZTDATA[i].base.toString(buf, sizeof(buf)) + ", tz:" + AAZTDATA[i].zone
|
|
+ ", delta:" + AAZTDATA[i].deltaDays + " day(s), opt:" + optDisp
|
|
+ ", result:" + res.toString(buf, sizeof(buf))
|
|
+ " - expected:" + AAZTDATA[i].expected.toString(buf, sizeof(buf)));
|
|
}
|
|
}
|
|
}
|
|
|
|
// Data in a separate file (Gregorian to Chinese lunar map)
|
|
#define INCLUDED_FROM_CALTEST_CPP
|
|
#include "caltestdata.h"
|
|
|
|
void CalendarTest::TestChineseCalendarMapping() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<TimeZone> zone(TimeZone::createTimeZone(UnicodeString("China")));
|
|
Locale locEnCalGregory = Locale::createFromName("en@calendar=gregorian");
|
|
Locale locEnCalChinese = Locale::createFromName("en@calendar=chinese");
|
|
LocalPointer<Calendar> calGregory(Calendar::createInstance(zone->clone(), locEnCalGregory, status));
|
|
LocalPointer<Calendar> calChinese(Calendar::createInstance(zone.orphan(), locEnCalChinese, status));
|
|
if ( U_FAILURE(status) ) {
|
|
errln("Fail: Calendar::createInstance fails for en with calendar=gregorian or calendar=chinese: %s", u_errorName(status));
|
|
} else {
|
|
const GregoToLunar * mapPtr = gregoToLunar; // in "caltestdata.h" included above
|
|
calGregory->clear();
|
|
calChinese->clear();
|
|
for (; mapPtr->gyr != 0; mapPtr++) {
|
|
status = U_ZERO_ERROR;
|
|
calGregory->set(mapPtr->gyr, mapPtr->gmo - 1, mapPtr->gda, 8, 0);
|
|
UDate date = calGregory->getTime(status);
|
|
calChinese->setTime(date, status);
|
|
if ( U_FAILURE(status) ) {
|
|
errln("Fail: for Gregorian %4d-%02d-%02d, calGregory->getTime or calChinese->setTime reports: %s",
|
|
mapPtr->gyr, mapPtr->gmo, mapPtr->gda, u_errorName(status));
|
|
continue;
|
|
}
|
|
int32_t era = calChinese->get(UCAL_ERA, status);
|
|
int32_t yr = calChinese->get(UCAL_YEAR, status);
|
|
int32_t mo = calChinese->get(UCAL_MONTH, status) + 1;
|
|
int32_t lp = calChinese->get(UCAL_IS_LEAP_MONTH, status);
|
|
int32_t da = calChinese->get(UCAL_DATE, status);
|
|
if ( U_FAILURE(status) ) {
|
|
errln("Fail: for Gregorian %4d-%02d-%02d, calChinese->get (for era, yr, mo, leapmo, da) reports: %s",
|
|
mapPtr->gyr, mapPtr->gmo, mapPtr->gda, u_errorName(status));
|
|
continue;
|
|
}
|
|
if (yr != mapPtr->cyr || mo != mapPtr->cmo || lp != mapPtr->clp || da != mapPtr->cda) {
|
|
errln("Fail: for Gregorian %4d-%02d-%02d, expected Chinese %2d-%02d(%d)-%02d, got %2d-%02d(%d)-%02d",
|
|
mapPtr->gyr, mapPtr->gmo, mapPtr->gda, mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, yr, mo, lp, da);
|
|
continue;
|
|
}
|
|
// If Grego->Chinese worked, try reverse mapping
|
|
calChinese->set(UCAL_ERA, era);
|
|
calChinese->set(UCAL_YEAR, mapPtr->cyr);
|
|
calChinese->set(UCAL_MONTH, mapPtr->cmo - 1);
|
|
calChinese->set(UCAL_IS_LEAP_MONTH, mapPtr->clp);
|
|
calChinese->set(UCAL_DATE, mapPtr->cda);
|
|
calChinese->set(UCAL_HOUR_OF_DAY, 8);
|
|
date = calChinese->getTime(status);
|
|
calGregory->setTime(date, status);
|
|
if ( U_FAILURE(status) ) {
|
|
errln("Fail: for Chinese %2d-%02d(%d)-%02d, calChinese->getTime or calGregory->setTime reports: %s",
|
|
mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, u_errorName(status));
|
|
continue;
|
|
}
|
|
yr = calGregory->get(UCAL_YEAR, status);
|
|
mo = calGregory->get(UCAL_MONTH, status) + 1;
|
|
da = calGregory->get(UCAL_DATE, status);
|
|
if ( U_FAILURE(status) ) {
|
|
errln("Fail: for Chinese %2d-%02d(%d)-%02d, calGregory->get (for yr, mo, da) reports: %s",
|
|
mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, u_errorName(status));
|
|
continue;
|
|
}
|
|
if (yr != mapPtr->gyr || mo != mapPtr->gmo || da != mapPtr->gda) {
|
|
errln("Fail: for Chinese %2d-%02d(%d)-%02d, Gregorian %4d-%02d-%02d, got %4d-%02d-%02d",
|
|
mapPtr->cyr, mapPtr->cmo, mapPtr->clp, mapPtr->cda, mapPtr->gyr, mapPtr->gmo, mapPtr->gda, yr, mo, da);
|
|
continue;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestClearMonth() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale::getRoot(), status));
|
|
if (failure(status, "construct Calendar")) return;
|
|
cal->set(2023, UCAL_JUNE, 29);
|
|
assertEquals("Calendar::get(UCAL_MONTH)", UCAL_JUNE, cal->get(UCAL_MONTH, status));
|
|
if (failure(status, "Calendar::get(UCAL_MONTH)")) return;
|
|
cal->clear(UCAL_MONTH);
|
|
assertEquals("Calendar::isSet(UCAL_MONTH) after clear(UCAL_MONTH)", false, !!cal->isSet(UCAL_MONTH));
|
|
assertEquals("Calendar::get(UCAL_MONTH after clear(UCAL_MONTH))", UCAL_JANUARY, cal->get(UCAL_MONTH, status));
|
|
if (failure(status, "Calendar::get(UCAL_MONTH)")) return;
|
|
|
|
cal->set(UCAL_ORDINAL_MONTH, 7);
|
|
assertEquals("Calendar::get(UCAL_MONTH) after set(UCAL_ORDINAL_MONTH, 7)", UCAL_AUGUST, cal->get(UCAL_MONTH, status));
|
|
if (failure(status, "Calendar::get(UCAL_MONTH) after set(UCAL_ORDINAL_MONTH, 7)")) return;
|
|
assertEquals("Calendar::get(UCAL_ORDINAL_MONTH) after set(UCAL_ORDINAL_MONTH, 7)", 7, cal->get(UCAL_ORDINAL_MONTH, status));
|
|
if (failure(status, "Calendar::get(UCAL_ORDINAL_MONTH) after set(UCAL_ORDINAL_MONTH, 7)")) return;
|
|
|
|
cal->clear(UCAL_ORDINAL_MONTH);
|
|
assertEquals("Calendar::isSet(UCAL_ORDINAL_MONTH) after clear(UCAL_ORDINAL_MONTH)", false, !!cal->isSet(UCAL_ORDINAL_MONTH));
|
|
assertEquals("Calendar::get(UCAL_MONTH) after clear(UCAL_ORDINAL_MONTH)", UCAL_JANUARY, cal->get(UCAL_MONTH, status));
|
|
if (failure(status, "Calendar::get(UCAL_MONTH) after clear(UCAL_ORDINAL_MONTH)")) return;
|
|
assertEquals("Calendar::get(UCAL_ORDINAL_MONTH) after clear(UCAL_ORDINAL_MONTH)", 0, cal->get(UCAL_ORDINAL_MONTH, status));
|
|
if (failure(status, "Calendar::get(UCAL_ORDINAL_MONTH) after clear(UCAL_ORDINAL_MONTH)")) return;
|
|
|
|
}
|
|
|
|
void CalendarTest::TestGregorianCalendarInTemporalLeapYear() {
|
|
// test from year 1800 to 2500
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct GregorianCalendar")) return;
|
|
for (int32_t year = 1900; year < 2400; ++year) {
|
|
gc.set(year, UCAL_MARCH, 7);
|
|
assertEquals("Calendar::inTemporalLeapYear",
|
|
gc.isLeapYear(year) == true, gc.inTemporalLeapYear(status) == true);
|
|
if (failure(status, "inTemporalLeapYear")) return;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::RunChineseCalendarInTemporalLeapYearTest(Calendar* cal) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct GregorianCalendar")) return;
|
|
LocalPointer<Calendar> leapTest(cal->clone());
|
|
// Start our test from 1900, Jan 1.
|
|
// Check every 29 days in exhausted mode.
|
|
int32_t incrementDays = 29;
|
|
int32_t startYear = 1900;
|
|
int32_t stopYear = 2400;
|
|
|
|
if (quick) {
|
|
incrementDays = 317;
|
|
stopYear = 2100;
|
|
}
|
|
int32_t yearForHasLeapMonth = -1;
|
|
bool hasLeapMonth = false;
|
|
for (gc.set(startYear, UCAL_JANUARY, 1);
|
|
gc.get(UCAL_YEAR, status) <= stopYear;
|
|
gc.add(UCAL_DATE, incrementDays, status)) {
|
|
cal->setTime(gc.getTime(status), status);
|
|
if (failure(status, "add/get/set/getTime/setTime incorrect")) return;
|
|
|
|
int32_t cal_year = cal->get(UCAL_EXTENDED_YEAR, status);
|
|
if (yearForHasLeapMonth != cal_year) {
|
|
leapTest->set(UCAL_EXTENDED_YEAR, cal_year);
|
|
leapTest->set(UCAL_MONTH, 0);
|
|
leapTest->set(UCAL_DATE, 1);
|
|
// seek any leap month
|
|
// check any leap month in the next 12 months.
|
|
for (hasLeapMonth = false;
|
|
(!hasLeapMonth) && cal_year == leapTest->get(UCAL_EXTENDED_YEAR, status);
|
|
leapTest->add(UCAL_MONTH, 1, status)) {
|
|
hasLeapMonth = leapTest->get(UCAL_IS_LEAP_MONTH, status) != 0;
|
|
}
|
|
yearForHasLeapMonth = cal_year;
|
|
}
|
|
if (failure(status, "error while figure out expectation")) return;
|
|
|
|
bool actualInLeap = cal->inTemporalLeapYear(status);
|
|
if (failure(status, "inTemporalLeapYear")) return;
|
|
if (hasLeapMonth != actualInLeap) {
|
|
logln("Gregorian y=%d m=%d d=%d => cal y=%d m=%s%d d=%d expected:%s actual:%s\n",
|
|
gc.get(UCAL_YEAR, status),
|
|
gc.get(UCAL_MONTH, status),
|
|
gc.get(UCAL_DATE, status),
|
|
cal->get(UCAL_EXTENDED_YEAR, status),
|
|
cal->get(UCAL_IS_LEAP_MONTH, status) == 1 ? "L" : "",
|
|
cal->get(UCAL_MONTH, status),
|
|
cal->get(UCAL_DAY_OF_MONTH, status),
|
|
hasLeapMonth ? "true" : "false",
|
|
actualInLeap ? "true" : "false");
|
|
}
|
|
assertEquals("inTemporalLeapYear", hasLeapMonth, actualInLeap);
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestChineseCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "chinese", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct ChineseCalendar")) return;
|
|
RunChineseCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestDangiCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "dangi", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct DangiCalendar")) return;
|
|
RunChineseCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestHebrewCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "hebrew", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
LocalPointer<Calendar> leapTest(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct HebrewCalendar")) return;
|
|
// Start our test from 1900, Jan 1.
|
|
// Check every 29 days in exhausted mode.
|
|
int32_t incrementDays = 29;
|
|
int32_t startYear = 1900;
|
|
int32_t stopYear = 2400;
|
|
|
|
if (quick) {
|
|
incrementDays = 317;
|
|
stopYear = 2100;
|
|
}
|
|
int32_t yearForHasLeapMonth = -1;
|
|
bool hasLeapMonth = false;
|
|
for (gc.set(startYear, UCAL_JANUARY, 1);
|
|
gc.get(UCAL_YEAR, status) <= stopYear;
|
|
gc.add(UCAL_DATE, incrementDays, status)) {
|
|
cal->setTime(gc.getTime(status), status);
|
|
if (failure(status, "add/get/set/getTime/setTime incorrect")) return;
|
|
|
|
int32_t cal_year = cal->get(UCAL_EXTENDED_YEAR, status);
|
|
if (yearForHasLeapMonth != cal_year) {
|
|
leapTest->set(UCAL_EXTENDED_YEAR, cal_year);
|
|
leapTest->set(UCAL_MONTH, 0);
|
|
leapTest->set(UCAL_DATE, 1);
|
|
// If 10 months after TISHRI is TAMUZ, then it is a leap year.
|
|
leapTest->add(UCAL_MONTH, 10, status);
|
|
hasLeapMonth = leapTest->get(UCAL_MONTH, status) == icu::HebrewCalendar::TAMUZ;
|
|
yearForHasLeapMonth = cal_year;
|
|
}
|
|
bool actualInLeap = cal->inTemporalLeapYear(status);
|
|
if (failure(status, "inTemporalLeapYear")) return;
|
|
if (hasLeapMonth != actualInLeap) {
|
|
logln("Gregorian y=%d m=%d d=7 => cal y=%d m=%d d=%d expected:%s actual:%s\n",
|
|
gc.get(UCAL_YEAR, status),
|
|
gc.get(UCAL_MONTH, status),
|
|
cal->get(UCAL_EXTENDED_YEAR, status),
|
|
cal->get(UCAL_MONTH, status),
|
|
cal->get(UCAL_DAY_OF_MONTH, status),
|
|
hasLeapMonth ? "true" : "false",
|
|
actualInLeap ? "true" : "false");
|
|
}
|
|
assertEquals("inTemporalLeapYear", hasLeapMonth, actualInLeap);
|
|
}
|
|
}
|
|
|
|
void CalendarTest::RunIslamicCalendarInTemporalLeapYearTest(Calendar* cal) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct GregorianCalendar")) return;
|
|
// Start our test from 1900, Jan 1.
|
|
// Check every 29 days in exhausted mode.
|
|
int32_t incrementDays = 29;
|
|
int32_t startYear = 1900;
|
|
int32_t stopYear = 2400;
|
|
|
|
if (quick) {
|
|
incrementDays = 317;
|
|
stopYear = 2100;
|
|
}
|
|
int32_t yearForHasLeapMonth = -1;
|
|
bool hasLeapMonth = false;
|
|
for (gc.set(startYear, UCAL_JANUARY, 1);
|
|
gc.get(UCAL_YEAR, status) <= stopYear;
|
|
gc.add(UCAL_DATE, incrementDays, status)) {
|
|
cal->setTime(gc.getTime(status), status);
|
|
if (failure(status, "set/getTime/setTime incorrect")) return;
|
|
int32_t cal_year = cal->get(UCAL_EXTENDED_YEAR, status);
|
|
if (yearForHasLeapMonth != cal_year) {
|
|
// If that year has exactly 355 days, it is a leap year.
|
|
hasLeapMonth = cal->getActualMaximum(UCAL_DAY_OF_YEAR, status) == 355;
|
|
yearForHasLeapMonth = cal_year;
|
|
}
|
|
|
|
bool actualInLeap = cal->inTemporalLeapYear(status);
|
|
if (failure(status, "inTemporalLeapYear")) return;
|
|
if (hasLeapMonth != actualInLeap) {
|
|
logln("Gregorian y=%d m=%d d=%d => cal y=%d m=%s%d d=%d expected:%s actual:%s\n",
|
|
gc.get(UCAL_EXTENDED_YEAR, status),
|
|
gc.get(UCAL_MONTH, status),
|
|
gc.get(UCAL_DAY_OF_MONTH, status),
|
|
cal->get(UCAL_EXTENDED_YEAR, status),
|
|
cal->get(UCAL_IS_LEAP_MONTH, status) == 1 ? "L" : "",
|
|
cal->get(UCAL_MONTH, status),
|
|
cal->get(UCAL_DAY_OF_MONTH, status),
|
|
hasLeapMonth ? "true" : "false",
|
|
actualInLeap ? "true" : "false");
|
|
}
|
|
assertEquals("inTemporalLeapYear", hasLeapMonth, actualInLeap);
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestIslamicCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "islamic", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct IslamicCalendar")) return;
|
|
RunIslamicCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestIslamicCivilCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "islamic-civil", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct IslamicCivilCalendar")) return;
|
|
RunIslamicCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestIslamicUmalquraCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "islamic-umalqura", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct IslamicUmalquraCalendar")) return;
|
|
RunIslamicCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestIslamicRGSACalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "islamic-rgsa", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct IslamicRGSACalendar")) return;
|
|
RunIslamicCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestIslamicTBLACalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "islamic-tbla", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct IslamicTBLACalendar")) return;
|
|
RunIslamicCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(Calendar* cal) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct GregorianCalendar")) return;
|
|
// Start our test from 1900, Jan 1.
|
|
// Check every 29 days in exhausted mode.
|
|
int32_t incrementDays = 29;
|
|
int32_t startYear = 1900;
|
|
int32_t stopYear = 2400;
|
|
|
|
if (quick) {
|
|
incrementDays = 317;
|
|
stopYear = 2100;
|
|
}
|
|
int32_t yearForHasLeapMonth = -1;
|
|
bool hasLeapMonth = false;
|
|
for (gc.set(startYear, UCAL_JANUARY, 1);
|
|
gc.get(UCAL_YEAR, status) <= stopYear;
|
|
gc.add(UCAL_DATE, incrementDays, status)) {
|
|
cal->setTime(gc.getTime(status), status);
|
|
if (failure(status, "set/getTime/setTime incorrect")) return;
|
|
int32_t cal_year = cal->get(UCAL_EXTENDED_YEAR, status);
|
|
if (yearForHasLeapMonth != cal_year) {
|
|
// If that year has exactly 355 days, it is a leap year.
|
|
hasLeapMonth = cal->getActualMaximum(UCAL_DAY_OF_YEAR, status) == 366;
|
|
if (failure(status, "getActualMaximum incorrect")) return;
|
|
yearForHasLeapMonth = cal_year;
|
|
}
|
|
bool actualInLeap = cal->inTemporalLeapYear(status);
|
|
if (failure(status, "inTemporalLeapYear")) return;
|
|
if (hasLeapMonth != actualInLeap) {
|
|
logln("Gregorian y=%d m=%d d=%d => cal y=%d m=%s%d d=%d expected:%s actual:%s\n",
|
|
gc.get(UCAL_EXTENDED_YEAR, status),
|
|
gc.get(UCAL_MONTH, status),
|
|
gc.get(UCAL_DAY_OF_MONTH, status),
|
|
cal->get(UCAL_EXTENDED_YEAR, status),
|
|
cal->get(UCAL_IS_LEAP_MONTH, status) == 1 ? "L" : "",
|
|
cal->get(UCAL_MONTH, status),
|
|
cal->get(UCAL_DAY_OF_MONTH, status),
|
|
hasLeapMonth ? "true" : "false",
|
|
actualInLeap ? "true" : "false");
|
|
}
|
|
assertEquals("inTemporalLeapYear", hasLeapMonth, actualInLeap);
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestTaiwanCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "roc", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct TaiwanCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestJapaneseCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "japanese", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct JapaneseCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestBuddhistCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "buddhist", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct BuddhistCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestPersianCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "persian", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct PersianCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestIndianCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "indian", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct IndianCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestCopticCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "coptic", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct CopticCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestEthiopicCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "ethiopic", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct EthiopicCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestEthiopicAmeteAlemCalendarInTemporalLeapYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "ethiopic-amete-alem", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct EthiopicAmeteAlemCalendar")) return;
|
|
Run366DaysIsLeapYearCalendarInTemporalLeapYearTest(cal.getAlias());
|
|
}
|
|
|
|
std::string monthCode(int32_t month, bool leap) {
|
|
std::string code("M");
|
|
if (month < 10) {
|
|
code += "0";
|
|
code += ('0' + month);
|
|
} else {
|
|
code += "1";
|
|
code += ('0' + (month % 10));
|
|
}
|
|
if (leap) {
|
|
code += "L";
|
|
}
|
|
return code;
|
|
}
|
|
std::string hebrewMonthCode(int32_t icuMonth) {
|
|
if (icuMonth == icu::HebrewCalendar::ADAR_1) {
|
|
return monthCode(icuMonth, true);
|
|
}
|
|
return monthCode(icuMonth < icu::HebrewCalendar::ADAR_1 ? icuMonth+1 : icuMonth, false);
|
|
}
|
|
|
|
void CalendarTest::RunChineseCalendarGetTemporalMonthCode(Calendar* cal) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct GregorianCalendar")) return;
|
|
// Start our test from 1900, Jan 1.
|
|
// Check every 29 days in exhausted mode.
|
|
int32_t incrementDays = 29;
|
|
int32_t startYear = 1900;
|
|
int32_t stopYear = 2400;
|
|
|
|
if (quick) {
|
|
incrementDays = 317;
|
|
startYear = 1950;
|
|
stopYear = 2050;
|
|
}
|
|
for (gc.set(startYear, UCAL_JANUARY, 1);
|
|
gc.get(UCAL_YEAR, status) <= stopYear;
|
|
gc.add(UCAL_DATE, incrementDays, status)) {
|
|
cal->setTime(gc.getTime(status), status);
|
|
if (failure(status, "set/getTime/setTime incorrect")) return;
|
|
int32_t cal_month = cal->get(UCAL_MONTH, status);
|
|
std::string expected = monthCode(
|
|
cal_month + 1, cal->get(UCAL_IS_LEAP_MONTH, status) != 0);
|
|
assertEquals("getTemporalMonthCode", expected.c_str(), cal->getTemporalMonthCode(status));
|
|
if (failure(status, "getTemporalMonthCode")) return;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestChineseCalendarGetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "chinese", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct ChineseCalendar")) return;
|
|
RunChineseCalendarGetTemporalMonthCode(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestDangiCalendarGetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "dangi", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct DangiCalendar")) return;
|
|
RunChineseCalendarGetTemporalMonthCode(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestHebrewCalendarGetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "hebrew", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct Calendar")) return;
|
|
// Start our test from 1900, Jan 1.
|
|
// Check every 29 days in exhausted mode.
|
|
int32_t incrementDays = 29;
|
|
int32_t startYear = 1900;
|
|
int32_t stopYear = 2400;
|
|
|
|
if (quick) {
|
|
incrementDays = 317;
|
|
stopYear = 2100;
|
|
}
|
|
for (gc.set(startYear, UCAL_JANUARY, 1);
|
|
gc.get(UCAL_YEAR, status) <= stopYear;
|
|
gc.add(UCAL_DATE, incrementDays, status)) {
|
|
cal->setTime(gc.getTime(status), status);
|
|
if (failure(status, "set/getTime/setTime incorrect")) return;
|
|
std::string expected = hebrewMonthCode(cal->get(UCAL_MONTH, status));
|
|
if (failure(status, "get failed")) return;
|
|
assertEquals("getTemporalMonthCode", expected.c_str(), cal->getTemporalMonthCode(status));
|
|
if (failure(status, "getTemporalMonthCode")) return;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::RunCECalendarGetTemporalMonthCode(Calendar* cal) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
if (failure(status, "construct Calendar")) return;
|
|
// Start testing from 1900
|
|
gc.set(1900, UCAL_JANUARY, 1);
|
|
cal->setTime(gc.getTime(status), status);
|
|
int32_t year = cal->get(UCAL_YEAR, status);
|
|
for (int32_t m = 0; m < 13; m++) {
|
|
std::string expected = monthCode(m+1, false);
|
|
for (int32_t y = year; y < year + 500 ; y++) {
|
|
cal->set(y, m, 1);
|
|
assertEquals("getTemporalMonthCode", expected.c_str(), cal->getTemporalMonthCode(status));
|
|
if (failure(status, "getTemporalMonthCode")) continue;
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
void CalendarTest::TestCopticCalendarGetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "coptic", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct CopticCalendar")) return;
|
|
RunCECalendarGetTemporalMonthCode(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestEthiopicCalendarGetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "ethiopic", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct EthiopicCalendar")) return;
|
|
RunCECalendarGetTemporalMonthCode(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestEthiopicAmeteAlemCalendarGetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "ethiopic-amete-alem", status);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct EthiopicAmeteAlemCalendar")) return;
|
|
RunCECalendarGetTemporalMonthCode(cal.getAlias());
|
|
}
|
|
|
|
void CalendarTest::TestGregorianCalendarSetTemporalMonthCode() {
|
|
|
|
struct TestCase {
|
|
int32_t gYear;
|
|
int32_t gMonth;
|
|
int32_t gDate;
|
|
const char* monthCode;
|
|
int32_t ordinalMonth;
|
|
} cases[] = {
|
|
{ 1911, UCAL_JANUARY, 31, "M01", 0 },
|
|
{ 1970, UCAL_FEBRUARY, 22, "M02", 1 },
|
|
{ 543, UCAL_MARCH, 3, "M03", 2 },
|
|
{ 2340, UCAL_APRIL, 21, "M04", 3 },
|
|
{ 1234, UCAL_MAY, 21, "M05", 4 },
|
|
{ 1931, UCAL_JUNE, 17, "M06", 5 },
|
|
{ 2000, UCAL_JULY, 1, "M07", 6 },
|
|
{ 2033, UCAL_AUGUST, 3, "M08", 7 },
|
|
{ 2013, UCAL_SEPTEMBER, 9, "M09", 8 },
|
|
{ 1849, UCAL_OCTOBER, 31, "M10", 9 },
|
|
{ 1433, UCAL_NOVEMBER, 30, "M11", 10 },
|
|
{ 2022, UCAL_DECEMBER, 25, "M12", 11 },
|
|
};
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc1(status);
|
|
GregorianCalendar gc2(status);
|
|
if (failure(status, "construct GregorianCalendar")) return;
|
|
for (auto& cas : cases) {
|
|
gc1.clear();
|
|
gc2.clear();
|
|
gc1.set(cas.gYear, cas.gMonth, cas.gDate);
|
|
|
|
gc2.set(UCAL_YEAR, cas.gYear);
|
|
gc2.setTemporalMonthCode(cas.monthCode, status);
|
|
gc2.set(UCAL_DATE, cas.gDate);
|
|
if (failure(status, "set/setTemporalMonthCode")) return;
|
|
|
|
assertTrue("by set and setTemporalMonthCode()", gc1.equals(gc2, status));
|
|
const char* actualMonthCode1 = gc1.getTemporalMonthCode(status);
|
|
const char* actualMonthCode2 = gc2.getTemporalMonthCode(status);
|
|
if (failure(status, "getTemporalMonthCode")) continue;
|
|
assertEquals("getTemporalMonthCode()", 0,
|
|
uprv_strcmp(actualMonthCode1, actualMonthCode2));
|
|
assertEquals("getTemporalMonthCode()", 0,
|
|
uprv_strcmp(cas.monthCode, actualMonthCode2));
|
|
assertEquals("ordinalMonth", cas.ordinalMonth, gc2.get(UCAL_ORDINAL_MONTH, status));
|
|
assertEquals("ordinalMonth", gc1.get(UCAL_ORDINAL_MONTH, status),
|
|
gc2.get(UCAL_ORDINAL_MONTH, status));
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestChineseCalendarSetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "chinese", status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct ChineseCalendar")) return;
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
|
|
struct TestCase {
|
|
int32_t gYear;
|
|
int32_t gMonth;
|
|
int32_t gDate;
|
|
int32_t cYear;
|
|
int32_t cMonth;
|
|
int32_t cDate;
|
|
const char* cMonthCode;
|
|
bool cLeapMonth;
|
|
int32_t cOrdinalMonth;
|
|
} cases[] = {
|
|
// https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2022.pdf
|
|
{ 2022, UCAL_DECEMBER, 15, 4659, UCAL_NOVEMBER, 22, "M11", false, 10},
|
|
// M01L is very hard to find. Cannot find a year has M01L in these several
|
|
// centuries.
|
|
// M02L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2004.pdf
|
|
{ 2004, UCAL_MARCH, 20, 4641, UCAL_FEBRUARY, 30, "M02", false, 1},
|
|
{ 2004, UCAL_MARCH, 21, 4641, UCAL_FEBRUARY, 1, "M02L", true, 2},
|
|
{ 2004, UCAL_APRIL, 18, 4641, UCAL_FEBRUARY, 29, "M02L", true, 2},
|
|
{ 2004, UCAL_APRIL, 19, 4641, UCAL_MARCH, 1, "M03", false, 3},
|
|
// M03L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/1995.pdf
|
|
{ 1955, UCAL_APRIL, 21, 4592, UCAL_MARCH, 29, "M03", false, 2},
|
|
{ 1955, UCAL_APRIL, 22, 4592, UCAL_MARCH, 1, "M03L", true, 3},
|
|
{ 1955, UCAL_MAY, 21, 4592, UCAL_MARCH, 30, "M03L", true, 3},
|
|
{ 1955, UCAL_MAY, 22, 4592, UCAL_APRIL, 1, "M04", false, 4},
|
|
// M12 https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/1996.pdf
|
|
{ 1956, UCAL_FEBRUARY, 11, 4592, UCAL_DECEMBER, 30, "M12", false, 12},
|
|
// M04L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2001.pdf
|
|
{ 2001, UCAL_MAY, 22, 4638, UCAL_APRIL, 30, "M04", false, 3},
|
|
{ 2001, UCAL_MAY, 23, 4638, UCAL_APRIL, 1, "M04L", true, 4},
|
|
{ 2001, UCAL_JUNE, 20, 4638, UCAL_APRIL, 29, "M04L", true, 4},
|
|
{ 2001, UCAL_JUNE, 21, 4638, UCAL_MAY, 1, "M05", false, 5},
|
|
// M05L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2009.pdf
|
|
{ 2009, UCAL_JUNE, 22, 4646, UCAL_MAY, 30, "M05", false, 4},
|
|
{ 2009, UCAL_JUNE, 23, 4646, UCAL_MAY, 1, "M05L", true, 5},
|
|
{ 2009, UCAL_JULY, 21, 4646, UCAL_MAY, 29, "M05L", true, 5},
|
|
{ 2009, UCAL_JULY, 22, 4646, UCAL_JUNE, 1, "M06", false, 6},
|
|
// M06L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2017.pdf
|
|
{ 2017, UCAL_JULY, 22, 4654, UCAL_JUNE, 29, "M06", false, 5},
|
|
{ 2017, UCAL_JULY, 23, 4654, UCAL_JUNE, 1, "M06L", true, 6},
|
|
{ 2017, UCAL_AUGUST, 21, 4654, UCAL_JUNE, 30, "M06L", true, 6},
|
|
{ 2017, UCAL_AUGUST, 22, 4654, UCAL_JULY, 1, "M07", false, 7},
|
|
// M07L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2006.pdf
|
|
{ 2006, UCAL_AUGUST, 23, 4643, UCAL_JULY, 30, "M07", false, 6},
|
|
{ 2006, UCAL_AUGUST, 24, 4643, UCAL_JULY, 1, "M07L", true, 7},
|
|
{ 2006, UCAL_SEPTEMBER, 21, 4643, UCAL_JULY, 29, "M07L", true, 7},
|
|
{ 2006, UCAL_SEPTEMBER, 22, 4643, UCAL_AUGUST, 1, "M08", false, 8},
|
|
// M08L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/1995.pdf
|
|
{ 1995, UCAL_SEPTEMBER, 24, 4632, UCAL_AUGUST, 30, "M08", false, 7},
|
|
{ 1995, UCAL_SEPTEMBER, 25, 4632, UCAL_AUGUST, 1, "M08L", true, 8},
|
|
{ 1995, UCAL_OCTOBER, 23, 4632, UCAL_AUGUST, 29, "M08L", true, 8},
|
|
{ 1995, UCAL_OCTOBER, 24, 4632, UCAL_SEPTEMBER, 1, "M09", false, 9},
|
|
// M09L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2014.pdf
|
|
{ 2014, UCAL_OCTOBER, 23, 4651, UCAL_SEPTEMBER, 30, "M09", false, 8},
|
|
{ 2014, UCAL_OCTOBER, 24, 4651, UCAL_SEPTEMBER, 1, "M09L", true, 9},
|
|
{ 2014, UCAL_NOVEMBER, 21, 4651, UCAL_SEPTEMBER, 29, "M09L", true, 9},
|
|
{ 2014, UCAL_NOVEMBER, 22, 4651, UCAL_OCTOBER, 1, "M10", false, 10},
|
|
// M10L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/1984.pdf
|
|
{ 1984, UCAL_NOVEMBER, 22, 4621, UCAL_OCTOBER, 30, "M10", false, 9},
|
|
{ 1984, UCAL_NOVEMBER, 23, 4621, UCAL_OCTOBER, 1, "M10L", true, 10},
|
|
{ 1984, UCAL_DECEMBER, 21, 4621, UCAL_OCTOBER, 29, "M10L", true, 10},
|
|
{ 1984, UCAL_DECEMBER, 22, 4621, UCAL_NOVEMBER, 1, "M11", false, 11},
|
|
// M11L https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2033.pdf
|
|
// https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2034.pdf
|
|
{ 2033, UCAL_DECEMBER, 21, 4670, UCAL_NOVEMBER, 30, "M11", false, 10},
|
|
{ 2033, UCAL_DECEMBER, 22, 4670, UCAL_NOVEMBER, 1, "M11L", true, 11},
|
|
{ 2034, UCAL_JANUARY, 19, 4670, UCAL_NOVEMBER, 29, "M11L", true, 11},
|
|
{ 2034, UCAL_JANUARY, 20, 4670, UCAL_DECEMBER, 1, "M12", false, 12},
|
|
// M12L is very hard to find. Cannot find a year has M01L in these several
|
|
// centuries.
|
|
};
|
|
GregorianCalendar gc1(status);
|
|
if (failure(status, "construct Calendar")) return;
|
|
for (auto& cas : cases) {
|
|
gc1.clear();
|
|
cc1->clear();
|
|
cc2->clear();
|
|
gc1.set(cas.gYear, cas.gMonth, cas.gDate);
|
|
cc1->setTime(gc1.getTime(status), status);
|
|
|
|
cc2->set(UCAL_EXTENDED_YEAR, cas.cYear);
|
|
cc2->setTemporalMonthCode(cas.cMonthCode, status);
|
|
cc2->set(UCAL_DATE, cas.cDate);
|
|
|
|
assertEquals("year", cas.cYear, cc1->get(UCAL_EXTENDED_YEAR, status));
|
|
assertEquals("month", cas.cMonth, cc1->get(UCAL_MONTH, status));
|
|
assertEquals("date", cas.cDate, cc1->get(UCAL_DATE, status));
|
|
assertEquals("is_leap_month", cas.cLeapMonth ? 1 : 0, cc1->get(UCAL_IS_LEAP_MONTH, status));
|
|
assertEquals("getTemporalMonthCode()", 0,
|
|
uprv_strcmp(cas.cMonthCode, cc1->getTemporalMonthCode(status)));
|
|
if (failure(status, "getTemporalMonthCode")) continue;
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc1->get(UCAL_ORDINAL_MONTH, status));
|
|
if (! cc2->equals(*cc1, status)) {
|
|
printf("g=%f %f vs %f. diff = %f %d/%d%s/%d vs %d/%d%s/%d\n",
|
|
gc1.getTime(status), cc1->getTime(status), cc2->getTime(status),
|
|
cc1->getTime(status) - cc2->getTime(status),
|
|
cc1->get(UCAL_EXTENDED_YEAR, status),
|
|
cc1->get(UCAL_MONTH, status)+1,
|
|
cc1->get(UCAL_IS_LEAP_MONTH, status) == 0 ? "" : "L",
|
|
cc1->get(UCAL_DATE, status),
|
|
cc2->get(UCAL_EXTENDED_YEAR, status),
|
|
cc2->get(UCAL_MONTH, status)+1,
|
|
cc2->get(UCAL_IS_LEAP_MONTH, status) == 0 ? "" : "L",
|
|
cc2->get(UCAL_DATE, status));
|
|
}
|
|
assertTrue("by set() and setTemporalMonthCode()", cc2->equals(*cc1, status));
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestHebrewCalendarSetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "hebrew", status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct HebrewCalendar")) return;
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
|
|
struct TestCase {
|
|
int32_t gYear;
|
|
int32_t gMonth;
|
|
int32_t gDate;
|
|
int32_t cYear;
|
|
int32_t cMonth;
|
|
int32_t cDate;
|
|
const char* cMonthCode;
|
|
int32_t cOrdinalMonth;
|
|
} cases[] = {
|
|
{ 2022, UCAL_JANUARY, 11, 5782, icu::HebrewCalendar::SHEVAT, 9, "M05", 4},
|
|
{ 2022, UCAL_FEBRUARY, 12, 5782, icu::HebrewCalendar::ADAR_1, 11, "M05L", 5},
|
|
{ 2022, UCAL_MARCH, 13, 5782, icu::HebrewCalendar::ADAR, 10, "M06", 6},
|
|
{ 2022, UCAL_APRIL, 14, 5782, icu::HebrewCalendar::NISAN, 13, "M07", 7},
|
|
{ 2022, UCAL_MAY, 15, 5782, icu::HebrewCalendar::IYAR, 14, "M08", 8},
|
|
{ 2022, UCAL_JUNE, 16, 5782, icu::HebrewCalendar::SIVAN, 17, "M09", 9},
|
|
{ 2022, UCAL_JULY, 17, 5782, icu::HebrewCalendar::TAMUZ, 18, "M10", 10},
|
|
{ 2022, UCAL_AUGUST, 18, 5782, icu::HebrewCalendar::AV, 21, "M11", 11},
|
|
{ 2022, UCAL_SEPTEMBER, 19, 5782, icu::HebrewCalendar::ELUL, 23, "M12", 12},
|
|
{ 2022, UCAL_OCTOBER, 20, 5783, icu::HebrewCalendar::TISHRI, 25, "M01", 0},
|
|
{ 2022, UCAL_NOVEMBER, 21, 5783, icu::HebrewCalendar::HESHVAN, 27, "M02", 1},
|
|
{ 2022, UCAL_DECEMBER, 22, 5783, icu::HebrewCalendar::KISLEV, 28, "M03", 2},
|
|
{ 2023, UCAL_JANUARY, 20, 5783, icu::HebrewCalendar::TEVET, 27, "M04", 3},
|
|
};
|
|
GregorianCalendar gc1(status);
|
|
if (failure(status, "construct Calendar")) return;
|
|
for (auto& cas : cases) {
|
|
gc1.clear();
|
|
cc1->clear();
|
|
cc2->clear();
|
|
gc1.set(cas.gYear, cas.gMonth, cas.gDate);
|
|
cc1->setTime(gc1.getTime(status), status);
|
|
|
|
cc2->set(UCAL_EXTENDED_YEAR, cas.cYear);
|
|
cc2->setTemporalMonthCode(cas.cMonthCode, status);
|
|
cc2->set(UCAL_DATE, cas.cDate);
|
|
|
|
assertEquals("year", cas.cYear, cc1->get(UCAL_EXTENDED_YEAR, status));
|
|
assertEquals("month", cas.cMonth, cc1->get(UCAL_MONTH, status));
|
|
assertEquals("date", cas.cDate, cc1->get(UCAL_DATE, status));
|
|
assertEquals("getTemporalMonthCode()", 0,
|
|
uprv_strcmp(cas.cMonthCode, cc1->getTemporalMonthCode(status)));
|
|
if (failure(status, "getTemporalMonthCode")) continue;
|
|
if (! cc2->equals(*cc1, status)) {
|
|
printf("g=%f %f vs %f. diff = %f %d/%d/%d vs %d/%d/%d\n",
|
|
gc1.getTime(status), cc1->getTime(status), cc2->getTime(status),
|
|
cc1->getTime(status) - cc2->getTime(status),
|
|
cc1->get(UCAL_EXTENDED_YEAR, status),
|
|
cc1->get(UCAL_MONTH, status)+1,
|
|
cc1->get(UCAL_DATE, status),
|
|
cc2->get(UCAL_EXTENDED_YEAR, status),
|
|
cc2->get(UCAL_MONTH, status)+1,
|
|
cc2->get(UCAL_DATE, status));
|
|
}
|
|
assertTrue("by set() and setTemporalMonthCode()", cc2->equals(*cc1, status));
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc1->get(UCAL_ORDINAL_MONTH, status));
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc2->get(UCAL_ORDINAL_MONTH, status));
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestCopticCalendarSetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "coptic", status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct CopticCalendar")) return;
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
|
|
struct TestCase {
|
|
int32_t gYear;
|
|
int32_t gMonth;
|
|
int32_t gDate;
|
|
int32_t cYear;
|
|
int32_t cMonth;
|
|
int32_t cDate;
|
|
const char* cMonthCode;
|
|
int32_t cOrdinalMonth;
|
|
} cases[] = {
|
|
{ 1900, UCAL_JANUARY, 1, 1616, icu::CopticCalendar::KIAHK, 23, "M04", 3},
|
|
{ 1900, UCAL_SEPTEMBER, 6, 1616, icu::CopticCalendar::NASIE, 1, "M13", 12},
|
|
{ 1900, UCAL_SEPTEMBER, 10, 1616, icu::CopticCalendar::NASIE, 5, "M13", 12},
|
|
{ 1900, UCAL_SEPTEMBER, 11, 1617, icu::CopticCalendar::TOUT, 1, "M01", 0},
|
|
|
|
{ 2022, UCAL_JANUARY, 11, 1738, icu::CopticCalendar::TOBA, 3, "M05", 4},
|
|
{ 2022, UCAL_FEBRUARY, 12, 1738, icu::CopticCalendar::AMSHIR, 5, "M06", 5},
|
|
{ 2022, UCAL_MARCH, 13, 1738, icu::CopticCalendar::BARAMHAT, 4, "M07", 6},
|
|
{ 2022, UCAL_APRIL, 14, 1738, icu::CopticCalendar::BARAMOUDA, 6, "M08", 7},
|
|
{ 2022, UCAL_MAY, 15, 1738, icu::CopticCalendar::BASHANS, 7, "M09", 8},
|
|
{ 2022, UCAL_JUNE, 16, 1738, icu::CopticCalendar::PAONA, 9, "M10", 9},
|
|
{ 2022, UCAL_JULY, 17, 1738, icu::CopticCalendar::EPEP, 10, "M11", 10},
|
|
{ 2022, UCAL_AUGUST, 18, 1738, icu::CopticCalendar::MESRA, 12, "M12", 11},
|
|
{ 2022, UCAL_SEPTEMBER, 6, 1738, icu::CopticCalendar::NASIE, 1, "M13", 12},
|
|
{ 2022, UCAL_SEPTEMBER, 10, 1738, icu::CopticCalendar::NASIE, 5, "M13", 12},
|
|
{ 2022, UCAL_SEPTEMBER, 11, 1739, icu::CopticCalendar::TOUT, 1, "M01", 0},
|
|
{ 2022, UCAL_SEPTEMBER, 19, 1739, icu::CopticCalendar::TOUT, 9, "M01", 0},
|
|
{ 2022, UCAL_OCTOBER, 20, 1739, icu::CopticCalendar::BABA, 10, "M02", 1},
|
|
{ 2022, UCAL_NOVEMBER, 21, 1739, icu::CopticCalendar::HATOR, 12, "M03", 2},
|
|
{ 2022, UCAL_DECEMBER, 22, 1739, icu::CopticCalendar::KIAHK, 13, "M04", 3},
|
|
|
|
{ 2023, UCAL_JANUARY, 1, 1739, icu::CopticCalendar::KIAHK, 23, "M04", 3},
|
|
{ 2023, UCAL_SEPTEMBER, 6, 1739, icu::CopticCalendar::NASIE, 1, "M13", 12},
|
|
{ 2023, UCAL_SEPTEMBER, 11, 1739, icu::CopticCalendar::NASIE, 6, "M13", 12},
|
|
{ 2023, UCAL_SEPTEMBER, 12, 1740, icu::CopticCalendar::TOUT, 1, "M01", 0},
|
|
|
|
{ 2030, UCAL_JANUARY, 1, 1746, icu::CopticCalendar::KIAHK, 23, "M04", 3},
|
|
{ 2030, UCAL_SEPTEMBER, 6, 1746, icu::CopticCalendar::NASIE, 1, "M13", 12},
|
|
{ 2030, UCAL_SEPTEMBER, 10, 1746, icu::CopticCalendar::NASIE, 5, "M13", 12},
|
|
{ 2030, UCAL_SEPTEMBER, 11, 1747, icu::CopticCalendar::TOUT, 1, "M01", 0},
|
|
};
|
|
GregorianCalendar gc1(status);
|
|
if (failure(status, "construct Calendar")) return;
|
|
for (auto& cas : cases) {
|
|
gc1.clear();
|
|
cc1->clear();
|
|
cc2->clear();
|
|
gc1.set(cas.gYear, cas.gMonth, cas.gDate);
|
|
cc1->setTime(gc1.getTime(status), status);
|
|
|
|
cc2->set(UCAL_EXTENDED_YEAR, cas.cYear);
|
|
cc2->setTemporalMonthCode(cas.cMonthCode, status);
|
|
cc2->set(UCAL_DATE, cas.cDate);
|
|
|
|
assertEquals("year", cas.cYear, cc1->get(UCAL_EXTENDED_YEAR, status));
|
|
assertEquals("month", cas.cMonth, cc1->get(UCAL_MONTH, status));
|
|
assertEquals("date", cas.cDate, cc1->get(UCAL_DATE, status));
|
|
assertEquals("getTemporalMonthCode()", 0,
|
|
uprv_strcmp(cas.cMonthCode, cc1->getTemporalMonthCode(status)));
|
|
if (failure(status, "getTemporalMonthCode")) continue;
|
|
assertTrue("by set() and setTemporalMonthCode()", cc2->equals(*cc1, status));
|
|
if (! cc2->equals(*cc1, status)) {
|
|
printf("g=%f %f vs %f. diff = %f %d/%d/%d vs %d/%d/%d\n",
|
|
gc1.getTime(status), cc1->getTime(status), cc2->getTime(status),
|
|
cc1->getTime(status) - cc2->getTime(status),
|
|
cc1->get(UCAL_EXTENDED_YEAR, status),
|
|
cc1->get(UCAL_MONTH, status)+1,
|
|
cc1->get(UCAL_DATE, status),
|
|
cc2->get(UCAL_EXTENDED_YEAR, status),
|
|
cc2->get(UCAL_MONTH, status)+1,
|
|
cc2->get(UCAL_DATE, status));
|
|
}
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc1->get(UCAL_ORDINAL_MONTH, status));
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc2->get(UCAL_ORDINAL_MONTH, status));
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestEthiopicCalendarSetTemporalMonthCode() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "ethiopic", status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "construct EthiopicCalendar")) return;
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
|
|
struct TestCase {
|
|
int32_t gYear;
|
|
int32_t gMonth;
|
|
int32_t gDate;
|
|
int32_t cYear;
|
|
int32_t cMonth;
|
|
int32_t cDate;
|
|
const char* cMonthCode;
|
|
int32_t cOrdinalMonth;
|
|
} cases[] = {
|
|
{ 1900, UCAL_JANUARY, 1, 1892, icu::EthiopicCalendar::TAHSAS, 23, "M04", 3},
|
|
{ 1900, UCAL_SEPTEMBER, 6, 1892, icu::EthiopicCalendar::PAGUMEN, 1, "M13", 12},
|
|
{ 1900, UCAL_SEPTEMBER, 10, 1892, icu::EthiopicCalendar::PAGUMEN, 5, "M13", 12},
|
|
{ 1900, UCAL_SEPTEMBER, 11, 1893, icu::EthiopicCalendar::MESKEREM, 1, "M01", 0},
|
|
|
|
{ 2022, UCAL_JANUARY, 11, 2014, icu::EthiopicCalendar::TER, 3, "M05", 4},
|
|
{ 2022, UCAL_FEBRUARY, 12, 2014, icu::EthiopicCalendar::YEKATIT, 5, "M06", 5},
|
|
{ 2022, UCAL_MARCH, 13, 2014, icu::EthiopicCalendar::MEGABIT, 4, "M07", 6},
|
|
{ 2022, UCAL_APRIL, 14, 2014, icu::EthiopicCalendar::MIAZIA, 6, "M08", 7},
|
|
{ 2022, UCAL_MAY, 15, 2014, icu::EthiopicCalendar::GENBOT, 7, "M09", 8},
|
|
{ 2022, UCAL_JUNE, 16, 2014, icu::EthiopicCalendar::SENE, 9, "M10", 9},
|
|
{ 2022, UCAL_JULY, 17, 2014, icu::EthiopicCalendar::HAMLE, 10, "M11", 10},
|
|
{ 2022, UCAL_AUGUST, 18, 2014, icu::EthiopicCalendar::NEHASSE, 12, "M12", 11},
|
|
{ 2022, UCAL_SEPTEMBER, 6, 2014, icu::EthiopicCalendar::PAGUMEN, 1, "M13", 12},
|
|
{ 2022, UCAL_SEPTEMBER, 10, 2014, icu::EthiopicCalendar::PAGUMEN, 5, "M13", 12},
|
|
{ 2022, UCAL_SEPTEMBER, 11, 2015, icu::EthiopicCalendar::MESKEREM, 1, "M01", 0},
|
|
{ 2022, UCAL_SEPTEMBER, 19, 2015, icu::EthiopicCalendar::MESKEREM, 9, "M01", 0},
|
|
{ 2022, UCAL_OCTOBER, 20, 2015, icu::EthiopicCalendar::TEKEMT, 10, "M02", 1},
|
|
{ 2022, UCAL_NOVEMBER, 21, 2015, icu::EthiopicCalendar::HEDAR, 12, "M03", 2},
|
|
{ 2022, UCAL_DECEMBER, 22, 2015, icu::EthiopicCalendar::TAHSAS, 13, "M04", 3},
|
|
|
|
{ 2023, UCAL_JANUARY, 1, 2015, icu::EthiopicCalendar::TAHSAS, 23, "M04", 3},
|
|
{ 2023, UCAL_SEPTEMBER, 6, 2015, icu::EthiopicCalendar::PAGUMEN, 1, "M13", 12},
|
|
{ 2023, UCAL_SEPTEMBER, 11, 2015, icu::EthiopicCalendar::PAGUMEN, 6, "M13", 12},
|
|
{ 2023, UCAL_SEPTEMBER, 12, 2016, icu::EthiopicCalendar::MESKEREM, 1, "M01", 0},
|
|
|
|
{ 2030, UCAL_JANUARY, 1, 2022, icu::EthiopicCalendar::TAHSAS, 23, "M04", 3},
|
|
{ 2030, UCAL_SEPTEMBER, 6, 2022, icu::EthiopicCalendar::PAGUMEN, 1, "M13", 12},
|
|
{ 2030, UCAL_SEPTEMBER, 10, 2022, icu::EthiopicCalendar::PAGUMEN, 5, "M13", 12},
|
|
{ 2030, UCAL_SEPTEMBER, 11, 2023, icu::EthiopicCalendar::MESKEREM, 1, "M01", 0},
|
|
};
|
|
GregorianCalendar gc1(status);
|
|
if (failure(status, "construct Calendar")) return;
|
|
for (auto& cas : cases) {
|
|
gc1.clear();
|
|
cc1->clear();
|
|
cc2->clear();
|
|
gc1.set(cas.gYear, cas.gMonth, cas.gDate);
|
|
cc1->setTime(gc1.getTime(status), status);
|
|
|
|
cc2->set(UCAL_EXTENDED_YEAR, cas.cYear);
|
|
cc2->setTemporalMonthCode(cas.cMonthCode, status);
|
|
cc2->set(UCAL_DATE, cas.cDate);
|
|
|
|
assertEquals("year", cas.cYear, cc1->get(UCAL_EXTENDED_YEAR, status));
|
|
assertEquals("month", cas.cMonth, cc1->get(UCAL_MONTH, status));
|
|
assertEquals("date", cas.cDate, cc1->get(UCAL_DATE, status));
|
|
assertEquals("getTemporalMonthCode()", 0,
|
|
uprv_strcmp(cas.cMonthCode, cc1->getTemporalMonthCode(status)));
|
|
if (failure(status, "getTemporalMonthCode")) continue;
|
|
if (! cc2->equals(*cc1, status)) {
|
|
printf("g=%f %f vs %f. diff = %f %d/%d/%d vs %d/%d/%d\n",
|
|
gc1.getTime(status), cc1->getTime(status), cc2->getTime(status),
|
|
cc1->getTime(status) - cc2->getTime(status),
|
|
cc1->get(UCAL_EXTENDED_YEAR, status),
|
|
cc1->get(UCAL_MONTH, status)+1,
|
|
cc1->get(UCAL_DATE, status),
|
|
cc2->get(UCAL_EXTENDED_YEAR, status),
|
|
cc2->get(UCAL_MONTH, status)+1,
|
|
cc2->get(UCAL_DATE, status));
|
|
}
|
|
assertTrue("by set() and setTemporalMonthCode()", cc2->equals(*cc1, status));
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc1->get(UCAL_ORDINAL_MONTH, status));
|
|
assertEquals("ordinalMonth", cas.cOrdinalMonth, cc2->get(UCAL_ORDINAL_MONTH, status));
|
|
}
|
|
}
|
|
|
|
void VerifyMonth(CalendarTest* test, const char* message, Calendar* cc, int32_t expectedMonth,
|
|
int32_t expectedOrdinalMonth, bool expectedLeapMonth,
|
|
const char* expectedMonthCode) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
std::string buf(message);
|
|
buf.append(" get(UCAL_MONTH)");
|
|
test->assertEquals(buf.c_str(), expectedMonth, cc->get(UCAL_MONTH, status));
|
|
buf = message;
|
|
buf.append(" get(UCAL_ORDINAL_MONTH)");
|
|
test->assertEquals(buf.c_str(), expectedOrdinalMonth, cc->get(UCAL_ORDINAL_MONTH, status));
|
|
buf = message;
|
|
buf.append(" get(UCAL_IS_LEAP_MONTH)");
|
|
test->assertEquals(buf.c_str(), expectedLeapMonth ? 1 : 0, cc->get(UCAL_IS_LEAP_MONTH, status));
|
|
buf = message;
|
|
buf.append(" getTemporalMonthCode()");
|
|
test->assertTrue(buf.c_str(), uprv_strcmp(cc->getTemporalMonthCode(status), expectedMonthCode) == 0);
|
|
}
|
|
|
|
void CalendarTest::TestMostCalendarsOrdinalMonthSet() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
std::unique_ptr<icu::StringEnumeration> enumeration(
|
|
Calendar::getKeywordValuesForLocale("calendar", l, false, status));
|
|
for (const char* name = enumeration->next(nullptr, status);
|
|
U_SUCCESS(status) && name != nullptr;
|
|
name = enumeration->next(nullptr, status)) {
|
|
|
|
// Test these three calendars differently.
|
|
if (uprv_strcmp(name, "chinese") == 0) continue;
|
|
if (uprv_strcmp(name, "dangi") == 0) continue;
|
|
if (uprv_strcmp(name, "hebrew") == 0) continue;
|
|
|
|
l.setKeywordValue("calendar", name, status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
LocalPointer<Calendar> cc3(cc1->clone());
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, 2134);
|
|
cc2->set(UCAL_EXTENDED_YEAR, 2134);
|
|
cc3->set(UCAL_EXTENDED_YEAR, 2134);
|
|
cc1->set(UCAL_MONTH, 5);
|
|
cc2->set(UCAL_ORDINAL_MONTH, 5);
|
|
cc3->setTemporalMonthCode("M06", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc1->set(UCAL_DATE, 23);
|
|
cc2->set(UCAL_DATE, 23);
|
|
cc3->set(UCAL_DATE, 23);
|
|
assertTrue("M06 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("M06 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), 5, 5, false, "M06");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), 5, 5, false, "M06");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), 5, 5, false, "M06");
|
|
|
|
cc1->set(UCAL_ORDINAL_MONTH, 6);
|
|
cc2->setTemporalMonthCode("M07", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc3->set(UCAL_MONTH, 6);
|
|
assertTrue("M07 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("M07 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), 6, 6, false, "M07");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), 6, 6, false, "M07");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), 6, 6, false, "M07");
|
|
|
|
cc1->setTemporalMonthCode("M08", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc2->set(UCAL_MONTH, 7);
|
|
cc3->set(UCAL_ORDINAL_MONTH, 7);
|
|
assertTrue("M08 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("M08 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), 7, 7, false, "M08");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), 7, 7, false, "M08");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), 7, 7, false, "M08");
|
|
|
|
cc1->set(UCAL_DATE, 3);
|
|
// For "M13", do not return error for these three calendars.
|
|
if ((uprv_strcmp(name, "coptic") == 0) ||
|
|
(uprv_strcmp(name, "ethiopic") == 0) ||
|
|
(uprv_strcmp(name, "ethiopic-amete-alem") == 0)) {
|
|
|
|
cc1->setTemporalMonthCode("M13", status);
|
|
assertEquals("setTemporalMonthCode(\"M13\")", U_ZERO_ERROR, status);
|
|
assertEquals("get(UCAL_MONTH) after setTemporalMonthCode(\"M13\")",
|
|
12, cc1->get(UCAL_MONTH, status));
|
|
assertEquals("get(UCAL_ORDINAL_MONTH) after setTemporalMonthCode(\"M13\")",
|
|
12, cc1->get(UCAL_ORDINAL_MONTH, status));
|
|
assertEquals("get", U_ZERO_ERROR, status);
|
|
} else {
|
|
cc1->setTemporalMonthCode("M13", status);
|
|
assertEquals("setTemporalMonthCode(\"M13\")", U_ILLEGAL_ARGUMENT_ERROR, status);
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
|
|
// Out of bound monthCodes should return error.
|
|
// These are not valid for calendar do not have a leap month
|
|
const char* kInvalidMonthCodes[] = {
|
|
"M00", "M14", "M01L", "M02L", "M03L", "M04L", "M05L", "M06L", "M07L",
|
|
"M08L", "M09L", "M10L", "M11L", "M12L"};
|
|
for (auto& cas : kInvalidMonthCodes) {
|
|
cc1->setTemporalMonthCode(cas, status);
|
|
assertEquals("setTemporalMonthCode(\"M13\")", U_ILLEGAL_ARGUMENT_ERROR, status);
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestChineseCalendarOrdinalMonthSet() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "chinese", status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
LocalPointer<Calendar> cc3(cc1->clone());
|
|
|
|
constexpr int32_t notLeapYear = 4591;
|
|
constexpr int32_t leapMarchYear = 4592;
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
|
|
cc1->set(UCAL_MONTH, UCAL_MARCH); cc1->set(UCAL_IS_LEAP_MONTH, 1);
|
|
cc2->set(UCAL_ORDINAL_MONTH, 3);
|
|
cc3->setTemporalMonthCode("M03L", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc1->set(UCAL_DATE, 1);
|
|
cc2->set(UCAL_DATE, 1);
|
|
cc3->set(UCAL_DATE, 1);
|
|
assertTrue("4592 M03L cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4592 M03L cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "4592 M03L cc1", cc1.getAlias(), UCAL_MARCH, 3, true, "M03L");
|
|
VerifyMonth(this, "4592 M03L cc2", cc2.getAlias(), UCAL_MARCH, 3, true, "M03L");
|
|
VerifyMonth(this, "4592 M03L cc3", cc3.getAlias(), UCAL_MARCH, 3, true, "M03L");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc1->set(UCAL_ORDINAL_MONTH, 5);
|
|
cc2->setTemporalMonthCode("M06", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc3->set(UCAL_MONTH, UCAL_JUNE); cc3->set(UCAL_IS_LEAP_MONTH, 0);
|
|
assertTrue("4591 M06 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4591 M06 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "4591 M06 cc1", cc1.getAlias(), UCAL_JUNE, 5, false, "M06");
|
|
VerifyMonth(this, "4591 M06 cc2", cc2.getAlias(), UCAL_JUNE, 5, false, "M06");
|
|
VerifyMonth(this, "4591 M06 cc3", cc3.getAlias(), UCAL_JUNE, 5, false, "M06");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc1->setTemporalMonthCode("M04", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc2->set(UCAL_MONTH, UCAL_APRIL); cc2->set(UCAL_IS_LEAP_MONTH, 0);
|
|
cc3->set(UCAL_ORDINAL_MONTH, 4);
|
|
assertTrue("4592 M04 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4592 M04 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
// 4592 has leap March so April is the 5th month in that year.
|
|
VerifyMonth(this, "4592 M04 cc1", cc1.getAlias(), UCAL_APRIL, 4, false, "M04");
|
|
VerifyMonth(this, "4592 M04 cc2", cc2.getAlias(), UCAL_APRIL, 4, false, "M04");
|
|
VerifyMonth(this, "4592 M04 cc3", cc3.getAlias(), UCAL_APRIL, 4, false, "M04");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
assertTrue("4591 M04 no leap month before cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4591 M04 no leap month before cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
// 4592 has no leap month before April so April is the 4th month in that year.
|
|
VerifyMonth(this, "4591 M04 cc1", cc1.getAlias(), UCAL_APRIL, 3, false, "M04");
|
|
VerifyMonth(this, "4591 M04 cc2", cc2.getAlias(), UCAL_APRIL, 3, false, "M04");
|
|
VerifyMonth(this, "4591 M04 cc3", cc3.getAlias(), UCAL_APRIL, 3, false, "M04");
|
|
|
|
// Out of bound monthCodes should return error.
|
|
// These are not valid for calendar do not have a leap month
|
|
UErrorCode expectedStatus = U_ILLEGAL_ARGUMENT_ERROR;
|
|
const char* kInvalidMonthCodes[] = { "M00", "M13", "M14" };
|
|
|
|
|
|
for (auto& cas : kInvalidMonthCodes) {
|
|
cc1->setTemporalMonthCode(cas, status);
|
|
if (status != expectedStatus) {
|
|
errln("setTemporalMonthCode(%s) should return U_ILLEGAL_ARGUMENT_ERROR", cas);
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestDangiCalendarOrdinalMonthSet() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
const char* name = "dangi";
|
|
l.setKeywordValue("calendar", name, status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
LocalPointer<Calendar> cc3(cc1->clone());
|
|
|
|
constexpr int32_t notLeapYear = 4287;
|
|
constexpr int32_t leapMarchYear = 4288;
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
|
|
cc1->set(UCAL_MONTH, UCAL_MARCH); cc1->set(UCAL_IS_LEAP_MONTH, 1);
|
|
cc2->set(UCAL_ORDINAL_MONTH, 3);
|
|
cc3->setTemporalMonthCode("M03L", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc1->set(UCAL_DATE, 1);
|
|
cc2->set(UCAL_DATE, 1);
|
|
cc3->set(UCAL_DATE, 1);
|
|
assertTrue("4288 M03L cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4288 M03L cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "4288 M03L cc1", cc1.getAlias(), UCAL_MARCH, 3, true, "M03L");
|
|
VerifyMonth(this, "4288 M03L cc2", cc2.getAlias(), UCAL_MARCH, 3, true, "M03L");
|
|
VerifyMonth(this, "4288 M03L cc3", cc3.getAlias(), UCAL_MARCH, 3, true, "M03L");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc1->set(UCAL_ORDINAL_MONTH, 5);
|
|
cc2->setTemporalMonthCode("M06", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc3->set(UCAL_MONTH, UCAL_JUNE); cc3->set(UCAL_IS_LEAP_MONTH, 0);
|
|
assertTrue("4287 M06 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4287 M06 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "4287 M06 cc1", cc1.getAlias(), UCAL_JUNE, 5, false, "M06");
|
|
VerifyMonth(this, "4287 M06 cc2", cc2.getAlias(), UCAL_JUNE, 5, false, "M06");
|
|
VerifyMonth(this, "4287 M06 cc3", cc3.getAlias(), UCAL_JUNE, 5, false, "M06");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, leapMarchYear);
|
|
cc1->setTemporalMonthCode("M04", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc2->set(UCAL_MONTH, UCAL_APRIL); cc2->set(UCAL_IS_LEAP_MONTH, 0);
|
|
cc3->set(UCAL_ORDINAL_MONTH, 4);
|
|
assertTrue("4288 M04 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4288 M04 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
// 4592 has leap March so April is the 5th month in that year.
|
|
VerifyMonth(this, "4288 M04 cc1", cc1.getAlias(), UCAL_APRIL, 4, false, "M04");
|
|
VerifyMonth(this, "4288 M04 cc2", cc2.getAlias(), UCAL_APRIL, 4, false, "M04");
|
|
VerifyMonth(this, "4288 M04 cc3", cc3.getAlias(), UCAL_APRIL, 4, false, "M04");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc2->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
cc3->set(UCAL_EXTENDED_YEAR, notLeapYear);
|
|
assertTrue("4287 M04 no leap month before cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("4287 M04 no leap month before cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
// 4592 has no leap month before April so April is the 4th month in that year.
|
|
VerifyMonth(this, "4287 M04 cc1", cc1.getAlias(), UCAL_APRIL, 3, false, "M04");
|
|
VerifyMonth(this, "4287 M04 cc2", cc2.getAlias(), UCAL_APRIL, 3, false, "M04");
|
|
VerifyMonth(this, "4287 M04 cc3", cc3.getAlias(), UCAL_APRIL, 3, false, "M04");
|
|
|
|
// Out of bound monthCodes should return error.
|
|
// These are not valid for calendar do not have a leap month
|
|
UErrorCode expectedStatus = U_ILLEGAL_ARGUMENT_ERROR;
|
|
const char* kInvalidMonthCodes[] = { "M00", "M13", "M14" };
|
|
|
|
for (auto& cas : kInvalidMonthCodes) {
|
|
cc1->setTemporalMonthCode(cas, status);
|
|
if (status != expectedStatus) {
|
|
errln("setTemporalMonthCode(%s) should return U_ILLEGAL_ARGUMENT_ERROR", cas);
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestHebrewCalendarOrdinalMonthSet() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
l.setKeywordValue("calendar", "hebrew", status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
LocalPointer<Calendar> cc3(cc1->clone());
|
|
|
|
// 5782 is leap year, 5781 is NOT.
|
|
cc1->set(UCAL_EXTENDED_YEAR, 5782);
|
|
cc2->set(UCAL_EXTENDED_YEAR, 5782);
|
|
cc3->set(UCAL_EXTENDED_YEAR, 5782);
|
|
cc1->set(UCAL_MONTH, icu::HebrewCalendar::ADAR_1);
|
|
cc2->set(UCAL_ORDINAL_MONTH, 5);
|
|
cc3->setTemporalMonthCode("M05L", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc1->set(UCAL_DATE, 1);
|
|
cc2->set(UCAL_DATE, 1);
|
|
cc3->set(UCAL_DATE, 1);
|
|
assertTrue("5782 M05L cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("5782 M05L cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), icu::HebrewCalendar::ADAR_1, 5, false, "M05L");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), icu::HebrewCalendar::ADAR_1, 5, false, "M05L");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), icu::HebrewCalendar::ADAR_1, 5, false, "M05L");
|
|
|
|
cc1->set(UCAL_ORDINAL_MONTH, 4);
|
|
cc2->setTemporalMonthCode("M05", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc3->set(UCAL_MONTH, icu::HebrewCalendar::SHEVAT);
|
|
assertTrue("5782 M05 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("5782 M05 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), icu::HebrewCalendar::SHEVAT, 4, false, "M05");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), icu::HebrewCalendar::SHEVAT, 4, false, "M05");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), icu::HebrewCalendar::SHEVAT, 4, false, "M05");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, 5781);
|
|
cc2->set(UCAL_EXTENDED_YEAR, 5781);
|
|
cc3->set(UCAL_EXTENDED_YEAR, 5781);
|
|
cc1->setTemporalMonthCode("M06", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc2->set(UCAL_MONTH, icu::HebrewCalendar::ADAR);
|
|
cc3->set(UCAL_ORDINAL_MONTH, 5);
|
|
assertTrue("5781 M06 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("5781 M06 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), icu::HebrewCalendar::ADAR, 5, false, "M06");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), icu::HebrewCalendar::ADAR, 5, false, "M06");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), icu::HebrewCalendar::ADAR, 5, false, "M06");
|
|
|
|
cc1->set(UCAL_EXTENDED_YEAR, 5782);
|
|
cc2->set(UCAL_EXTENDED_YEAR, 5782);
|
|
cc3->set(UCAL_EXTENDED_YEAR, 5782);
|
|
assertTrue("5782 M06 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("5782 M06 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), icu::HebrewCalendar::ADAR, 6, false, "M06");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), icu::HebrewCalendar::ADAR, 6, false, "M06");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), icu::HebrewCalendar::ADAR, 6, false, "M06");
|
|
|
|
cc1->set(UCAL_ORDINAL_MONTH, 7);
|
|
cc2->setTemporalMonthCode("M07", status);
|
|
if (failure(status, "setTemporalMonthCode failure")) return;
|
|
cc3->set(UCAL_MONTH, icu::HebrewCalendar::NISAN);
|
|
assertTrue("5782 M07 cc2==cc1 set month by UCAL_MONTH and UCAL_UCAL_ORDINAL_MONTH", cc2->equals(*cc1, status));
|
|
assertTrue("5782 M07 cc2==cc3 set month by UCAL_MONTH and setTemporalMonthCode", cc2->equals(*cc3, status));
|
|
if (failure(status, "equals failure")) return;
|
|
VerifyMonth(this, "cc1", cc1.getAlias(), icu::HebrewCalendar::NISAN, 7, false, "M07");
|
|
VerifyMonth(this, "cc2", cc2.getAlias(), icu::HebrewCalendar::NISAN, 7, false, "M07");
|
|
VerifyMonth(this, "cc3", cc3.getAlias(), icu::HebrewCalendar::NISAN, 7, false, "M07");
|
|
|
|
// Out of bound monthCodes should return error.
|
|
// These are not valid for calendar do not have a leap month
|
|
UErrorCode expectedStatus = U_ILLEGAL_ARGUMENT_ERROR;
|
|
const char* kInvalidMonthCodes[] = { "M00", "M13", "M14",
|
|
"M01L", "M02L", "M03L", "M04L",
|
|
/* M05L could be legal */
|
|
"M06L", "M07L", "M08L", "M09L", "M10L", "M11L", "M12L",
|
|
};
|
|
|
|
for (auto& cas : kInvalidMonthCodes) {
|
|
cc1->setTemporalMonthCode(cas, status);
|
|
if (status != expectedStatus) {
|
|
errln("setTemporalMonthCode(%s) should return U_ILLEGAL_ARGUMENT_ERROR", cas);
|
|
}
|
|
status = U_ZERO_ERROR;
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestCalendarAddOrdinalMonth() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
gc.set(2022, UCAL_DECEMBER, 16);
|
|
Locale l(Locale::getRoot());
|
|
std::unique_ptr<icu::StringEnumeration> enumeration(
|
|
Calendar::getKeywordValuesForLocale("calendar", l, false, status));
|
|
for (const char* name = enumeration->next(nullptr, status);
|
|
U_SUCCESS(status) && name != nullptr;
|
|
name = enumeration->next(nullptr, status)) {
|
|
|
|
l.setKeywordValue("calendar", name, status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
|
|
cc1->setTime(gc.getTime(status), status);
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
|
|
for (int i = 0; i < 8; i++) {
|
|
for (int j = 1; j < 8; j++) {
|
|
cc1->add(UCAL_MONTH, j, status);
|
|
cc2->add(UCAL_ORDINAL_MONTH, j, status);
|
|
if (failure(status, "add j")) return;
|
|
assertTrue("two add produce the same result", cc2->equals(*cc1, status));
|
|
}
|
|
for (int j = 1; j < 8; j++) {
|
|
cc1->add(UCAL_MONTH, -j, status);
|
|
cc2->add(UCAL_ORDINAL_MONTH, -j, status);
|
|
if (failure(status, "add -j")) return;
|
|
assertTrue("two add produce the same result", cc2->equals(*cc1, status));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestCalendarRollOrdinalMonth() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
gc.set(2022, UCAL_DECEMBER, 16);
|
|
Locale l(Locale::getRoot());
|
|
std::unique_ptr<icu::StringEnumeration> enumeration(
|
|
Calendar::getKeywordValuesForLocale("calendar", l, false, status));
|
|
for (const char* name = enumeration->next(nullptr, status);
|
|
U_SUCCESS(status) && name != nullptr;
|
|
name = enumeration->next(nullptr, status)) {
|
|
|
|
l.setKeywordValue("calendar", name, status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
|
|
cc1->setTime(gc.getTime(status), status);
|
|
LocalPointer<Calendar> cc2(cc1->clone());
|
|
|
|
for (int i = 0; i < 8; i++) {
|
|
for (int j = 1; j < 8; j++) {
|
|
cc1->roll(UCAL_MONTH, j, status);
|
|
cc2->roll(UCAL_ORDINAL_MONTH, j, status);
|
|
if (failure(status, "roll j")) return;
|
|
assertTrue("two add produce the same result", cc2->equals(*cc1, status));
|
|
}
|
|
for (int j = 1; j < 8; j++) {
|
|
cc1->roll(UCAL_MONTH, -j, status);
|
|
cc2->roll(UCAL_ORDINAL_MONTH, -j, status);
|
|
if (failure(status, "roll -j")) return;
|
|
assertTrue("two add produce the same result", cc2->equals(*cc1, status));
|
|
}
|
|
for (int j = 1; j < 3; j++) {
|
|
cc1->roll(UCAL_MONTH, true, status);
|
|
cc2->roll(UCAL_ORDINAL_MONTH, true, status);
|
|
if (failure(status, "roll true")) return;
|
|
assertTrue("two add produce the same result", cc2->equals(*cc1, status));
|
|
}
|
|
for (int j = 1; j < 3; j++) {
|
|
cc1->roll(UCAL_MONTH, false, status);
|
|
cc2->roll(UCAL_ORDINAL_MONTH, false, status);
|
|
if (failure(status, "roll false")) return;
|
|
assertTrue("two add produce the same result", cc2->equals(*cc1, status));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestLimitsOrdinalMonth() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
gc.set(2022, UCAL_DECEMBER, 16);
|
|
Locale l(Locale::getRoot());
|
|
std::unique_ptr<icu::StringEnumeration> enumeration(
|
|
Calendar::getKeywordValuesForLocale("calendar", l, false, status));
|
|
|
|
struct Expectation {
|
|
const char* calendar;
|
|
int32_t min;
|
|
int32_t max;
|
|
int32_t greatestMin;
|
|
int32_t leastMax;
|
|
} kExpectations[] = {
|
|
{ "gregorian", 0, 11, 0, 11 },
|
|
{ "japanese", 0, 11, 0, 11 },
|
|
{ "buddhist", 0, 11, 0, 11 },
|
|
{ "roc", 0, 11, 0, 11 },
|
|
{ "persian", 0, 11, 0, 11 },
|
|
{ "islamic-civil", 0, 11, 0, 11 },
|
|
{ "islamic", 0, 11, 0, 11 },
|
|
{ "hebrew", 0, 12, 0, 11 },
|
|
{ "chinese", 0, 12, 0, 11 },
|
|
{ "indian", 0, 11, 0, 11 },
|
|
{ "coptic", 0, 12, 0, 12 },
|
|
{ "ethiopic", 0, 12, 0, 12 },
|
|
{ "ethiopic-amete-alem", 0, 12, 0, 12 },
|
|
{ "iso8601", 0, 11, 0, 11 },
|
|
{ "dangi", 0, 12, 0, 11 },
|
|
{ "islamic-umalqura", 0, 11, 0, 11 },
|
|
{ "islamic-tbla", 0, 11, 0, 11 },
|
|
{ "islamic-rgsa", 0, 11, 0, 11 },
|
|
};
|
|
|
|
for (const char* name = enumeration->next(nullptr, status);
|
|
U_SUCCESS(status) && name != nullptr;
|
|
name = enumeration->next(nullptr, status)) {
|
|
l.setKeywordValue("calendar", name, status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
bool found = false;
|
|
for (auto& exp : kExpectations) {
|
|
if (uprv_strcmp(exp.calendar, name) == 0) {
|
|
found = true;
|
|
assertEquals("getMinimum(UCAL_ORDINAL_MONTH)",
|
|
exp.min, cc1->getMinimum(UCAL_ORDINAL_MONTH));
|
|
assertEquals("getMaximum(UCAL_ORDINAL_MONTH)",
|
|
exp.max, cc1->getMaximum(UCAL_ORDINAL_MONTH));
|
|
assertEquals("getMinimum(UCAL_ORDINAL_MONTH)",
|
|
exp.greatestMin, cc1->getGreatestMinimum(UCAL_ORDINAL_MONTH));
|
|
assertEquals("getMinimum(UCAL_ORDINAL_MONTH)",
|
|
exp.leastMax, cc1->getLeastMaximum(UCAL_ORDINAL_MONTH));
|
|
break;
|
|
}
|
|
}
|
|
if (!found) {
|
|
errln("Cannot find expectation");
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::TestMaxActualLimitsWithoutGet23006() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
gc.set(2025, UCAL_AUGUST, 8);
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=chinese"), status), status);
|
|
cal->setTime(gc.getTime(status), status);
|
|
int32_t beforeCallingGet = cal->getActualMaximum(UCAL_DAY_OF_MONTH, status);
|
|
cal->get(UCAL_DAY_OF_MONTH, status);
|
|
int32_t afterCallingGet = cal->getActualMaximum(UCAL_DAY_OF_MONTH, status);
|
|
assertEquals("getActualMaximum() should return same value before/after calling get()",
|
|
beforeCallingGet, afterCallingGet);
|
|
assertEquals("getActualMaximum() should return 29 before calling get()",
|
|
29, beforeCallingGet);
|
|
|
|
gc.set(2026, UCAL_AUGUST, 8);
|
|
cal->setTime(gc.getTime(status), status);
|
|
beforeCallingGet = cal->getActualMaximum(UCAL_DAY_OF_MONTH, status);
|
|
cal->get(UCAL_DAY_OF_MONTH, status);
|
|
afterCallingGet = cal->getActualMaximum(UCAL_DAY_OF_MONTH, status);
|
|
assertEquals("getActualMaximum() should return same value before/after calling get()",
|
|
beforeCallingGet, afterCallingGet);
|
|
assertEquals("getActualMaximum() should return 29 before calling get()",
|
|
30, beforeCallingGet);
|
|
}
|
|
void CalendarTest::TestActualLimitsOrdinalMonth() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
gc.set(2022, UCAL_DECEMBER, 16);
|
|
Locale l(Locale::getRoot());
|
|
std::unique_ptr<icu::StringEnumeration> enumeration(
|
|
Calendar::getKeywordValuesForLocale("calendar", l, false, status));
|
|
|
|
struct TestCases {
|
|
const char* calendar;
|
|
int32_t extended_year;
|
|
int32_t actualMinOrdinalMonth;
|
|
int32_t actualMaxOrdinalMonth;
|
|
} cases[] = {
|
|
{ "gregorian", 2021, 0, 11 },
|
|
{ "gregorian", 2022, 0, 11 },
|
|
{ "gregorian", 2023, 0, 11 },
|
|
{ "japanese", 2021, 0, 11 },
|
|
{ "japanese", 2022, 0, 11 },
|
|
{ "japanese", 2023, 0, 11 },
|
|
{ "buddhist", 2021, 0, 11 },
|
|
{ "buddhist", 2022, 0, 11 },
|
|
{ "buddhist", 2023, 0, 11 },
|
|
{ "roc", 2021, 0, 11 },
|
|
{ "roc", 2022, 0, 11 },
|
|
{ "roc", 2023, 0, 11 },
|
|
{ "persian", 1400, 0, 11 },
|
|
{ "persian", 1401, 0, 11 },
|
|
{ "persian", 1402, 0, 11 },
|
|
{ "hebrew", 5782, 0, 12 },
|
|
{ "hebrew", 5783, 0, 11 },
|
|
{ "hebrew", 5789, 0, 11 },
|
|
{ "hebrew", 5790, 0, 12 },
|
|
{ "chinese", 4645, 0, 11 },
|
|
{ "chinese", 4646, 0, 12 },
|
|
{ "chinese", 4647, 0, 11 },
|
|
{ "dangi", 4645 + 304, 0, 11 },
|
|
{ "dangi", 4646 + 304, 0, 12 },
|
|
{ "dangi", 4647 + 304, 0, 11 },
|
|
{ "indian", 1944, 0, 11 },
|
|
{ "indian", 1945, 0, 11 },
|
|
{ "indian", 1946, 0, 11 },
|
|
{ "coptic", 1737, 0, 12 },
|
|
{ "coptic", 1738, 0, 12 },
|
|
{ "coptic", 1739, 0, 12 },
|
|
{ "ethiopic", 2013, 0, 12 },
|
|
{ "ethiopic", 2014, 0, 12 },
|
|
{ "ethiopic", 2015, 0, 12 },
|
|
{ "ethiopic-amete-alem", 2014, 0, 12 },
|
|
{ "ethiopic-amete-alem", 2015, 0, 12 },
|
|
{ "ethiopic-amete-alem", 2016, 0, 12 },
|
|
{ "iso8601", 2022, 0, 11 },
|
|
{ "islamic-civil", 1443, 0, 11 },
|
|
{ "islamic-civil", 1444, 0, 11 },
|
|
{ "islamic-civil", 1445, 0, 11 },
|
|
{ "islamic", 1443, 0, 11 },
|
|
{ "islamic", 1444, 0, 11 },
|
|
{ "islamic", 1445, 0, 11 },
|
|
{ "islamic-umalqura", 1443, 0, 11 },
|
|
{ "islamic-umalqura", 1444, 0, 11 },
|
|
{ "islamic-umalqura", 1445, 0, 11 },
|
|
{ "islamic-tbla", 1443, 0, 11 },
|
|
{ "islamic-tbla", 1444, 0, 11 },
|
|
{ "islamic-tbla", 1445, 0, 11 },
|
|
{ "islamic-rgsa", 1443, 0, 11 },
|
|
{ "islamic-rgsa", 1444, 0, 11 },
|
|
{ "islamic-rgsa", 1445, 0, 11 },
|
|
};
|
|
|
|
for (auto& cas : cases) {
|
|
l.setKeywordValue("calendar", cas.calendar, status);
|
|
LocalPointer<Calendar> cc1(Calendar::createInstance(l, status));
|
|
if (failure(status, "Construct Calendar")) return;
|
|
cc1->set(UCAL_EXTENDED_YEAR, cas.extended_year);
|
|
cc1->set(UCAL_ORDINAL_MONTH, 0);
|
|
cc1->set(UCAL_DATE, 1);
|
|
assertEquals("getActualMinimum(UCAL_ORDINAL_MONTH)",
|
|
cas.actualMinOrdinalMonth, cc1->getActualMinimum(UCAL_ORDINAL_MONTH, status));
|
|
assertEquals("getActualMaximum(UCAL_ORDINAL_MONTH)",
|
|
cas.actualMaxOrdinalMonth, cc1->getActualMaximum(UCAL_ORDINAL_MONTH, status));
|
|
}
|
|
}
|
|
|
|
// The Lunar year which majorty part fall into 1889 and the early part of 1890
|
|
// should have no leap months, but currently ICU calculate and show there is
|
|
// a Leap month after the 12th month and before the first month of the Chinese
|
|
// Calendar which overlapping most of the 1890 year in Gregorian.
|
|
//
|
|
// We use the value from
|
|
// https://ytliu0.github.io/ChineseCalendar/table_period.html?period=qing
|
|
// and https://ytliu0.github.io/ChineseCalendar/index_chinese.html
|
|
// as the expected value. The same results were given by many several other
|
|
// sites not just his one.
|
|
//
|
|
// There should be a Leap month after the 2nd month of the Chinese Calendar year
|
|
// mostly overlapping with 1890 and should have no leap month in the Chinese
|
|
// Calendar year mostly overlapping with 1889.
|
|
void CalendarTest::TestChineseCalendarMonthInSpecialYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
GregorianCalendar gc(status);
|
|
ChineseCalendar cal(Locale::getRoot(), status);
|
|
if (failure(status, "Constructor failed")) return;
|
|
struct TestCase {
|
|
int32_t gyear;
|
|
int32_t gmonth;
|
|
int32_t gdate;
|
|
int32_t cmonth; // 0-based month number: 1st month = 0, 2nd month = 1.
|
|
int32_t cdate;
|
|
bool cleapmonth;
|
|
} cases[] = {
|
|
// Gregorian Chinese Calendar
|
|
// First some recent date
|
|
// From https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2022.pdf
|
|
{ 2022, UCAL_DECEMBER, 15, 11-1, 22, false},
|
|
// ^-- m-1 to convert to 0-based month from 1-based.
|
|
// From https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2023.pdf
|
|
{ 2023, UCAL_MARCH, 21, 2-1, 30, false},
|
|
{ 2023, UCAL_MARCH, 22, 2-1, 1, true},
|
|
// We know there are some problematic year, especially those involved
|
|
// the rare cases of M11L and M12L.
|
|
// Check 1890 and 2033.
|
|
//
|
|
// 2033 has M11L
|
|
// From https://www.hko.gov.hk/tc/gts/time/calendar/pdf/files/2033.pdf
|
|
{ 2033, UCAL_DECEMBER, 21, 11-1, 30, false},
|
|
{ 2033, UCAL_DECEMBER, 22, 11-1, 1, true},
|
|
// Here are the date we get from multiple external sources
|
|
// https://ytliu0.github.io/ChineseCalendar/index_chinese.html
|
|
// https://ytliu0.github.io/ChineseCalendar/table_period.html?period=qing
|
|
// There should have no leap 12th month in the year mostly overlapping
|
|
// 1889 but should have a leap 2th month in the year mostly overlapping
|
|
// with 1890.
|
|
{ 1890, UCAL_JANUARY, 1, 12-1, 11, false},
|
|
{ 1890, UCAL_JANUARY, 20, 12-1, 30, false},
|
|
{ 1890, UCAL_JANUARY, 21, 1-1, 1, false},
|
|
{ 1890, UCAL_FEBRUARY, 1, 1-1, 12, false},
|
|
{ 1890, UCAL_FEBRUARY, 19, 2-1, 1, false},
|
|
{ 1890, UCAL_MARCH, 1, 2-1, 11, false},
|
|
{ 1890, UCAL_MARCH, 21, 2-1, 1, true},
|
|
{ 1890, UCAL_APRIL, 1, 2-1, 12, true},
|
|
{ 1890, UCAL_APRIL, 18, 2-1, 29, true},
|
|
{ 1890, UCAL_APRIL, 19, 3-1, 1, false},
|
|
{ 1890, UCAL_APRIL, 20, 3-1, 2, false},
|
|
};
|
|
for (auto& cas : cases) {
|
|
gc.set(cas.gyear, cas.gmonth, cas.gdate);
|
|
cal.setTime(gc.getTime(status), status);
|
|
if (failure(status, "getTime/setTime failed")) return;
|
|
int32_t actual_month = cal.get(UCAL_MONTH, status);
|
|
int32_t actual_date = cal.get(UCAL_DATE, status);
|
|
int32_t actual_in_leap_month = cal.get(UCAL_IS_LEAP_MONTH, status);
|
|
if (failure(status, "get failed")) return;
|
|
if (cas.cmonth != actual_month ||
|
|
cas.cdate != actual_date ||
|
|
cas.cleapmonth != (actual_in_leap_month != 0)) {
|
|
if (cas.gyear == 1890 &&
|
|
logKnownIssue("ICU-22230", "Problem between 1890/1/21 and 1890/4/18")) {
|
|
continue;
|
|
}
|
|
errln("Fail: Gregorian(%d/%d/%d) should be Chinese %d%s/%d but got %d%s/%d",
|
|
cas.gyear, cas.gmonth+1, cas.gdate,
|
|
cas.cmonth+1, cas.cleapmonth ? "L" : "" , cas.cdate,
|
|
actual_month+1, ((actual_in_leap_month != 0) ? "L" : ""), actual_date );
|
|
}
|
|
}
|
|
}
|
|
|
|
// Test the stack will not overflow with dangi calendar during "roll".
|
|
void CalendarTest::TestDangiOverflowIsLeapMonthBetween22507() {
|
|
Locale locale("en@calendar=dangi");
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(
|
|
*TimeZone::getGMT(), locale, status));
|
|
cal->clear();
|
|
status = U_ZERO_ERROR;
|
|
cal->add(UCAL_MONTH, 1242972234, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->roll(UCAL_MONTH, 1249790538, status);
|
|
status = U_ZERO_ERROR;
|
|
// Without the fix, the stack will overflow during this roll().
|
|
cal->roll(UCAL_MONTH, 1246382666, status);
|
|
}
|
|
|
|
void CalendarTest::TestFWWithISO8601() {
|
|
// ICU UCAL_SUNDAY is 1, UCAL_MONDAY is 2, ... UCAL_SATURDAY is 7.
|
|
const char *locales[] = {
|
|
"",
|
|
"en-u-ca-iso8601-fw-sun",
|
|
"en-u-ca-iso8601-fw-mon",
|
|
"en-u-ca-iso8601-fw-tue",
|
|
"en-u-ca-iso8601-fw-wed",
|
|
"en-u-ca-iso8601-fw-thu",
|
|
"en-u-ca-iso8601-fw-fri",
|
|
"en-u-ca-iso8601-fw-sat"
|
|
};
|
|
for (int i = UCAL_SUNDAY; i <= UCAL_SATURDAY; i++) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
const char* locale = locales[i];
|
|
LocalPointer<Calendar> cal(
|
|
Calendar::createInstance(
|
|
Locale(locale), status), status);
|
|
if (failure(status, "Constructor failed")) continue;
|
|
std::string msg("Calendar::createInstance(\"");
|
|
msg = msg + locale + "\")->getFirstDayOfWeek()";
|
|
assertEquals(msg.c_str(), i, cal->getFirstDayOfWeek());
|
|
}
|
|
}
|
|
void CalendarTest::TestRollWeekOfYear() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l("zh_TW@calendar=chinese");
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status), status);
|
|
cal->set(UCAL_EXTENDED_YEAR, -1107626);
|
|
cal->set(UCAL_MONTH, UCAL_JANUARY);
|
|
cal->set(UCAL_DATE, 1);
|
|
cal->roll(UCAL_WEEK_OF_YEAR, 0x7fffff, status);
|
|
cal->roll(UCAL_WEEK_OF_YEAR, 1, status);
|
|
}
|
|
|
|
void CalendarTest::verifyFirstDayOfWeek(const char* locale, UCalendarDaysOfWeek expected) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l = Locale::forLanguageTag(locale, status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(l, status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
assertEquals(locale,
|
|
expected, cal->getFirstDayOfWeek(status));
|
|
U_ASSERT(U_SUCCESS(status));
|
|
}
|
|
|
|
/**
|
|
* Test "First Day Overrides" behavior
|
|
* https://unicode.org/reports/tr35/tr35-dates.html#first-day-overrides
|
|
* And data in <firstDay> of
|
|
* https://github.com/unicode-org/cldr/blob/main/common/supplemental/supplementalData.xml
|
|
*
|
|
* Examples of region for First Day of a week
|
|
* Friday: MV
|
|
* Saturday: AE AF
|
|
* Sunday: US JP
|
|
* Monday: GB
|
|
*/
|
|
void CalendarTest::TestFirstDayOfWeek() {
|
|
// Test -u-fw- value
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-fw-sun-rg-mvzzzz-sd-usca", UCAL_SUNDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-fw-mon-rg-mvzzzz-sd-usca", UCAL_MONDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-fw-tue-rg-mvzzzz-sd-usca", UCAL_TUESDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-fw-wed-rg-mvzzzz-sd-usca", UCAL_WEDNESDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-fw-thu-rg-mvzzzz-sd-usca", UCAL_THURSDAY);
|
|
verifyFirstDayOfWeek("en-AE-u-ca-iso8601-fw-fri-rg-aezzzz-sd-usca", UCAL_FRIDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-fw-sat-rg-mvzzzz-sd-usca", UCAL_SATURDAY);
|
|
|
|
// Test -u-rg- value
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-rg-mvzzzz-sd-usca", UCAL_FRIDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-rg-aezzzz-sd-usca", UCAL_MONDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-rg-uszzzz-sd-usca", UCAL_SUNDAY);
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-rg-gbzzzz-sd-usca", UCAL_MONDAY);
|
|
|
|
// Test -u-ca-iso8601
|
|
verifyFirstDayOfWeek("en-MV-u-ca-iso8601-sd-mv00", UCAL_MONDAY);
|
|
verifyFirstDayOfWeek("en-AE-u-ca-iso8601-sd-aeaj", UCAL_MONDAY);
|
|
verifyFirstDayOfWeek("en-US-u-ca-iso8601-sd-usca", UCAL_MONDAY);
|
|
|
|
// Test Region Tags only
|
|
verifyFirstDayOfWeek("en-MV", UCAL_FRIDAY);
|
|
verifyFirstDayOfWeek("en-AE", UCAL_MONDAY);
|
|
verifyFirstDayOfWeek("en-US", UCAL_SUNDAY);
|
|
verifyFirstDayOfWeek("dv-GB", UCAL_MONDAY);
|
|
|
|
// Test -u-sd-
|
|
verifyFirstDayOfWeek("en-u-sd-mv00", UCAL_FRIDAY);
|
|
verifyFirstDayOfWeek("en-u-sd-aeaj", UCAL_MONDAY);
|
|
verifyFirstDayOfWeek("en-u-sd-usca", UCAL_SUNDAY);
|
|
verifyFirstDayOfWeek("dv-u-sd-gbsct", UCAL_MONDAY);
|
|
|
|
// Test Add Likely Subtags algorithm produces a region
|
|
// dv => dv_Thaa_MV => Friday
|
|
verifyFirstDayOfWeek("dv", UCAL_FRIDAY);
|
|
// und_Thaa => dv_Thaa_MV => Friday
|
|
verifyFirstDayOfWeek("und-Thaa", UCAL_FRIDAY);
|
|
|
|
// ssh => ssh_Arab_AE => Saturday
|
|
verifyFirstDayOfWeek("ssh", UCAL_MONDAY);
|
|
|
|
// en => en_Latn_US => Sunday
|
|
verifyFirstDayOfWeek("en", UCAL_SUNDAY);
|
|
// und_Hira => ja_Hira_JP => Sunday
|
|
verifyFirstDayOfWeek("und-Hira", UCAL_SUNDAY);
|
|
|
|
verifyFirstDayOfWeek("zxx", UCAL_MONDAY);
|
|
}
|
|
|
|
void CalendarTest::Test22633ChineseOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=chinese"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->setTime(2043071457431218011677338081118001787485161156097100985923226601036925437809699842362992455895409920480414647512899096575018732258582416938813614617757317338664031880042592085084690242819214720523061081124318514531466365480449329351434046537728.000000, status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->set(UCAL_EXTENDED_YEAR, -1594662558);
|
|
cal->get(UCAL_YEAR, status);
|
|
assertTrue("Should return failure", U_FAILURE(status));
|
|
|
|
cal->setTime(17000065021099877464213620139773683829419175940649608600213244013003611130029599692535053209683880603725167923910423116397083334648012657787978113960494455603744210944.000000, status);
|
|
cal->add(UCAL_YEAR, 1935762034, status);
|
|
assertTrue("Should return falure", U_FAILURE(status));
|
|
|
|
status = U_ZERO_ERROR;
|
|
cal->set(UCAL_ERA, 1651667877);
|
|
cal->add(UCAL_YEAR, 1935762034, status);
|
|
assertTrue("Should return falure", U_FAILURE(status));
|
|
}
|
|
|
|
void CalendarTest::Test22962BuddhistOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=uddhist"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->clear();
|
|
cal->set(UCAL_WEEK_OF_YEAR, 1666136);
|
|
cal->set(UCAL_YEAR, -1887379272);
|
|
cal->fieldDifference(
|
|
261830011167902373443927125260580558779842815957727840993886210772873194951140935848493861585917165011373697198856398176256.000000,
|
|
UCAL_YEAR_WOY, status);
|
|
assertTrue("Should return falure", U_FAILURE(status));
|
|
}
|
|
|
|
void CalendarTest::Test22962ChineseOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=chinese"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->add(UCAL_DAY_OF_WEEK_IN_MONTH, 1661092210, status);
|
|
cal->add(UCAL_MINUTE, -1330638081, status);
|
|
cal->add(UCAL_MONTH, 643194, status);
|
|
assertTrue("Should return falure", U_FAILURE(status));
|
|
}
|
|
|
|
void CalendarTest::Test22633IndianOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=indian"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->roll(UCAL_EXTENDED_YEAR, -2120158417, status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
}
|
|
void CalendarTest::Test22633IslamicUmalquraOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=islamic-umalqura"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->roll(UCAL_YEAR, -134404585, status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
}
|
|
|
|
void CalendarTest::Test22633PersianOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=persian"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->add(UCAL_ORDINAL_MONTH, 1594095615, status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
|
|
cal->clear();
|
|
cal->fieldDifference(
|
|
-874417153152678003697180890506448687181167523704194267774844295805672585701302166100950793070884718009504322601688549650298776623158701367393457997817732662883592665106020013730689242515513560464852918376875667091108609655859551000798163265126400.000000,
|
|
UCAL_YEAR, status);
|
|
assertFalse("Should not return success", U_SUCCESS(status));
|
|
}
|
|
|
|
void CalendarTest::Test22633HebrewOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en@calendar=hebrew"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->clear();
|
|
cal->roll(UCAL_JULIAN_DAY, -335544321, status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
cal->roll(UCAL_JULIAN_DAY, -1812424430, status);
|
|
assertEquals("Should return U_ILLEGAL_ARGUMENT_ERROR",
|
|
U_ILLEGAL_ARGUMENT_ERROR, status);
|
|
}
|
|
|
|
void CalendarTest::Test22633AMPMOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(Locale("en"), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
cal->setTimeZone(*TimeZone::getGMT());
|
|
cal->clear();
|
|
// Test to set a value > limit should not cause internal overflow.
|
|
cal->set(UCAL_AM_PM, 370633137);
|
|
assertEquals("set large odd value for UCAL_AM_PM should be treated as PM",
|
|
12.0 * 60.0 * 60.0 *1000.0, cal->getTime(status));
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
|
|
cal->set(UCAL_AM_PM, 370633138);
|
|
assertEquals("set large even value for UCAL_AM_PM should be treated as AM",
|
|
0.0, cal->getTime(status));
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
}
|
|
|
|
void CalendarTest::RunTestOnCalendars(void(TestFunc)(Calendar*, UCalendarDateFields)) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale locale = Locale::getEnglish();
|
|
LocalPointer<StringEnumeration> values(
|
|
Calendar::getKeywordValuesForLocale("calendar", locale, false, status),
|
|
status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
if (U_FAILURE(status)) {
|
|
return;
|
|
}
|
|
const char* value = nullptr;
|
|
while ((value = values->next(nullptr, status)) != nullptr && U_SUCCESS(status)) {
|
|
locale.setKeywordValue("calendar", value, status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(*TimeZone::getGMT(), locale, status), status);
|
|
assertTrue("Should return success", U_SUCCESS(status));
|
|
for (int32_t i = 0; i < UCAL_FIELD_COUNT; i++) {
|
|
TestFunc(cal.getAlias(), static_cast<UCalendarDateFields>(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
// This test is designed to work with undefined behavior sanitizer UBSAN to
|
|
// ensure we do not have math operation overflow int32_t.
|
|
void CalendarTest::Test22633SetGetTimeOverflow() {
|
|
RunTestOnCalendars([](Calendar* cal, UCalendarDateFields field) {
|
|
auto f = [](Calendar* cal, UCalendarDateFields field, int32_t value) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->clear();
|
|
cal->set(field, value);
|
|
cal->getTime(status);
|
|
};
|
|
f(cal, field, INT32_MAX);
|
|
f(cal, field, INT32_MIN);
|
|
});
|
|
}
|
|
|
|
void CalendarTest::Test22633Set2FieldsGetTimeOverflow() {
|
|
RunTestOnCalendars([](Calendar* cal, UCalendarDateFields field) {
|
|
auto f = [](Calendar* cal, UCalendarDateFields field, int32_t value) {
|
|
for (int32_t j = 0; j < UCAL_FIELD_COUNT; j++) {
|
|
UCalendarDateFields field2 = static_cast<UCalendarDateFields>(j);
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->clear();
|
|
cal->set(field, value);
|
|
cal->set(field2, value);
|
|
cal->getTime(status);
|
|
}
|
|
};
|
|
f(cal, field, INT32_MAX);
|
|
f(cal, field, INT32_MIN);
|
|
});
|
|
}
|
|
|
|
void CalendarTest::Test22633SetAddGetTimeOverflow() {
|
|
RunTestOnCalendars([](Calendar* cal, UCalendarDateFields field) {
|
|
auto f = [](Calendar* cal, UCalendarDateFields field, int32_t value) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->clear();
|
|
cal->set(field, value);
|
|
cal->add(field, value, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->getTime(status);
|
|
};
|
|
f(cal, field, INT32_MAX);
|
|
f(cal, field, INT32_MIN);
|
|
});
|
|
}
|
|
|
|
void CalendarTest::Test22633AddTwiceGetTimeOverflow() {
|
|
RunTestOnCalendars([](Calendar* cal, UCalendarDateFields field) {
|
|
auto f = [](Calendar* cal, UCalendarDateFields field, int32_t value) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->clear();
|
|
cal->add(field, value, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->add(field, value, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->getTime(status);
|
|
};
|
|
f(cal, field, INT32_MAX);
|
|
f(cal, field, INT32_MIN);
|
|
});
|
|
}
|
|
|
|
void CalendarTest::Test22633SetRollGetTimeOverflow() {
|
|
RunTestOnCalendars([](Calendar* cal, UCalendarDateFields field) {
|
|
auto f = [](Calendar* cal, UCalendarDateFields field, int32_t value) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->clear();
|
|
cal->set(field, value);
|
|
cal->roll(field, value, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->getTime(status);
|
|
};
|
|
f(cal, field, INT32_MAX);
|
|
f(cal, field, INT32_MIN);
|
|
});
|
|
}
|
|
|
|
void CalendarTest::Test22633RollTwiceGetTimeOverflow() {
|
|
RunTestOnCalendars([](Calendar* cal, UCalendarDateFields field) {
|
|
auto f = [](Calendar* cal, UCalendarDateFields field, int32_t value) {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
cal->clear();
|
|
cal->roll(field, value, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->roll(field, value, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->getTime(status);
|
|
};
|
|
f(cal, field, INT32_MAX);
|
|
f(cal, field, INT32_MIN);
|
|
});
|
|
}
|
|
|
|
void CalendarTest::TestChineseCalendarComputeMonthStart() { // ICU-22639
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
|
|
// An extended year for which hasLeapMonthBetweenWinterSolstices is true.
|
|
constexpr int32_t eyear = 4643;
|
|
constexpr int64_t monthStart = 2453764;
|
|
|
|
LocalPointer<Calendar> calendar(
|
|
Calendar::createInstance(Locale("en_US@calendar=chinese"), status),
|
|
status);
|
|
if (failure(status, "Calendar::createInstance")) return;
|
|
|
|
// This test case is a friend of ChineseCalendar and may access internals.
|
|
const ChineseCalendar& chinese =
|
|
*dynamic_cast<ChineseCalendar*>(calendar.getAlias());
|
|
|
|
// The initial value of hasLeapMonthBetweenWinterSolstices should be false.
|
|
assertFalse("hasLeapMonthBetweenWinterSolstices [#1]",
|
|
chinese.hasLeapMonthBetweenWinterSolstices);
|
|
|
|
assertEquals("monthStart", monthStart,
|
|
chinese.handleComputeMonthStart(eyear, 0, false, status));
|
|
|
|
// Calling a const method must not haved changed the state of the object.
|
|
assertFalse("hasLeapMonthBetweenWinterSolstices [#2]",
|
|
chinese.hasLeapMonthBetweenWinterSolstices);
|
|
}
|
|
|
|
void CalendarTest::Test22633HebrewLargeNegativeDay() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> calendar(
|
|
Calendar::createInstance(Locale("en-u-ca-hebrew"), status),
|
|
status);
|
|
calendar->clear();
|
|
calendar->set(UCAL_DAY_OF_YEAR, -2147483648);
|
|
calendar->get(UCAL_HOUR, status);
|
|
assertEquals("status return without hang", status, U_ILLEGAL_ARGUMENT_ERROR);
|
|
}
|
|
void CalendarTest::Test23069HebrewHanukkah() {
|
|
// Based on Hanukkah data in
|
|
// https://en.wikipedia.org/wiki/Jewish_and_Israeli_holidays_2000%E2%80%932050
|
|
struct TestCase {
|
|
int32_t hebrewYear;
|
|
int32_t gregorianYear;
|
|
int32_t gregorianMonth;
|
|
int32_t gregorianDate;
|
|
} cases[] = {
|
|
{ 5760, 1999, UCAL_DECEMBER, 4},
|
|
{ 5761, 2000, UCAL_DECEMBER, 22},
|
|
{ 5762, 2001, UCAL_DECEMBER, 10},
|
|
{ 5763, 2002, UCAL_NOVEMBER, 30},
|
|
{ 5764, 2003, UCAL_DECEMBER, 20},
|
|
{ 5765, 2004, UCAL_DECEMBER, 8},
|
|
{ 5766, 2005, UCAL_DECEMBER, 26},
|
|
{ 5767, 2006, UCAL_DECEMBER, 16},
|
|
{ 5768, 2007, UCAL_DECEMBER, 5},
|
|
{ 5769, 2008, UCAL_DECEMBER, 22},
|
|
{ 5770, 2009, UCAL_DECEMBER, 12},
|
|
{ 5771, 2010, UCAL_DECEMBER, 2},
|
|
{ 5772, 2011, UCAL_DECEMBER, 21},
|
|
{ 5773, 2012, UCAL_DECEMBER, 9},
|
|
{ 5774, 2013, UCAL_NOVEMBER, 28},
|
|
{ 5775, 2014, UCAL_DECEMBER, 17},
|
|
{ 5776, 2015, UCAL_DECEMBER, 7},
|
|
{ 5777, 2016, UCAL_DECEMBER, 25},
|
|
{ 5778, 2017, UCAL_DECEMBER, 13},
|
|
{ 5779, 2018, UCAL_DECEMBER, 3},
|
|
{ 5780, 2019, UCAL_DECEMBER, 23},
|
|
{ 5781, 2020, UCAL_DECEMBER, 11},
|
|
{ 5782, 2021, UCAL_NOVEMBER, 29},
|
|
{ 5783, 2022, UCAL_DECEMBER, 19},
|
|
{ 5784, 2023, UCAL_DECEMBER, 8},
|
|
{ 5785, 2024, UCAL_DECEMBER, 26},
|
|
{ 5786, 2025, UCAL_DECEMBER, 15},
|
|
{ 5787, 2026, UCAL_DECEMBER, 5},
|
|
{ 5788, 2027, UCAL_DECEMBER, 25},
|
|
{ 5789, 2028, UCAL_DECEMBER, 13},
|
|
{ 5790, 2029, UCAL_DECEMBER, 2},
|
|
{ 5791, 2030, UCAL_DECEMBER, 21},
|
|
{ 5792, 2031, UCAL_DECEMBER, 10},
|
|
{ 5793, 2032, UCAL_NOVEMBER, 28},
|
|
{ 5794, 2033, UCAL_DECEMBER, 17},
|
|
{ 5795, 2034, UCAL_DECEMBER, 7},
|
|
{ 5796, 2035, UCAL_DECEMBER, 26},
|
|
{ 5797, 2036, UCAL_DECEMBER, 14},
|
|
{ 5798, 2037, UCAL_DECEMBER, 3},
|
|
{ 5799, 2038, UCAL_DECEMBER, 22},
|
|
{ 5800, 2039, UCAL_DECEMBER, 12},
|
|
{ 5801, 2040, UCAL_NOVEMBER, 30},
|
|
{ 5802, 2041, UCAL_DECEMBER, 18},
|
|
{ 5803, 2042, UCAL_DECEMBER, 8},
|
|
{ 5804, 2043, UCAL_DECEMBER, 27},
|
|
{ 5805, 2044, UCAL_DECEMBER, 15},
|
|
{ 5806, 2045, UCAL_DECEMBER, 4},
|
|
{ 5807, 2046, UCAL_DECEMBER, 24},
|
|
{ 5808, 2047, UCAL_DECEMBER, 13},
|
|
{ 5809, 2048, UCAL_NOVEMBER, 30},
|
|
{ 5810, 2049, UCAL_DECEMBER, 20},
|
|
{ 5811, 2050, UCAL_DECEMBER, 10},
|
|
};
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> hebrew(
|
|
Calendar::createInstance(Locale("en-u-ca-hebrew"), status),
|
|
status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
LocalPointer<GregorianCalendar> gregorian(
|
|
new GregorianCalendar(hebrew->getTimeZone(), status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
for (auto& cas : cases) {
|
|
hebrew->clear();
|
|
// Test Hebrew Calendar to Gregorian Calendar.
|
|
// Hanukkah is the 25th day of Kislev
|
|
hebrew->set(UCAL_YEAR, cas.hebrewYear);
|
|
hebrew->set(UCAL_MONTH, icu::HebrewCalendar::KISLEV);
|
|
hebrew->set(UCAL_DATE, 25);
|
|
gregorian->setTime(hebrew->getTime(status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
int32_t year = gregorian->get(UCAL_YEAR, status);
|
|
int32_t month = gregorian->get(UCAL_MONTH, status);
|
|
int32_t date = gregorian->get(UCAL_DATE, status);
|
|
assertEquals("Hebrew to Gregorian Calendar year", year, cas.gregorianYear);
|
|
assertEquals("Hebrew to Gregorian Calendar month", month, cas.gregorianMonth);
|
|
assertEquals("Hebrew to Gregorian Calendar date", date, cas.gregorianDate);
|
|
if (year != cas.gregorianYear || month != cas.gregorianMonth || date != cas.gregorianDate) {
|
|
printf("Hebrew year %d Gregorain Date(%d/%d/%d) but should be Date(%d/%d/%d)\n",
|
|
cas.hebrewYear, year, 1+month, date,
|
|
cas.gregorianYear, 1+cas.gregorianMonth, cas.gregorianDate);
|
|
}
|
|
// Test Gregorian Calendar to Hebrew Calendar.
|
|
gregorian->clear();
|
|
gregorian->set(UCAL_YEAR, cas.gregorianYear);
|
|
gregorian->set(UCAL_MONTH, cas.gregorianMonth);
|
|
gregorian->set(UCAL_DATE, cas.gregorianDate);
|
|
hebrew->setTime(gregorian->getTime(status), status);
|
|
U_ASSERT(U_SUCCESS(status));
|
|
year = hebrew->get(UCAL_YEAR, status);
|
|
month = hebrew->get(UCAL_MONTH, status);
|
|
date = hebrew->get(UCAL_DATE, status);
|
|
assertEquals("Gregorian to Hebrew Calendar year", year, cas.hebrewYear);
|
|
assertEquals("Gregorian to Hebrew Calendar month", month, icu::HebrewCalendar::KISLEV);
|
|
assertEquals("Gregorian to Hebrew Calendar date", date, 25);
|
|
if (year != cas.hebrewYear || month != icu::HebrewCalendar::KISLEV || date != 25) {
|
|
printf("Gregorian year %d Hebrew Date(%d/%d/%d) but should be Date(%d/%d/25)\n",
|
|
cas.gregorianYear, year, 1+month, date,
|
|
cas.hebrewYear, 1+icu::HebrewCalendar::KISLEV);
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::Test22730JapaneseOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> calendar(
|
|
Calendar::createInstance(Locale("en-u-ca-japanese"), status),
|
|
status);
|
|
calendar->clear();
|
|
calendar->roll(UCAL_EXTENDED_YEAR, -1946156856, status);
|
|
assertEquals("status return without overflow", status, U_ILLEGAL_ARGUMENT_ERROR);
|
|
}
|
|
|
|
void CalendarTest::Test22730CopticOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> calendar(
|
|
Calendar::createInstance(Locale("tn-BW-u-ca-coptic"), status),
|
|
status);
|
|
calendar->clear();
|
|
calendar->set(UCAL_JULIAN_DAY, -2147456654);
|
|
calendar->roll(UCAL_ORDINAL_MONTH, 6910543, status);
|
|
assertEquals("status return without overflow", status, U_ILLEGAL_ARGUMENT_ERROR);
|
|
}
|
|
|
|
void CalendarTest::Test22962ComputeJulianDayOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> calendar(
|
|
Calendar::createInstance(Locale("nds-NL-u-ca-islamic-umalqura"), status),
|
|
status);
|
|
calendar->clear();
|
|
calendar->set(UCAL_YEAR, -2147483648);
|
|
calendar->set(UCAL_WEEK_OF_YEAR, 33816240);
|
|
calendar->get(UCAL_ERA, status);
|
|
assertEquals("status return without overflow", status, U_ILLEGAL_ARGUMENT_ERROR);
|
|
}
|
|
void CalendarTest::TestAddOverflow() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
|
|
LocalPointer<Calendar> calendar(
|
|
Calendar::createInstance(Locale("en"), status),
|
|
status);
|
|
if (failure(status, "Calendar::createInstance")) return;
|
|
for (int32_t i = 0; i < UCAL_FIELD_COUNT; i++) {
|
|
status = U_ZERO_ERROR;
|
|
calendar->setTime(0, status);
|
|
calendar->add(static_cast<UCalendarDateFields>(i), INT32_MAX / 2, status);
|
|
calendar->add(static_cast<UCalendarDateFields>(i), INT32_MAX, status);
|
|
if ((i == UCAL_ERA) ||
|
|
(i == UCAL_YEAR) ||
|
|
(i == UCAL_YEAR_WOY) ||
|
|
(i == UCAL_EXTENDED_YEAR) ||
|
|
(i == UCAL_IS_LEAP_MONTH) ||
|
|
(i == UCAL_MONTH) ||
|
|
(i == UCAL_ORDINAL_MONTH) ||
|
|
(i == UCAL_ZONE_OFFSET) ||
|
|
(i == UCAL_DST_OFFSET)) {
|
|
assertTrue("add INT32_MAX should fail", U_FAILURE(status));
|
|
} else {
|
|
assertTrue("add INT32_MAX should still success", U_SUCCESS(status));
|
|
}
|
|
|
|
status = U_ZERO_ERROR;
|
|
calendar->setTime(0, status);
|
|
calendar->add(static_cast<UCalendarDateFields>(i), INT32_MIN / 2, status);
|
|
calendar->add(static_cast<UCalendarDateFields>(i), INT32_MIN, status);
|
|
if ((i == UCAL_YEAR) ||
|
|
(i == UCAL_YEAR_WOY) ||
|
|
(i == UCAL_EXTENDED_YEAR) ||
|
|
(i == UCAL_IS_LEAP_MONTH) ||
|
|
(i == UCAL_ZONE_OFFSET) ||
|
|
(i == UCAL_DST_OFFSET)) {
|
|
assertTrue("add INT32_MIN should fail", U_FAILURE(status));
|
|
} else {
|
|
assertTrue("add INT32_MIN should still success", U_SUCCESS(status));
|
|
}
|
|
}
|
|
}
|
|
|
|
void CalendarTest::Test22962MonthAddOneOverflow() {
|
|
Locale locale("ckb_IQ@calendar=ethiopic-amete-alem");
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
LocalPointer<Calendar> cal(Calendar::createInstance(
|
|
*TimeZone::getGMT(), locale, status));
|
|
cal->clear();
|
|
status = U_ZERO_ERROR;
|
|
cal->fieldDifference(
|
|
(double)115177509667085876226560460721710683457425563915331054206329829993967720136006086546037257220523631494518538798239249720325557586193565921621016454170342731307548672.0,
|
|
UCAL_MONTH, status);
|
|
status = U_ZERO_ERROR;
|
|
cal->set(UCAL_DAY_OF_WEEK_IN_MONTH , -2111799174);
|
|
cal->add(UCAL_ERA, -1426056846, status);
|
|
assertTrue("Should return failure", U_FAILURE(status));
|
|
}
|
|
void CalendarTest::Test22750Roll() {
|
|
UErrorCode status = U_ZERO_ERROR;
|
|
Locale l(Locale::getRoot());
|
|
std::unique_ptr<icu::StringEnumeration> enumeration(
|
|
Calendar::getKeywordValuesForLocale("calendar", l, false, status));
|
|
// Test every calendar
|
|
for (const char* name = enumeration->next(nullptr, status);
|
|
U_SUCCESS(status) && name != nullptr;
|
|
name = enumeration->next(nullptr, status)) {
|
|
UErrorCode status2 = U_ZERO_ERROR;
|
|
l.setKeywordValue("calendar", name, status2);
|
|
LocalPointer<Calendar> calendar(Calendar::createInstance(l, status2));
|
|
if (failure(status2, "Calendar::createInstance")) return;
|
|
calendar->add(UCAL_DAY_OF_WEEK_IN_MONTH, 538976288, status2);
|
|
calendar->roll(UCAL_DATE, 538976288, status2);
|
|
}
|
|
}
|
|
|
|
#endif /* #if !UCONFIG_NO_FORMATTING */
|
|
|
|
//eof
|